1. 程式人生 > >while迴圈中使用輸出語句停止死迴圈的原因

while迴圈中使用輸出語句停止死迴圈的原因

直接看程式碼

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會去同步主記憶體和工作記憶體的共享變數的值,也能夠停止死迴圈,不過不推薦這樣實現,存在不確定性。