java線程之Happens before規則案例詳解
正文
happens-before 規定瞭對共享變量的寫操作對其它線程的讀操作可見,它是可見性與有序性的一套規則總結,拋開以下 happens-before 規則,JMM 並不能保證一個線程對共享變量的寫,對於其它線程對該共享變量的讀可見.
案例1
線程解鎖 m 之前對變量的寫,對於接下來對 m 加鎖的其它線程對該變量的讀可見
static int x; static Object m = new Object(); new Thread(()->{ synchronized(m) { x = 10; } },"t1").start(); new Thread(()->{ synchronized(m) { System.out.println(x); } },"t2").start(); /* 運行結果: 10 */
案例2
線程對 volatile 變量的寫,對接下來其它線程對該變量的讀可見
volatile static int x; new Thread(()->{ x = 10; },"t1").start(); new Thread(()->{ System.out.println(x); },"t2").start(); /* 運行結果: 10 */
案例3
線程 start 前對變量的寫,對該線程開始後對該變量的讀可見
static int x; x = 10; new Thread(()->{ System.out.println(x); },"t2").start(); /* 運行結果: 10 */
案例4
線程結束前對變量的寫,對其它線程得知它結束後的讀可見(比如其它線程調用 t1.isAlive() 或 t1.join()等待 它結束)
static int x; Thread t1 = new Thread(()->{ x = 10; },"t1"); t1.start(); t1.join(); System.out.println(x); /* 運行結果: 10 */
案例5
線程 t1 打斷 t2(interrupt)前對變量的寫,對於其他線程得知 t2 被打斷後對變量的讀可見(通過 t2.interrupted 或 t2.isInterrupted)
static int x; public static void main(String[] args) { Thread t2 = new Thread(()->{ while(true) { if(Thread.currentThread().isInterrupted()) { System.out.println(x); break; } } },"t2"); t2.start(); new Thread(()->{ sleep(1); x = 10; t2.interrupt(); },"t1").start(); while(!t2.isInterrupted()) { Thread.yield(); } System.out.println(x); } /* 運行結果: 10 */
案例6
對變量默認值(0,false,null)的寫,對其它線程對該變量的讀可見
static int a; public static void main(String[] args) { new Thread(()->{ System.out.println(a); }).start(); } /* 運行結果: 0 */
案例7
具有傳遞性,如果 x hb-> y 並且 y hb-> z 那麼有 x hb-> z ,配合 volatile 的防指令重排,有下面的例子
volatile static int x; static int y; new Thread(()->{ y = 10; x = 20; },"t1").start(); new Thread(()->{ // x=20 對 t2 可見, 同時 y=10 也對 t2 可見 System.out.println(x); },"t2").start(); /* 運行結果: 20 */
以上就是java線程之Happens-before規則的詳細內容,更多關於線程Happens-before規則的資料請關註WalkonNet其它相關文章!
推薦閱讀:
- Java多線程Thread類的使用及註意事項
- 一篇文章掌握Java Thread的類及其常見方法
- Java線程中斷interrupt的常用方法
- java中斷機制實例講解
- 新手瞭解java 多線程基礎知識