while迴圈中使用輸出語句停止死迴圈的原因
阿新 • • 發佈:2018-12-04
直接看程式碼
public class WhileTest { private boolean flag = true; public void setFlag(boolean flag) { this.flag = flag; } public void say() { while(flag) { } System.out.println("--------------執行緒停止------------------------------------"); } public static void main(String[] args) throws InterruptedException { final WhileTest wt = new WhileTest(); Thread t = new Thread(new Runnable() { @Override public void run() { wt.say(); } }); t.start(); Thread.currentThread().sleep(500); wt.setFlag(false); } }
很明顯,會出現死迴圈,因為主執行緒修改共享變數的值,另一個執行緒並且讀取到修改
常見的解決方案是
flag 加上volatile關鍵字,強制重新整理共享變數的值和主記憶體的值一致。
但是,如果在while迴圈體中加上一段輸出語句,也能夠停止執行緒,原因在哪裡,看下原始碼
public void say() { while(flag) { System.out.println(""); } System.out.println("--------------執行緒停止------------------------------------"); }
public void println(String x) {
synchronized (this) {
print(x);
newLine();
}
}
原來是因為,輸出語句的內容,有一個同步程式碼塊,進入、離開同步程式碼塊,都會和主記憶體的共享變數的值保證一致,從而實現了可見性。
其實,如果while迴圈內,加上Thread.sleep語句,給CPU一段時間,cpu會去同步主記憶體和工作記憶體的共享變數的值,也能夠停止死迴圈,不過不推薦這樣實現,存在不確定性。