如何解决其中的 可见性 和有序性 导致的问题,这也就引出来了明天的配角——Java 内存模型。
一、什么是 Java 内存模型?
导致可见性的起因是缓存,导致有序性的起因是编译优化,那解决可见性、有序性最间接的方法就是 禁用缓存和编译优化,但这样尽管解决了问题,但也导致带来的性能优化都没了。
因而,解决方案是:提出一套规定和办法,是程序员能在该禁用的时候禁用,不该禁用的时候不禁用。
Java 内存模型标准就是来解决这个问题的 —— 提供按需禁用缓存和编译优化的办法
具体来说,这些办法包含 volatile、synchronized 和 final 三个关键字,以及六项 Happens-Before 规定,这也正是本期的重点内容。
二、Happens-Before 规定
Q:如何了解 Happens-Before 呢?
A:后面一个操作的后果对后续操作是可见的。但不能了解为前一个操作产生在后续操作的后面。
只有最终语义是对的,编译器怎么优化都行。
1、程序的程序性规定
这条规定是指在一个线程中,依照程序程序,后面的操作 Happens-Before 于后续的任意操作。
2、volatile 变量规定
这条规定是指对一个 volatile 变量的写操作,Happens-Before 于后续对这个 volatile 变量的读操作。
3、传递性
这条规定是指如果 A Happens-Before B,且 B Happens-Before C,那么 A Happens-Before C。
4、管程中锁的规定
这条规定是指对一个锁的解锁 Happens-Before 于后续对这个锁的加锁。
5、线程 start() 规定
这条是对于线程启动的。它是指主线程 A 启动子线程 B 后,子线程 B 可能看到主线程在启动子线程 B 前的操作。
6、线程 join() 规定
它是指主线程 A 期待子线程 B 实现(主线程 A 通过调用子线程 B 的 join() 办法实现),当子线程 B 实现后(主线程 A 中 join() 办法返回),主线程可能看到子线程的操作。当然所谓的“看到”,指的是对共享变量的操作。
纳闷
Q:volatile、synchronized 和 final 能了解是提供给程序员用的,六项 Happens-Before 规定是束缚谁的呢?
A:这是给程序员的保障,依照提供的规定写,就能保障 Happens-Before 的语义。
参考文章:
Java 内存模型以及 happens-before 规定