as-if-seria和happens-beforel语义理解

在java程序运行的时候,编译器并不一定会根据java程序的顺序去执行,可能会依据优化策略对程序进行重排序,这就是指令重排,主要是提高了程序执行的并行度。

但是在做指令重排的前提下,不能违背程序原本的语义,即本来执行结果为10,不能因为指令重排了,结果变成了8,这就是严重的Bug了。

来看一下jvm在做指令重排序之前,对程序开发者做的保证。

as-if-serial

程序在单线程执行的时候,执行结果有时候与程序语句的顺序是有很大关联的。比如下面这个例子:

int r = 3; //1
int j = r-1; //2
int m = r*r; //3

程序能得到正确执行结果与语句的顺序相关联。所以指令重排序要保证单线程下java程序运行结果与语义的结果要一致,细节不用关心,可能内部也做了优化或者顺序的调整,这就是as-if-serial语义。

happends-before

在同一个线程或者多个线程之间,如果A线程的a操作 happends-before B线程的b操作,那么JMM向程序员保证a操作对b操作是可见的。

1、如果a操作happends-before 另一个b操作,那么a操作需要对b操作是可见的,并且a操作发生在b操作之前

2、JMM只是保证了happends-before语义的正确性,即使内部发生了指令重排序,那么我们也不用管,只要满足第一条,我们就认为他是符合happends-before语义的。换句话说,JMM这种优化就是合法的。

posted @ 2021-03-02 15:00  Java民工陆小凤  阅读(178)  评论(0编辑  收藏  举报