java平台类库包含了丰富的并发基础构建模块,如线程安全的容器类以及各种用于协调多个相互协作的线程控制流的同步工具类。

同步容器类

  同步容器类包括Vector和Hashtable,是早期JDK的一部分,此外还有Collections.synchronizedXXX等工厂方法创建的。这些类实现安全的方式是,将他们的状态封装起来,并对每个public方法进行同步,从而 使得每次只有一个线程能访问容器的状态。

  同步容器类都是线程安全的,但是对于某些复合操作需要额外的加锁来保护。常见复合操作有:迭代(反复访问元素,直到遍历所有元素)、跳转(根据指定顺序找到当期元素的下一个元素)以及条件运算(如:如没有则添加)。

public static Object getLast(Vector list){
int lastIndex = list.size() - 1;
return list.get(lastIndex)
} public static void deleteLast(Vector list){
int lastIndex = list.size() - 1;
list.remove(lastIndex);
}

  上面例子中,Vector中定义了两个方法,它们都执行先检查再运行操作。先获取数组大小,再获取或删除最后一个元素。这些方法看似没问题,并且都是线程安全的,也不破坏Vector。但是从调用者角度来看,就有问题了。可能A线程调用getLast的过程中,B线程调用了deleteLast,Vector元素减少,导致A线程调用失败。

  同步容器类遵守同步策略,即支持客户端加锁,因此只要我们知道应该使用那个锁,就能创建一些新的操作。这些新操作与容器与其他操作都是原子操作。同步容器通过自身的锁来保护它的每个方法。通过获取容器的锁,就能使上面的方法称为原子操作。size和get操作之间不会有其他操作。

public static Object getLast(Vector list){
synchronized(list){
int lastIndex = list.size() - 1;
return list.get(lastIndex)
} } public static void deleteLast(Vector list){
synchronized(list){
int lastIndex = list.size() - 1;
list.remove(lastIndex);
}
}

  同样的问题也会出现在遍历上,如下面的例子:

for(int i=0; i< vector.size(); i++)
doSomgthing(vector.get(i));

  如果另外一个线程删除一个元素,会导致ArrayIndexOutBoundsException异常。我们可以通过加锁来解决迭代不可靠问题,避免其他线程在遍历过程修改Vector。但也带来性能问题,迭代期间其他线程无法访问它。

synchronized(vector){
for(int i=0; i< vector.size(); i++)
doSomgthing(vector.get(i));
}

  上面的例子在未加锁的情况下都可能抛出异常,这并不意味着Vector不是线程安全的。Vector仍然是线程安全的,抛出的异常也与其规范保持一致。

迭代器与ConcurrentModificationException

  对容器进行迭代的标准方式是使用Iterator,使用for-each语法,也是调用Iterator。在设计同步容器类的时候并没有考虑并发修改问题,它们表现出的行为是“及时失败”的,意味着在迭代过程中,如果有其他线程修改容器,会抛出ConcurrentModificationException异常。它们实现的方式是,将计数器变化与容器关联起来,放迭代器件计数器被修改,那么hasNext或next将抛出异常。这是设计上的一个权衡。

  要想避免ConcurrentModificationException,就必须在迭代过程持有容器的锁。但是如果容器规模很大,迭代过程持有锁,将导致严重的性能问题。一种替代方式就是“克隆容器”,并在副本上迭代。克隆过程仍然需要加锁,同时存在显著的性能开销。克隆容器的好坏取决于过个元素,如容器大小,迭代时,每个元素执行的操作等。

隐藏迭代器

  加锁可以防止迭代抛出ConcurrentModificationException异常,但是需要在所有迭代的地方进行加锁。实际情况通常更加复杂,有些情况下可能会忽略隐藏的迭代器。

  

public class HiddenIterator{
private final Set(Integer) set = new HashSet<>(); 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" + set)
}
}

  addTenThings方法可能抛出ConcurrentModificationException异常,因为在打印输出的时候进行字符串连接,会调用set的toString方法,toString方法会对容器进行迭代。在使用println前必须获取HiddenIterator的锁,但是实际应用中可能忽略。

  封装对象的状态有助于维持不变性,封装对象的同步机制有助有确保实施同步策略。

  如果使用synchronizedSet来包装HashSet,并且对同步代码进行封装,就不会发生这种错误。

  除了toString,hashCode和equals等方法也会间接执行迭代操作。当容器作为另一个容器的元素和键值时,就会出现这种情况。同样,containsAll,removeAll等方法,以及把容器作为参数的构造函数都会对容器进行迭代。这些间接操作都有可能抛出ConcurrentModificationException异常。

