synchronized和volatile的一些问题

来源:互联网 发布:枸杞 怎么泡水 知乎 编辑:程序博客网 时间:2024/04/29 22:09

关于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不能保证多线程安全






原创粉丝点击