Java内存模型深度解读

Java内存模型规范了Java虚拟机与计算机内存是如何协同工做的。Java虚拟机是一个完整的计算机的一个模型,所以这个模型天然也包含一个内存模型——又称为Java内存模型。html

若是你想设计表现良好的并发程序,理解Java内存模型是很是重要的。Java内存模型规定了如何和什么时候能够看到由其余线程修改事后的共享变量的值,以及在必须时如何同步的访问共享变量。java

原始的Java内存模型存在一些不足,所以Java内存模型在Java1.5时被从新修订。这个版本的Java内存模型在Java8中人在使用。缓存

Java内存模型内部原理

Java内存模型把Java虚拟机内部划分为线程栈和堆。这张图演示了Java内存模型的逻辑视图。多线程

每个运行在Java虚拟机里的线程都拥有本身的线程栈。这个线程栈包含了这个线程调用的方法当前执行点相关的信息。一个线程仅能访问本身的线程栈。一个线程建立的本地变量对其它线程不可见,仅本身可见。即便两个线程执行一样的代码,这两个线程任然在在本身的线程栈中的代码来建立本地变量。所以,每一个线程拥有每一个本地变量的独有版本。架构

全部原始类型的本地变量都存放在线程栈上,所以对其它线程不可见。一个线程可能向另外一个线程传递一个原始类型变量的拷贝,可是它不能共享这个原始类型变量自身。并发

堆上包含在Java程序中建立的全部对象,不管是哪个对象建立的。这包括原始类型的对象版本。若是一个对象被建立而后赋值给一个局部变量,或者用来做为另外一个对象的成员变量,这个对象任然是存放在堆上。spa

下面这张图演示了调用栈和本地变量存放在线程栈上,对象存放在堆上。线程

一个本地变量多是原始类型,在这种状况下,它老是“呆在”线程栈上。设计

一个本地变量也多是指向一个对象的一个引用。在这种状况下,引用(这个本地变量)存放在线程栈上,可是对象自己存放在堆上。code

一个对象可能包含方法,这些方法可能包含本地变量。这些本地变量任然存放在线程栈上,即便这些方法所属的对象存放在堆上。

一个对象的成员变量可能随着这个对象自身存放在堆上。无论这个成员变量是原始类型仍是引用类型。

静态成员变量跟随着类定义一块儿也存放在堆上。

存放在堆上的对象能够被全部持有对这个对象引用的线程访问。当一个线程能够访问一个对象时,它也能够访问这个对象的成员变量。若是两个线程同时调用同一个对象上的同一个方法,它们将会都访问这个对象的成员变量,可是每个线程都拥有这个本地变量的私有拷贝。

下图演示了上面提到的点:

两个线程拥有一些列的本地变量。其中一个本地变量(Local Variable 2)执行堆上的一个共享对象(Object 3)。这两个线程分别拥有同一个对象的不一样引用。这些引用都是本地变量,所以存放在各自线程的线程栈上。这两个不一样的引用指向堆上同一个对象。

注意,这个共享对象(Object 3)持有Object2和Object4一个引用做为其成员变量(如图中Object3指向Object2和Object4的箭头)。经过在Object3中这些成员变量引用,这两个线程就能够访问Object2和Object4。

这张图也展现了指向堆上两个不一样对象的一个本地变量。在这种状况下,指向两个不一样对象的引用不是同一个对象。理论上,两个线程均可以访问Object1和Object5,若是两个线程都拥有两个对象的引用。可是在上图中,每个线程仅有一个引用指向两个对象其中之一。

所以,什么类型的Java代码会致使上面的内存图呢?以下所示:

