synchronized和volatile的一些问题

本文探讨了Java多线程中的关键概念,包括内存可见性模型、原子性操作及synchronized与volatile的区别。通过实例说明volatile为何无法确保某些操作的多线程安全性。

关于java多线程的一些常见问题。

①java的内存可见性模型


故每个线程都有自己的工作内存。从主内存中读取X的值,改变X的值之后将最新的值写入到主内存之中,各个线程之间通过主内存进行交流。

②java中的内存可见性

内存可见性指的是比如线程一每次从主内存中读取最新的X的值。改变X后立马将X值刷新到主内存之中。

③原子性操作

比如X++;这一操作就不是原子性操作。它可以分解为三步操作。首先线程从主内存中读取X的值,然后在自己的工作内存中对X做出修改,最后将X的值刷新到主内存中。


synchronized和volatile的比较:

synchronized能够同时保证内存可见性和原子性操作。但是volatile只能保证内存可见性,也就是每次都是从内存中读取最新的数据,修改之后立马刷新到内存区。但是却不会保证原子性操作。所以volatile不能保证X++操作的多线程安全,但是synchronized却可以。

为什么volatilevo不能保证X++操作的多线程安全:


比如有两个线程A和B,同时X的初始值是5,接着执行了下面的操作。

①A 首先从主内存中读取到了X的值,这时它的CPU将执行权交给了B。故A只是读取了X的值,但是还没有做出修改。

②B从主内存中读取X的值,这时是5

③B执行X++操作,将B的值改为6

④B将X的值刷新到主内存中

⑤A获得CPU执行权,执行X++操作。(注意:执行X++操作之前在A的工作内存中X的值是5)将X变为6

⑥将X的最新值6刷新到主内存之中

所以经过以上步骤,理论上X应该是7,应为有两次++操作,但是由于volatile不能保证操作的原子性,所以导致结果不一定符合预期,也就是说volatile不能保证多线程安全






评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值