Java內存模型中的工作內存和主內存既可能對應計算機硬件CPU寄存器、也可能時CPU緩存,還有可能時內存,是一種抽象概念劃分餘真實物理硬件的交叉
Java內存中線程的主內存和工作內存的交互是由Java虛擬機定義瞭如下的8種操作來完成的
lock(鎖定):作用於主內存的變量,一個變量在同一時間只能一個線程鎖定,該操作表示這條線成獨佔這個變量
unlock(解鎖):作用於主內存的變量,表示這個變量的狀態由處於鎖定狀態被釋放,這樣其他線程才能對該變量進行鎖定
read(讀取):作用於主內存變量,表示把一個主內存變量的值傳輸到線程的工作內存,以便隨後的load操作使用
load(載入):作用於線程的工作內存的變量,表示把read操作從主內存中讀取的變量的值放到工作內存的變量副本中(副本是相對於主內存的變量而言的)
use(使用):作用於線程的工作內存中的變量,表示把工作內存中的一個變量的值傳遞給執行引擎,每當虛擬機遇到一個需要使用變量的值的字節碼指令時就會執行該操作
assign(賦值):作用於線程的工作內存的變量,表示把執行引擎返回的結果賦值給工作內存中的變量,每當虛擬機遇到一個給變量賦值的字節碼指令時就會執行該操作
store(存儲):作用於線程的工作內存中的變量,把工作內存中的一個變量的值傳遞給主內存,以便隨後的write操作使用
write(寫入):作用於主內存的變量,把store操作從工作內存中得到的變量的值放入主內存的變量中
Java內存模型還規定了在執行上述八種基本操作時,必須滿足如下規則:
如果要把一個變量從主內存中複製到工作內存,就需要按順尋地執行read和load操作, 如果把變量從工作內存中同步回主內存中,就要按順序地執行store和write操作。但Java內存模型只要求上述操作必須按順序執行,而沒有保證必須是連續執行。
不允許read和load、store和write操作之一單獨出現
不允許一個線程丟棄它的最近assign的操作,即變量在工作內存中改變了之後必須同步到主內存中。
不允許一個線程無原因地(沒有發生過任何assign操作)把數據從工作內存同步回主內存中。
一個新的變量只能在主內存中誕生,不允許在工作內存中直接使用一個未被初始化(load或assign)的變量。即就是對一個變量實施use和store操作之前,必須先執行過了assign和load操作。
一個變量在同一時刻只允許一條線程對其進行lock操作,但lock操作可以被同一條線程重複執行多次,多次執行lock後,只有執行相同次數的unlock操作,變量纔會被解鎖。lock和unlock必須成對出現。
如果對一個變量執行lock操作,將會清空工作內存中此變量的值,在執行引擎使用這個變量前需要重新執行load或assign操作初始化變量的值
如果一個變量事先沒有被lock操作鎖定,則不允許對它執行unlock操作;也不允許去unlock一個被其他線程鎖定的變量。
對一個變量執行unlock操作之前,必須先把此變量同步到主內存中(執行store和write操作)。