作者简介:大家好,我是码炫码哥,前中兴通讯、美团架构师,现任某互联网公司CTO,兼职码炫课堂主讲源码系列专题
代表作:《jdk源码&多线程&高并发》,《深入tomcat源码解析》,《深入netty源码解析》,《深入dubbo源码解析》,《深入springboot源码解析》,《深入spring源码解析》,《深入redis源码解析》等
联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬。码炫课堂的个人空间-码炫码哥个人主页-面试,源码等
synchronized 可以在并发环境下保证原子性、可见性和有序性,而 volatile 只能保证可见性和有序,那为什么有了 synchronized 后还需要 volatile 呢?主要还是 synchronized 存在如下两个缺点:
- 1、性能损耗
虽然 synchronized 做了很多优化,例如自旋锁、适应性自旋、锁消除、锁粗化、偏向锁和轻量级锁等,但无论它做了多少优化,它毕竟还是一种锁,只要是锁,在获取锁和释放锁的过程中就一定存在性能损耗。
volatile 是一个比较轻量级的操作,只针对变量,volatile 变量的读操作的性能与普通变量几乎无差别,写操作由于需要插入内存屏障所以会慢一些,但即便如此,volatile在大多数场景下也比锁的开销要低。
- 2、产生阻塞
无论是同步方法(基于 ACC_SYNCHRONIZED)还是同步代码块(基于 monitorenter、monitorexit) 都是基于 Monitor 实现。当多个线程同时访问一段同步代码时,首先会进入 Entry Set,当有一个线程获取到对象的锁之后,才能进行 The Owner 区域,其他线程还会继续在 Entry Set 等待。并且当某个线程调用了 wait() 方法后,会释放锁并进入 Wait Set 等待;

所以,synchronized 的本质是一种阻塞锁,而 volatile 则是一种轻量级的同步机制,不涉及锁机制,它是基于内存屏障实现的,不会有阻塞锁带来的性能损耗和阻塞的问题;
630

被折叠的 条评论
为什么被折叠?