public class MyRunnable implements Runnable() { public void run() { methodOne(); } public void methodOne() { int localVariable1 = 45; MySharedObject localVariable2 = MySharedObject.sharedInstance; //... do more with local variables.  methodTwo(); } public void methodTwo() { Integer localVariable1 = new Integer(99); //... do more with local variable.  } } public class MySharedObject { //static variable pointing to instance of MySharedObject public static final MySharedObject sharedInstance = new MySharedObject(); //member variables pointing to two objects on the heap public Integer object2 = new Integer(22); public Integer object4 = new Integer(44); public long member1 = 12345; public long member1 = 67890; }
 
 

若是两个线程同时执行run()方法,就会出现上图所示的情景。run()方法调用methodOne()方法,methodOne()调用methodTwo()方法。methodOne()声明了一个原始类型的本地变量和一个引用类型的本地变量。

每一个线程执行methodOne()都会在它们对应的线程栈上建立localVariable1localVariable2的私有拷贝。localVariable1变量彼此彻底独立,仅“生活”在每一个线程的线程栈上。一个线程看不到另外一个线程对它的localVariable1私有拷贝作出的修改。

每一个线程执行methodOne()时也将会建立它们各自的localVariable2拷贝。然而,两个localVariable2的不一样拷贝都指向堆上的同一个对象。代码中经过一个静态变量设置localVariable2指向一个对象引用。仅存在一个静态变量的一份拷贝,这份拷贝存放在堆上。所以,localVariable2的两份拷贝都指向由MySharedObject指向的静态变量的同一个实例。MySharedObject实例也存放在堆上。它对应于上图中的Object3。

注意,MySharedObject类也包含两个成员变量。这些成员变量随着这个对象存放在堆上。这两个成员变量指向另外两个Integer对象。这些Integer对象对应于上图中的Object2和Object4.

注意,methodTwo()建立一个名为localVariable的本地变量。这个成员变量是一个指向一个Integer对象的对象引用。这个方法设置localVariable1引用指向一个新的Integer实例。在执行methodTwo方法时,localVariable1引用将会在每一个线程中存放一份拷贝。这两个Integer对象实例化将会被存储堆上,可是每次执行这个方法时,这个方法都会建立一个新的Integer对象,两个线程执行这个方法将会建立两个不一样的Integer实例。methodTwo方法建立的Integer对象对应于上图中的Object1和Object5。

还有一点,MySharedObject类中的两个long类型的成员变量是原始类型的。由于,这些变量是成员变量,因此它们任然随着该对象存放在堆上,仅有本地变量存放在线程栈上。

硬件内存架构

现代硬件内存模型与Java内存模型有一些不一样。理解内存模型架构以及Java内存模型如何与它协同工做也是很是重要的。这部分描述了通用的硬件内存架构,下面的部分将会描述Java内存是如何与它“联手”工做的。

下面是现代计算机硬件架构的简单图示:

一个现代计算机一般由两个或者多个CPU。其中一些CPU还有多核。从这一点能够看出,在一个有两个或者多个CPU的现代计算机上同时运行多个线程是可能的。每一个CPU在某一时刻运行一个线程是没有问题的。这意味着,若是你的Java程序是多线程的,在你的Java程序中每一个CPU上一个线程可能同时(并发)执行。

每一个CPU都包含一系列的寄存器,它们是CPU内内存的基础。CPU在寄存器上执行操做的速度远大于在主存上执行的速度。这是由于CPU访问寄存器的速度远大于主存。

每一个CPU可能还有一个CPU缓存层。实际上,绝大多数的现代CPU都有必定大小的缓存层。CPU访问缓存层的速度快于访问主存的速度,但一般比访问内部寄存器的速度还要慢一点。一些CPU还有多层缓存,但这些对理解Java内存模型如何和内存交互不是那么重要。只要知道CPU中能够有一个缓存层就能够了。

一个计算机还包含一个主存。全部的CPU均可以访问主存。主存一般比CPU中的缓存大得多。

一般状况下,当一个CPU须要读取主存时,它会将主存的部分读到CPU缓存中。它甚至可能将缓存中的部份内容读到它的内部寄存器中,而后在寄存器中执行操做。当CPU须要将结果写回到主存中去时,它会将内部寄存器的值刷新到缓存中,而后在某个时间点将值刷新回主存。

当CPU须要在缓存层存放一些东西的时候,存放在缓存中的内容一般会被刷新回主存。CPU缓存能够在某一时刻将数据局部写到它的内存中,和在某一时刻局部刷新它的内存。它不会再某一时刻读/写整个缓存。一般,在一个被称做“cache lines”的更小的内存块中缓存被更新。一个或者多个缓存行可能被读到缓存,一个或者多个缓存行可能再被刷新回主存。

Java内存模型和硬件内存架构之间的桥接

上面已经提到,Java内存模型与硬件内存架构之间存在差别。硬件内存架构没有区分线程栈和堆。对于硬件,全部的线程栈和堆都分布在主内中。部分线程栈和堆可能有时候会出如今CPU缓存中和CPU内部的寄存器中。以下图所示:

当对象和变量被存放在计算机中各类不一样的内存区域中时,就可能会出现一些具体的问题。主要包括以下两个方面:

  • 线程对共享变量修改的可见性
  • 当读,写和检查共享变量时出现race conditions

下面咱们专门来解释如下这两个问题。

共享对象可见性

若是两个或者更多的线程在没有正确的使用Volatile声明或者同步的状况下共享一个对象,一个线程更新这个共享对象可能对其它线程来讲是不接见的。

想象一下,共享对象被初始化在主存中。跑在CPU上的一个线程将这个共享对象读到CPU缓存中。而后修改了这个对象。只要CPU缓存没有被刷新会主存,对象修改后的版本对跑在其它CPU上的线程都是不可见的。这种方式可能致使每一个线程拥有这个共享对象的私有拷贝,每一个拷贝停留在不一样的CPU缓存中。

下图示意了这种情形。跑在左边CPU的线程拷贝这个共享对象到它的CPU缓存中,而后将count变量的值修改成2。这个修改对跑在右边CPU上的其它线程是不可见的,由于修改后的count的值尚未被刷新回主存中去。

解决这个问题你可使用Java中的volatile关键字。volatile关键字能够保证直接从主存中读取一个变量,若是这个变量被修改后,老是会被写回到主存中去。

Race Conditions

若是两个或者更多的线程共享一个对象,多个线程在这个共享对象上更新变量,就有可能发生race conditions

想象一下,若是线程A读一个共享对象的变量count到它的CPU缓存中。再想象一下,线程B也作了一样的事情,可是往一个不一样的CPU缓存中。如今线程A将count加1,线程B也作了一样的事情。如今count已经被增在了两个,每一个CPU缓存中一次。

若是这些增长操做被顺序的执行,变量count应该被增长两次,而后原值+2被写回到主存中去。

然而,两次增长都是在没有适当的同步下并发执行的。不管是线程A仍是线程B将count修改后的版本写回到主存中取,修改后的值仅会被原值大1,尽管增长了两次。

下图演示了上面描述的状况:

解决这个问题可使用Java同步块。一个同步块能够保证在同一时刻仅有一个线程能够进入代码的临界区。同步块还能够保证代码块中全部被访问的变量将会从主存中读入,当线程退出同步代码块时,全部被更新的变量都会被刷新回主存中去,无论这个变量是否被声明为volatile

相关文章
相关标签/搜索