死磕并发之volatile关键字详解

  • volatile关键字可以说是java虚拟机提供的最轻量级的同步机制。

当一个变量定义为volatile之后,它将具备两种特性:

  • 保证此变量对所有线程的可见性。“可见性”是指当一条线程修改了这个变量的值,新值对于其他线程来说是可以立即得知的。
    普通变量和volatile关键字修饰的变量都是依赖主内存作为传递媒介的方式实现可见性的。唯一的区别就是: volatile修饰的变量保证每次修改后都能立即同步到主内存中,每次使用时都必须从主内存中刷新,而普通变量则不能保证。
  • 禁止指令重排序优化。普通变量仅仅会保证在该方法执行过程中所有依赖赋值结果的地方都能够获取到正确的结果,而不能保证变量赋值操作顺序与程序代码中的执行顺序一致。因为一个线程的方法执行过程中无法感知到这点,这也就是java内存模型中描述的所谓的“线程内表现为串行的语义”。

例如下面这几行代码,看上去没什么问题,线程B循环等待上下文加载,线程A加载上下文,成功后修改状态。但是如果发生了指令重排序,线程A中初始化和修改状态发生了指令重排序,导致上下文还未初始化完成而线程B已经跳出循环开始执行,这时就会报错。

  • 这里java代码的重排只是为了方便理解,真正的指令重排是在字节码指令层面。
boolean contextReady = false;

//在线程A中执行:
context = loadContext();
contextReady = true;


在线程B中执行:
while( ! contextReady ){ 
   sleep(200);
}
doAfterContextReady (context);
  • volatile关键字并不能保证原子性,在不符合以下两条规则的场景中,还是用通过加锁来保证原子性。
  • 运算结果并不依赖变量的当前值,或者能够确保只有单一的线程修改变量的值。
  • 变量不需要与其他的状态变量共同参与不变约束。
  • 性能方面

volatile变量读操作性能消耗与普通变量几乎没什么区别,但是写操作可能会慢一些,因为它需要在本地代码中插入许多内存屏障指令来保证处理器不发生乱序执行。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值