ConcurrentMap与CopyOnWrite容器

说到ConcurrentMap,一定要先说HashMap和HashTable,看面试题我们也会背了,什么HashMap线程不安全,HashTable线程安全,HashMap效率高,HashTable效率低.那么为什么HashTable效率低,线程安全?

上HashTable的put方法.↓↓↓↓↓

    public synchronized V put(K key, V value) {
       ..........
    }

上面一看就知道,HashTable的put方法用了synchronized来保证线程安全,不仅仅是put,还包括get,contains等等,自己去看jdk源码~~

那么这种采用同步的方式,显然高并发的场景下效率极其低下,所以在jdk5的时候,出了一个牛逼的工具包concurent,利用ConcurrentMap.ConcurrentMap有两个重要的实现:
ConcurrentHashMap和ConcurrentSkipListMap(支持并发排序,弥补ConcurrentHashMap).ConcurrentHashMap内部使用段来表示这些不同的部分,每个段其实就是一个HashTable,他们有自己的锁.只要多个修改操作发生在不同的段上,它们就可以并发进行.把一个整体分成了16个段.也就是最高支持16个线程的并发修改操作.这也是在多线程场景时减小锁的粒度从而降低锁竞争的一种方案.并且代码中大多共享变量使用volatile关键字声明,目的是第一时间获取修改的内容,性能非常的好.

当然啦,如果有两个线程同时操作同一个段,那么肯定是会被阻塞着~具体实现看源码吧还是挺复杂的~

然后是CopyOnWrite容器,和上面一样,对应着也有线程安全的Vector,集合中的线程安全类.看jdk源码↓↓↓↓↓

    public synchronized E get(int index) {
        ..............
    }

Vector的get方法,add方法等等也是加了synchronized修饰来保证线程安全.那么CopyOnWrite是如何实现的呢?

(简书不能画图- -,只能手写思想,不懂就多读几遍..)

CopyOnWrite容器是当我们往一个容器中添加元素的时候,不直接往当前容器添加,而是先将当前容器进行copy,复制出新的容器,然后新的容器里添加元素,添加完元素之后,再将原容器的引用指向新的容器.这样做的好处是我们可以对CopyOnWrite容器进行并发的读,而不需要加锁,因为当前容器不会添加任何元素,所以CopyOnWrite容器也是一种读写分离的思想,读和写不同的容器.(面试的时候问,直接说最后一句话就可以了~~~~)

CopyOnWrite也有两种实现:CopyOnWriteArrayList和CopyOnWriteArraySet

来看看CopyOnWriteArrayList底层add实现.

    public boolean add(E e) {
        final ReentrantLock lock = this.lock;
        lock.lock();
        try {
            Object[] elements = getArray();
            int len = elements.length;
            Object[] newElements = Arrays.copyOf(elements, len + 1);
            newElements[len] = e;
            setArray(newElements);
            return true;
        } finally {
            lock.unlock();
        }
    }

ReentrantLock是重入锁,是为了保证写安全,所以不会出现多个写的问题!为什么不用ReentrantReadWriteLock读写锁,因为ReentrantLock效率高啊,只能这么想,也不知道写jdk源码的人怎么想.至于这个锁,将来讲~~

关于CopyOnWrite有一点要说的是,使用CopyOnWrite适合读多写少的场景,你想,如果容器有1000个元素去copy那代价太大了还不如用同步,是吧

有啥不懂的请加qq727865942,微信号 cto_zej,觉得是干货请打赏~~~~~~~~~~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,429评论 11 349
  • Java8张图 11、字符串不变性 12、equals()方法、hashCode()方法的区别 13、...
    Miley_MOJIE阅读 9,119评论 0 11
  • 锁 锁是用来控制多个线程访问共享资源的方式,一般来说,一个锁能够防止多个线程同时访问共享资源(但是有些锁可以允许多...
    黄俊彬阅读 5,323评论 1 15
  • 人一生中要完成许多事情,事情有分大小,每一件事想要做好都不是那么容易的。有的人却可以出类拔萃,能够在自己的领域做到...
    吉日良辰阅读 3,325评论 4 4
  • 秀丽笔写出来的感觉,像是毛笔,但还是和毛笔不同。 这个印……刻出来的,鹤不像鹤鸡不像鸡。 用木刻刀琢磨一下午。手边...
    不吃炒蛋脱你胖次阅读 2,669评论 0 1