亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

多線程 - 關于Java內存可見性的問題

瀏覽:92日期:2023-12-02 08:39:19

問題描述

請看以下代碼

public class TestVolatile {public static void main(String[] args) throws InterruptedException {ThreadDemo td = new ThreadDemo();new Thread(td).start();Thread.sleep(1);while(true){ if(td.isFlag()){System.out.println('------------------');break; }} }}class ThreadDemo implements Runnable { private boolean flag = false; @Override public void run() {try { Thread.sleep(200);} catch (InterruptedException e) {}flag = true;System.out.println('flag=' + isFlag()); } public boolean isFlag() {return flag; } public void setFlag(boolean flag) {this.flag = flag; }}

把Thread.sleep(1)換成Thread.sleep(1000)就能獲取flag修改后的值,即td.isFlag()返回true。雖然看了Java內存模型的概念,但我不知道如何解釋這段代碼,誰能解釋一下?

相關問題: Java多線程的工作內存是什么?

問題解答

回答1:

你得先說說你的預期效果是啥?問問題要問清楚啊

回答2:

這個期待是沒有規范支撐的。代碼中沒有做任何能保證 '子線程寫 happen-before 主線程讀' 的事情。

sleep(1000)后看到修改只是巧合,一個JVM如果在更久后才讓主線程看到,甚至永遠不讓主線程看到都不違反規范。

回答3:

你的程序應該是想測試 volatile 關鍵字的功能。但是 “把 Thread.sleep(1) 換成 Thread.sleep(1000) 就能獲得預期效果” 這樣做理解上是不對的。首先,程序中總共有兩個線程,主線程(暫稱 線程M)和 new Thread(td) (暫稱 線程T)。

當寫 Thread.sleep(1) 的時候,線程M 在 1ms 之后,便開始在 while(true) 循環中檢查 td.isFlag() 的值,但是因為內存可見性的關系,線程M 并不能及時讀取 線程T 中 flag 的值,所以此時導致了死循環;

當寫 Thread.sleep(1000) 的時候,M 在 1000ms 之后,開始在 while(true) 循環中檢查 td.isFlag() 的值;但是 T 在 200ms 的時候,便將 flag 的值設為 true 了,所以,M 在 1000ms 之后檢測 td.isFlag() 的值肯定是返回 true 的,那么第一次判斷便會返回 true,產生輸出并跳出 while(true) 循環。

為了讓 線程M 及時讀取到 線程T 中 flag 的值,需要將 flag 使用 volatile 關鍵字進行修飾:

private volatile boolean flag = false;

那么每次對 flag 的修改,其他線程都立馬可見。關于 volatile 的使用,可以參考我的博客:Java 多線程(6):volatile 關鍵字的使用

回答4:

可以參考如下三個代碼:其中第一個和你的情況一樣,由于多線程的可見性問題,可能導致無限循環下去。第二個是使用synchronized解決此問題,大多數工作場景用這個好第三個是使用volatile解決,但這個關鍵字只保證可見性,在實際場景中局限性比較大,得慎用

public class StopThread {private static boolean stopRequested;public static void main(String[] args) throws InterruptedException {Thread backgroundThread = new Thread(new Runnable() {@Override public void run() {@SuppressWarnings('unused')int i = 0;while(!stopRequested) {// System.out.println('加上這一句程序就可以終止,否則無限循環下去'); i++;} }});backgroundThread.start();TimeUnit.SECONDS.sleep(1);stopRequested = true; }}

public class StopThread2 {private static boolean stopRequested;public static synchronized boolean getStopRequested() {return stopRequested; }public static synchronized void requestStop() {stopRequested = true; }public static void main(String[] args) throws InterruptedException {Thread backgroundThread = new Thread(new Runnable() {@Override public void run() {@SuppressWarnings('unused')int i = 0;while(!getStopRequested()/* stopRequested */) { i++;} }});backgroundThread.start();TimeUnit.SECONDS.sleep(1);requestStop();/* stopRequested = true; */ }}

public class StopThread3 {private static volatile boolean stopRequested; public static void main(String[] args) throws InterruptedException {Thread backgroundThread = new Thread(new Runnable() {@Override public void run() {@SuppressWarnings('unused')int i = 0;while(stopRequested) { i++;} }});backgroundThread.start();TimeUnit.SECONDS.sleep(1);stopRequested = true; }}

標簽: java
相關文章:
主站蜘蛛池模板: 亲子乱子xxxxxx | 午夜国产精品影院在线观看 | 日韩高清一区 | 谁有免费黄色网址 | 天天草综合网 | 国产成人亚洲综合一区 | 成年大片免费播放视频人 | 久久99精品视香蕉蕉 | 亚洲成a人片在线播放观看国产 | 制服丝袜中文在线 | 午夜性影院 | 麻豆免费在线视频 | 亚洲欧洲日韩另类自拍 | 爱爱视频免费看 | 日本一级毛片视频无遮挡免费 | 日韩视频欧美视频 | 欧美a级黄色片 | 在线免费黄色网址 | 性视频播放免费视频 | 99热久久国产综合精品久久国产 | 99精品国产美女福到在线不卡 | 亚洲精品国产成人99久久 | 51国产午夜精品免费视频 | 影音先锋日韩资源 | 国产日韩欧美亚洲精品95 | 91国偷自产一区二区三区蜜臀 | 91在线免费公开视频 | 青青在线精品2018国产 | 国产综合色在线视频区色吧图片 | 91精品视频免费在线观看 | 91妖精视频 | 精精国产www视频在线观看免费 | 欧美精品在线一区二区三区 | 久久精品视频免费在线观看 | 视频在线观看黄 | 欧日韩一区二区三区 | 毛片啪啪啪 | 手机免费看黄在线高清视频 | 国产一级一级一级国产片 | 国产黄色毛片视频 | 久久综合九色综合97婷婷女人 |