Java并发:Java内存模型(三)

本文详细解析了Java中volatile关键字的内存语义,包括其可见性和原子性特性,以及如何通过内存屏障实现volatile写-读的happens-before关系,确保线程间的数据一致性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

接上一篇讲到了重排序顺序一致性的问题,这里继续

4. volatile的内存语义

       当声明共享变volatile后,对这量的/写将会很特了揭开volatile的神秘面纱,下面将介volatile的内存语义volatile内存语义实现

   4.1 volatile的特性

        理解volatile特性的一个好方法是把volatile量的/写,看成是使用同一个锁对这些单/写操作做了同步。下面通具体的示例来明,示例代如下。

           

有多个线程分别调用上面程序的3个方法,个程序在语义上和下面程序等价。

           

        如上面示例程序所示,一个volatile量的/写操作,与一个普通量的/写操作都是使用同一个锁来同步,它行效果相同。

        锁的happens-before规则证释的两个线程之的内存可性,意味着对一个volatile量的是能看到(任意线程)对这volatile量最后的写入。

        锁的语义决定了界区代行具有原子性。意味着,即使是64位的long型和double型变量,只要它是volatile量,对该变量的/写就具有原子性。如果是多个volatile操作或类似于volatile++种复合操作,些操作整体上不具有原子性。

        简而言之,volatile量自身具有下列特性。

·性。一个volatile量的是能看到(任意线程)对这volatile量最后的写入。

·原子性:任意volatile量的/写具有原子性,但似于volatile++种复合操作不具有原子性。

4.2 volatile写-读建立的happens-before关系

     请看下面使用volatile量的示例代

                 

       假设线Awriter()方法之后,线Breader()方法。根据happens-before规则个过程建立的happens-before关系可以分3

1)根据程序次序规则1 happens-before 2;3 happens-before 4

2)根据volatile规则2 happens-before 3

3)根据happens-before传递规则1 happens-before 4

上述happens-before关系的形化表形式如下。

                                           

        在上图中,每一个箭头链接的两个点,代表了一个happens-before关系。黑色箭表示程 序顺规则;橙色箭表示volatile规则色箭表示规则后提供的happens-before

        这A线程写一个volatile量后,B线同一个volatile量。A线程在写volatile量之前所有可见的共享量,在B线同一个volatile量后,将立即B线程可

4.3 volatile写-读的内存语义

        volatile写的内存语义是:

        当写一个volatileJMM会把该线对应的本地内存中的共享刷新到主内存。

        以上面示例程序VolatileExample例,假设线A首先writer()方法,随后线B行reader()方法,初始两个线程的本地内存中的flaga都是初始状3-17线A行volatile写后,共享量的状示意

                                 

        如3-17所示,线A在写flag量后,本地内存A中被线A更新的两个共享量的值被刷新到主内存中。此时,本地内存A和主内存中的共享量的是一致的。

      volatile读的内存语义如下:

      当一个volatileJMM会把该线对应的本地内存置无效。线程接下来将从主内存中读取共享量。

3-18为线B同一个volatile量后,共享量的状示意

                                                

        如图所示,在flag量后,本地内存B包含的被置无效。此线B从主内存中读取共享量。线B取操作将致本地内存B与主内存中的共享量的值变成一致。如果我们volatile写和volatile两个步骤综合起来看的,在读线B一个volatile变量后,写线A在写volatile量之前所有可的共享量的都将立即对读线B可见。

      下面volatile写和volatile的内存语义做个总结

  • 线A写一个volatile量,实质上是线A向接下来将对要读这volatile量的某个线出了(其共享量所做修改的)消息。
  • 线B一个volatile量,实质上是线B接收了之前某个线出的(在写volatile量之前共享量所做修改的)消息。
  • 线A写一个volatile量,随后线B读这volatile量,实质上是线A内存向线B送消息。

4.4 volatile内存语义的实现

       前文提到过重排序分为编译器重排序和理器重排序。实现volatile内存语义JMM会分别限制两种型的重排序型。表3-5JMM针对编译器制定的volatile重排序规则表。

     

       举例来说,第三行最后一个元格的意思是:在程序中,当第一个操作普通量的或写时,如果第二个操作volatile写,则编译器不能重排序两个操作。

      从表3-5我可以看出。

           ·当第二个操作是volatile,不管第一个操作是什么,都不能重排序。规则确保volatile写之前的操作不会被编译器重排序到volatile写之后。

           ·当第一个操作是volatile读时,不管第二个操作是什么,都不能重排序。规则确保volatile读之后的操作不会被编译器重排序到volatile之前。

           ·当第一个操作是volatile写,第二个操作是volatile读时,不能重排序。

        为了实现volatile的内存语义编译器在生成字节码时,会在指令序列中插入内存屏障来禁止特定类型的理器重排序。编译器来发现一个最布置来最小化插入屏障的总数几乎不可能。为此,JMM采取保守策略。下面是基于保守策略的JMM内存屏障插入策略。

     ·在每个volatile写操作的前面插入一个StoreStore屏障。

     ·在每个volatile写操作的后面插入一个StoreLoad屏障。

     ·在每个volatile操作的后面插入一个LoadLoad屏障。

     ·在每个volatile操作的后面插入一个LoadStore屏障。

上述内存屏障插入策略非常保守,但它可以保在任意理器平台,任意的程序中都能得到正确的volatile内存语义。下面是保守策略下,volatile写插入内存屏障后生成的指令序列示意,如3-19所示。

                          

 

        图3-19中的StoreStore屏障可以保volatile写之前,其前面的所有普通写操作已经对任意处理器可了。是因StoreStore屏障将保障上面所有的普通写在volatile写之前刷新到主内存。这里比较有意思的是,volatile写后面的StoreLoad屏障。此屏障的作用是避免volatile写与后面可能有的volatile/写操作重排序。因为编译器常常无法准确判断在一个volatile写的后面是否需要插入一个StoreLoad屏障(比如,一个volatile写之后方法立即return)。了保能正确实现volatile的内存语义JMM在采取了保守策略:在每个volatile写的后面,或者在每个volatile读的前面插入一个StoreLoad屏障。从整体行效率的角度考JMM终选择了在每个volatile写的后面插入一个StoreLoad屏障。因volatile-内存语义的常使用模式是:一个写线程写volatile量,多个读线同一个volatile量。当读线程的数量大大超线,选择在volatile写之后插入StoreLoad屏障将来可行效率的提升。从里可以看到JMM在实现上的一个特点:首先确保正确性,然后再去追求行效率。

      下面是在保守策略下,volatile插入内存屏障后生成的指令序列示意,如3-20所示。

                     

          图3-20中的LoadLoad屏障用来禁止理器把上面的volatile与下面的普通重排序。LoadStore屏障用来禁止理器把上面的volatile与下面的普通写重排序。上述volatile写和volatile的内存屏障插入策略非常保守。在实际执,只要不改变 volatile写-的内存语义编译器可以根据具体情况省略不必要的屏障。

 本文内容来自《java并发编程的艺术》一书。未完.......

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值