有了synchronized为什么还需要volatile?

 作者简介:大家好,我是码炫码哥,前中兴通讯、美团架构师,现任某互联网公司CTO,兼职码炫课堂主讲源码系列专题


代表作:《jdk源码&多线程&高并发》,《深入tomcat源码解析》,《深入netty源码解析》,《深入dubbo源码解析》,《深入springboot源码解析》,《深入spring源码解析》,《深入redis源码解析》等


联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬。码炫课堂的个人空间-码炫码哥个人主页-面试,源码等

synchronized 可以在并发环境下保证原子性、可见性和有序性,而 volatile 只能保证可见性和有序,那为什么有了 synchronized 后还需要 volatile 呢?主要还是 synchronized 存在如下两个缺点:

  • 1、性能损耗

虽然 synchronized 做了很多优化,例如自旋锁、适应性自旋、锁消除、锁粗化、偏向锁和轻量级锁等,但无论它做了多少优化,它毕竟还是一种锁,只要是锁,在获取锁和释放锁的过程中就一定存在性能损耗。

volatile 是一个比较轻量级的操作,只针对变量,volatile 变量的读操作的性能与普通变量几乎无差别,写操作由于需要插入内存屏障所以会慢一些,但即便如此,volatile在大多数场景下也比锁的开销要低。

  • 2、产生阻塞

无论是同步方法(基于 ACC_SYNCHRONIZED)还是同步代码块(基于 monitorentermonitorexit) 都是基于 Monitor 实现。当多个线程同时访问一段同步代码时,首先会进入 Entry Set,当有一个线程获取到对象的锁之后,才能进行 The Owner 区域,其他线程还会继续在 Entry Set 等待。并且当某个线程调用了 wait() 方法后,会释放锁并进入 Wait Set 等待;

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值