线程安全之Collections.synchronizedList

有些容器是线程安全的,比如:Vector, 而有些是不安全的,如:List。Collections.synchronizedList的作用是把本身不是线程安全的容器变成线程安全的,如:

public List<E> list = Collection.synchronizedList(new ArrayList<E>());

但是注意一点的是,所谓的线程安全仅仅指的是直接使用它提供的函数,如list.add(),list.get(i); 也就是说仅仅是原子操作的时候才是线程安全的。如果我们要使用非原子操作的时候,如:

public class ListHelper<E> {
   public List<E> list = Collection.synchronizedList(new ArrayList<E>());
    ...
   public synchronized boolean putIfAbsent(E X){
      boolean absent = !list.contains(x);
      if(absent) 
          list.add(x);
      return absent;
  } }

以上这种方式是不能保证线程安全的。为什么呢?问题在于list的操作并不是原子性操作。这里先进行contains操作,如果没有就进行调加。在多线程调用list的情况下。这里synchronized 锁住的对象是ListHelper,而不是list。list使用的是synchronizedList,就相当于 contains和add前都加上了synchronized,虽然都是同步方法,但是由于多线程的重排序性,无法保证执行的顺序。所以无法确保当putIfAbsent执行时另一个线程不会修改链表。 那么,如果来保证线程安全呢?可通过客户端加锁的方法:

   public class ListHelper<E> {
   public List<E> list = Collection.synchronizedList(new ArrayList<E>());
    ...
   public synchronized boolean putIfAbsent(E X){
      synchronize(list){

      boolean absent = !list.contains(x);
      if(absent) 
          list.add(x);
      return absent;
  } }}

通过客户端加锁的方法可以保证线程安全性,但是它会将类的加锁代码分布到其他的方法里面,也就是如果ListHelper扩展多一个方法,如果也想putIfAbsent一样的话也需要加锁synchronize(this){}。这样程序显得不够健壮。有一种更好的方法:组合。

  public class ListHelper<T> implements List<T> {
    private final List<T>  list;
    
   public ListHelper(List<T>  list) {this.list = list;}

   public synchronized boolean putIfAbsent(T X){
      boolean absent = !list.contains(x);
      if(absent) 
          list.add(x);
      return absent;
  } 
 //其它方法

 public synchronized void clear(){

     list.clear();
 }}

转载于:https://my.oschina.net/jianxiao/blog/175470

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值