原子性、可见性与有序性

原子性

原子性是指一个操作是不可中断的。即使是在多个线程一起执行的时候,一个操作一旦开始,就不会被其它线程干扰

由 Java 内存模型直接保证的原子性变量操作包括 read、load、assign、use、store 和 write,我们大致可以认为基本数据类型的访问读写是具备原子性的(例外就是 long 和 double 的非原子性协定)。

尽管虚拟机未把 lock 和 unlock 操作直接开放给用户使用,但是却提供了更高层次的字节码指令monitorenter 和 monitorexit 来隐式地使用这两个操作,这两个字节码指令反映到 Java 代码中就是同步块——synchronized 关键字,因此在 synchronized 块之间的操作也具备原子性。

可见性

可见性(Visibility):可见性是指当一个线程修改了共享变量的值,其他线程能够立即得知这个修改

除了 volatile 以外,Java 还有两个关键字能实现可见性,即 synchronized 和 final

  • 同步块的可见性是由「对一个变量执行 unlock 操作之前,必须先把此变量同步回主内存中(执行 store、write 操作)」这条规则获得的,
  • final 关键字的可见性是指:保证一个对象的构建方法结束前,所有 final 成员变量都必须完成初始化(前提是没有 this 引用溢出)。在构造器中一旦初始化完成,并且构造器没有把「this」的引用传递出去,那在其他线程中就能看见 final 字段的值。
    • this 引用逃逸 是指在构造函数返回之前其他线程就持有该对象的引用。调用尚未构造完全的对象的方法可能引发令人疑惑的错误, 因此应该避免 this 逃逸的发生。
      • 例子:在构造函数中启动线程 / 注册监听器/ 创建匿名内部类。

有序性

如果在本线程内观察,所有的操作都是有序的;如果在一个线程中观察另一个线程,所有的操作都是无序的。前半句是指「线程内表现为串行的语义」(Within-Thread As-If-Serial Semantics),后半句是指「指令重排序」现象和「工作内存与主内存同步延迟」现象。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,418评论 11 349
  • Java8张图 11、字符串不变性 12、equals()方法、hashCode()方法的区别 13、...
    Miley_MOJIE阅读 3,788评论 0 11
  • 柴米油盐酱醋茶,琴棋书画诗酒花
    52赫兹_4770阅读 280评论 0 2
  • ​终于闲下来回到家中,总结反思下整个2016。读研前的自己,到底是怎么活着?或许是看口红试色刷一下午微博,或许是自...
    糖不甩甩甩阅读 241评论 0 2
  • 那就等天空褪成蓝色 云里藏满歌
    小雅杰阅读 142评论 0 2