Java并发编程:内存模型
什么是内存模型,为什么需要它
由于时钟频率越来与难以提高,因此许多处理器制造厂商都开始转而生产多核处理器,因此能够提高的只有硬件并发性。
对于并发应用程序中的线程来说,它们在大部分时间里都执行各自的任务。只有当多个线程要共享数据时,才必须协调它们之间的操作。
- 平台的内存模型
- 在共享内存的多处理器体系架构中,每个处理器都拥有自己的缓存,并且定期与主内存进行协调。在不同的处理器架构中提供了不同级别的缓存一致性。
在架构定义的内存模型中将告诉应用程序可以从内存系统中获得怎样的保证,此外还定义了一些特殊的指令(称为内存栅栏或栅栏),当需要共享数据时,这些指令就能实现额外的存储协调保证。为了能使Java开发人员无须关心不同架构上内存模型之间的差异,Java还提供了自己的内存模型,并且JVM通过在适当的位置上插入内存栅栏来屏蔽JMM与地层平台内存模型之间的差异。
- 重排序
- 各种使操作延迟或者看似乱序执行的不同原因,都可以归为重排序。内存级的重排序会使程序的行为变得不可预测。
public class PossibleReordering { static int x = 0, y = 0; static int a = 0, b = 0; public static void main(String[] args) throws InterruptedException { Thread one = new Thread(new Runnable() { public void run() { a = 1; x = b; } }); Thread other = new Thread(new Runnable() { public void run() { b = 1; y = a; } }); one.start(); other.start(); one.join(); other.join(); System.out.println("(" + x + "," + y + ")"); } }很容易想象PossibleReordering是如何输出(1,0)或(0,1)或(1,1)的;线程A可以在线程B开始之前就执行完成,线程B也可以在线程A开始之前执行完成,或者二者的操作交替执行。但奇怪的是,PossibleReordering还可以输出(0,0)。由于每个线程中的各个操作之间不存在的数据流依赖性,因此这些操作操作可以乱序执行。

- Java内存模型
- Java内存模型是通过各种操作来定义的,包括对变量的读/写操作,监视器的加锁和释放操作,以及线程的启动和合并操作。JMM为程序中所有的操作定义了一个偏序关系,称为Happens-Before。
如果连个操作之间缺乏Happens-Before关系,那么JVM可以对它们任意地重排序。
当一个变量被多个线程读取并且至少一个线程写入时,如果在读操作和写操作之间没有依靠Happens-Before来排序,那么就会产生数据竞争问题。
Happens-Before规则包括:
程序顺序规则。如果程序中操作A在操作B之前,那么在线程中A操作将在B操作之前执行。
监视器锁规则。在监视器锁上的解锁操作必须在同一个监视器锁上的加锁操作之前执行。
volatile变量规则。对volatile变量的写入操作必须在该变量的读操作之前执行。
线程启动规则。在线程上对Thread.Start的调用必须在该线程中执行任何操作之前执行。
线程结束规则。线程中的任何操作都必须在其他线程检测到该线程已经结束之前执行,或者从Thread.join中成功返回,或者在调用Thread.isAlive时返回false。
中断规则。当一个线程在另一个线程上调用interrupt时,必须在被中断线程检测到interrupt调用之前执行(通过抛出InterruptedException,或者调用isInterrupted和interrupted)。
终结器规则。对象的构造函数必须在启动该对象的终结器之前执行完成。
传递性。如果操作A在操作B之前执行,并且操作B在操作C之前执行,那么操作A必须在操作C之前执行。发布
- 不安全发布
- 当缺少Happens-Before关系时,就可能出现重排序问题,这就解释了为什么在没有充分同步的情况下发布一个对象会导致另一个线程看到一个只被部分构造的对象。
如果无法确保发布共享引用的操作在另一个线程加载改共享引用之前执行,那么对新对象的写入操作将与对象中各个域的写入操作重排序。
@NotThreadSafe public class UnsafeLazyInitialization { private static Resources resource; public static Resource getInsatance() { if(resource == null) resource = new Resource(); // 不安全发布 return resource; } }线程A写入resource的操作与线程B读取resource的操作之前不存在Happen-Before关系。在发布对象时存在数据竞争问题,因此B并不一定能看到Resource的正确状态。
由于在两个线程中都没有使用同步,因此线程B看到的线程A中的操作顺序,可能与线程A执行这些操作时的顺序并不相同。因此,即使线程A初始化Resource实例之后在将resource设置为指向它,线程B仍可能看到对resource的写入操作将在对Resource各个域的写入操作之前发生。
- 安全的初始化模式
@ThreadSafe
public class SafeLazyInitialization {
private static Resources resource;
public synchronized static Resource getInsatance() {
if(resource == null)
resource = new Resource();
return resource;
}
}
This chapter requires login to view full content. You are viewing a preview.
Login to View Full Content