volatile关键字原理
-
加了volatile关键字的变量在汇编层面的指令会加上Lock前缀
加了Lock会执行以下两步:1.将当前处理器缓存行的数据写回内存。2.这个写操作会使得其他CPU里面缓存了相应的数据失效
一般变量数据改变不会立即写回内存,只是更新缓存行的数据
加了volatile关键字的变量会将变量立即写回内存,但是其他cpu内的数据仍然不同步, 所以要利用缓存一致协议,处理器通过嗅探总线上的数据,如果发现数据不一致,就将缓存行内的数据设置为无效,当处理器使用这个变量时发现数据无效,就重新在内存里面读取。
-
缓存一致性会阻止同时修改被两个以上处理器缓存的内存区域数据
一个处理器的缓存写到内存,会使得其他处理器的缓存失效
-
volatile优化
LinkedTransferQueue(其中每个引用都是4字节,内部类PaddedAotmicReference就是追加字节至64)利用字节追加的方式使高速缓冲行的内容为64字节,不支持部分填充,在处理器修改头结点时, 因为缓存一致性协议,就会锁定整个缓存行,为了防止头结点尾节点同时被加载进缓存行,导致其他处理器无法修改尾节点, 所以要追加字节,填充缓存行,如果高速缓存行不是64字节,或者写的频率较低,就不用追加至64字节
-
Java实现原子操作方式
使用CAS 导致aba问题、cpu消耗大、每次只能有一个共享变量,可以使用合并变量的方式,或者使用AtomicReference保证对象之间的原子性,将多个变量放在这个对象里面。
使用锁机制,偏向锁,轻量级锁,互斥锁,除了偏向锁,其他锁都用了循环CAS 方式获得锁、释放锁。