java 并发编程实践--客户端加锁机制。

在看Java Concurrency programm in practice.的客户端编程时。指出同步需要注意是不是同一个锁。书上给出线程安全的方法。

程序清单4-15,P61

public class ListHelper<E> {
    public List<E> list = Collections.synchroinedList(new ArrayList<E>());
    
    public synchronized boolean putIfAbsent(E e) {
        boolean absent = !list.contains(e);
        if (absent) {
            list.add(e)
        }
        
        return absent;
 }

根据我的理解ListHelper的putIfAbsent方法的锁是加在其实例对象上。看了Collections源码,synchronizedList 使用的是同步块,是SynchronizedList 内部类的 final Object mutex 上,为什么是线程安全的?难道是,mutex 是其成员变量,所以会排斥使用成员变量的实现同步?。请给位大神解答, 谢谢。 可以讲一下原理吗?

阅读 4.5k
4 个回答

Java中的任何对象都可以作为锁,这是 synchronized 锁机制的基础,有以下三种情况:

  1. 对于普通同步方法,锁是当前实例对象;
  2. 静态同步方法,锁是当前类的 Class 对象;
  3. 对于同步代码块,锁是 synchonized 括号里配置的对象。

你给出的ListHelper的例子是第一种情况,锁是ListHelper的实例对象,而SynchronizedList是第三种情况,锁是mutex变量,不管是哪种情况,线程都必须在获得锁之后才能继续执行,这就保证同一时刻只能有一个线程执行该操作,也就实现了线程安全。

补充 2 这种情况,synchronized(class) 也是锁当前类 class

我知道了,synchronizedList在构造器中将mutex = this.

新手上路,请多包涵

由于list是一个public域,所以在执行putIfAbsent可以有别的线程操作list,并且putIfAbsent方法不是一个原子性操作,所以这不是一个线程安全的类型。

具体修改方法有:
1.私有化list域,这样对list进行操作的线程一定是当前putIfAbsent线程。
2.使用客户端加锁,对该方法和list使用相同的锁,由于synchronized(list)就是使用list的内部锁,所以可以实现只有一个线程能操作list。
3.修改putIfAbsent方法为原子操作,不满足题意故去除。

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题