JAVA并发编程实战---隐藏迭代器

读《JAVA并发编程实战》笔记

加锁可以防止迭代器抛出ConcurrentModificationException,必须要记住在所有对共享容器进行迭代的地方都需要加锁。实际情况要更加复杂,因为在某些情况下,迭代器会隐藏起来,如下代码:

public class HiddenIterator {

	private final Set<Integer> set = new HashSet<Integer>();
	
	public synchronized void add(Integer i) {
		set.add(i);
	}
	
	public synchronized void remove(Integer i) {
		set.remove(i);
	}
	
	public void addTenThings() {
		Random r = new Random();
		for (int i=0; i < 10 ; i++) {
			add(r.nextInt());
		}
		System.out.println("DEBUG: added ten elements to " + set);
	}
}

在HiddenIterator中没有显示的迭代操作,但在如下代码中将执行迭代操作

System.out.println("DEBUG: added ten elements to " + set);

 编译器将字符串的连接操作转换为StringBuilder.append(Object),而这个方法又会调用容器的toString方法

AbstractCollection.toString()方法

public String toString() {
        Iterator<E> it = iterator();
        if (! it.hasNext())
            return "[]";

        StringBuilder sb = new StringBuilder();
        sb.append('[');
        for (;;) {
            E e = it.next();
            sb.append(e == this ? "(this Collection)" : e);
            if (! it.hasNext())
                return sb.append(']').toString();
            sb.append(',').append(' ');
        }
    }

标准容器的toString方法将迭代容器,并在每个元素上调用toString来生成容器内容的格式化表示,addTenThings方法可能会抛出ConcurrentModificationException,因为在生成调试消息的过程中,toString对容器进行迭代。

真正的问题在于HiddenIterator不是线程安全的,两种解决办法

  • 在使用println中的set之前必须首先获取HiddenIterator的锁
  • HiddenIterator用synchronizedSet来包装HashSet,并且对同步代码进行包装

 

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值