java并发-同步容器类的更多相关文章

  1. Java并发-同步容器篇

    作者:汤圆 个人博客:javalover.cc 前言 官人们好啊,我是汤圆,今天给大家带来的是<Java并发-同步容器篇>,希望有所帮助,谢谢 文章如果有问题,欢迎大家批评指正,在此谢过啦 ...

  2. Java并发——同步容器与并发容器

    同步容器类 早期版本的JDK提供的同步容器类为Vector和Hashtable,JDK1.2 提供了Collections.synchronizedXxx等工程方法,将普通的容器继续包装.对每个共有方 ...

  3. Java并发--同步容器

    为了方便编写出线程安全的程序,Java里面提供了一些线程安全类和并发工具,比如:同步容器.并发容器.阻塞队列.Synchronizer(比如CountDownLatch).今天我们就来讨论下同步容器. ...

  4. Java并发—同步容器和并发容器

    简述同步容器与并发容器 在Java并发编程中,经常听到同步容器.并发容器之说,那什么是同步容器与并发容器呢?同步容器可以简单地理解为通过synchronized来实现同步的容器,比如Vector.Ha ...

  5. Java多线程——同步容器类

    1.同步容器类 同步容器类包括Vector和Hashtable,是早期JDK的一部分,这些类实现的方法是:将它们的状态封装起来,并对每个共有的方法进行同步,使得每个线程只有一个线程能访问它们. 1.1 ...

  6. Java并发——同步工具类

    CountDownLatch  同步倒数计数器 CountDownLatch是一个同步倒数计数器.CountDownLatch允许一个或多个线程等待其他线程完成操作. CountDownLatch对象 ...

  7. JAVA并发同步互斥实现方式总结

    大家都知道加锁是用来在并发情况防止同一个资源被多方抢占的有效手段,加锁其实就是同步互斥(或称独占)也行,即:同一时间不论有多少并发请求,只有一个能处理,其余要么排队等待,要么放弃执行.关于锁的实现网上 ...

  8. Java 并发同步工具(转)

    转自:https://www.jianshu.com/p/e80043ac4115 在 java 1.5 中,提供了一些非常有用的辅助类来帮助我们进行并发编程,比如 CountDownLatch,Cy ...

  9. Java 并发同步器之CountDownLatch、CyclicBarrier

    一.简介 1.CountDownLatch是一个同步计数器,构造时传入int参数,该参数就是计数器的初始值,每调用一次countDown()方法,计数器减1,计数器大于0 时,await()方法会阻塞 ...

随机推荐

  1. 【比赛】HNOI2018 总结

    一将功成万骨枯,我就是给那些队爷做基数的 一.比赛过程 Day1 看完题,暴力好打,然后就打 觉得第三题模型很好建啊,先看第三题吧(结果第三题是最..的) 图建出来,先看树的情况,设dp试一下 结果一 ...

  2. 【BZOJ4912】天才黑客(最短路,虚树)

    [BZOJ4912]天才黑客(最短路,虚树) 题面 BZOJ 洛谷 题解 \(Anson\)爷讲过的题目,然而我还是不会做 只有照着\(zsy\)的程序打我才会做....果然太弱了. 这道题目显然是把 ...

  3. 【loj6059】Sum

    Portal --> loj6059 Solution ​​ 看过去第一反应是..大力数位dp!然后看了一眼数据范围... ​ 但是这没有什么关系!注意到我们不需要考虑前导零了,可以直接快乐dp ...

  4. git使用笔记(十三)ls-files

    By francis_hao    Mar 18,2018   git ls-fles 显示index和工作区的文件的信息. 概要 git ls-files [-z] [-t] [-v]        ...

  5. html中的事件属性

    Window 事件属性 针对 window 对象触发的事件(应用到 <body> 标签): 属性 值 描述 onafterprint script 文档打印之后运行的脚本. onbefor ...

  6. 洛谷P3048 [USACO12FEB]牛的IDCow IDs

    P3048 [USACO12FEB]牛的IDCow IDs 12通过 67提交 题目提供者lin_toto 标签USACO2012 难度普及/提高- 时空限制1s / 128MB 提交  讨论  题解 ...

  7. Servlet3.0 新特性

    Servlet3.0 的注解 Servlet 允许开发人员采用注解的方式来配置 Servlet.Filter.Listener. Servlet3.0 规范在 javax.servlet.annota ...

  8. flask学习之解决Internal Server Error问题的方式之一

    最近在学习flask web development的时候,遇到了这么一个问题,就是照着书上敲的代码,跑起来是Internal server error,由于中途学的时候为了方便,改用pycharm来 ...

  9. MakeDown的使用

    Makedown的使用 之前有用博客园来写博客,但是因为它的界面不好看,所以中途就放弃了.后来也使用过"有道云笔记",发现其写的笔记的界面很简洁工整.有道云笔记的书写原理和Make ...

  10. bzoj 1367: [Baltic2004]sequence

    1367: [Baltic2004]sequence Time Limit: 20 Sec  Memory Limit: 64 MB Description Input Output 一个整数R Sa ...