一個 println 竟然比 volatile 還好使?

来源:https://www.cnblogs.com/Jcloud/archive/2023/09/25/17727299.html
-Advertisement-
Play Games

前兩天一個小伙伴突然找我求助,說準備換個坑,最近在系統複習多線程知識,但遇到了一個刷新認知的問題…… ...


前兩天一個小伙伴突然找我求助,說準備換個坑,最近在系統複習多線程知識,但遇到了一個刷新認知的問題……

小伙伴:Effective JAVA 里的併發章節里,有一段關於可見性的描述。下麵這段代碼會出現死迴圈,這個我能理解,JMM 記憶體模型嘛,JMM 不保證 stopRequested 的修改能被及時的觀測到。

static boolean stopRequested = false;

public static void main(String[] args) throws InterruptedException {

    Thread backgroundThread = new Thread(() -> {
        int i = 0;
        while (!stopRequested) {
            i++;
        }
    }) ;
    backgroundThread.start();
    TimeUnit.MICROSECONDS.sleep(10);
    stopRequested = true ;
}



但奇怪的是在我加了一行列印之後,就不會出現死迴圈了!難道我一行 println 能比 volatile 還好使啊?這倆也沒關係啊

static boolean stopRequested = false;

public static void main(String[] args) throws InterruptedException {

    Thread backgroundThread = new Thread(() -> {
        int i = 0;
        while (!stopRequested) {
            
            // 加上一行列印,迴圈就能退出了!
        	System.out.println(i++);
        }
    }) ;
    backgroundThread.start();
    TimeUnit.MICROSECONDS.sleep(10);
    stopRequested = true ;
}



我:小伙子八股文背的挺熟啊,JMM 張口就來。

我:這個……其實是 JIT 乾的好事,導致你的迴圈無法退出。JMM 只是一個邏輯上的記憶體模型規範,JIT可以根據JMM的規範來進行優化。

比如你第一個例子里,你用-Xint禁用 JIT,就可以退出死迴圈了,不信你試試?

小伙伴:WK,真的可以,加上 -Xint 迴圈就退出了,好神奇!JIT 是個啥啊?還能有這種功效?

image.png

JIT(Just-in-Time) 的優化

眾所周知,JAVA 為了實現跨平臺,增加了一層 JVM,不同平臺的 JVM 負責解釋執行位元組碼文件。雖然有一層解釋會影響效率,但好處是跨平臺,位元組碼文件是平臺無關的。
image.png
在 JAVA 1.2 之後,增加了即時編譯(Just-in-Time Compilation,簡稱 JIT)的機制,在運行時可以將執行次數較多的熱點代碼編譯為機器碼,這樣就不需要 JVM 再解釋一遍了,可以直接執行,增加運行效率。
image.png

但 JIT 編譯器在編譯位元組碼時,可不僅僅是簡單的直接將位元組碼翻譯成機器碼,它在編譯的同時還會做很多優化,比如迴圈展開、方法內聯等等……

這個問題出現的原因,就是因為 JIT 編譯器的優化技術之一 -表達式提升(expression hoisting)導致的。

表達式提升(expression hoisting)

先來看個例子,在這個hoisting方法中,for 迴圈里每次都會定義一個變數y,然後通過將 x*y 的結果存儲在一個 result 變數中,然後使用這個變數進行各種操作

public void hoisting(int x) {
	for (int i = 0; i < 1000; i = i + 1) {
		// 迴圈不變的計算 
		int y = 654;
		int result = x * y;
		
		// ...... 基於這個 result 變數的各種操作
	}
}



但是這個例子里,result 的結果是固定的,並不會跟著迴圈而更新。所以完全可以將 result 的計算提取到迴圈之外,這樣就不用每次計算了。JIT 分析後會對這段代碼進行優化,進行表達式提升的操作:

public void hoisting(int x) {
	int y = 654;
	int result = x * y;
    
	for (int i = 0; i < 1000; i = i + 1) {	
		// ...... 基於這個 result 變數的各種操作
	}
}



這樣一來,result 不用每次計算了,而且也完全不影響執行結果,大大提升了執行效率。

註意,編譯器更喜歡局部變數,而不是靜態變數或者成員變數;因為靜態變數是“逃逸在外的”,多個線程都可以訪問到,而局部變數是線程私有的,不會被其他線程訪問和修改。

編譯器在處理靜態變數/成員變數時,會比較保守,不會輕易優化。

像你問題里的這個例子中,stopRequested就是個靜態變數,編譯器本不應該對其進行優化處理;

static boolean stopRequested = false;// 靜態變數

public static void main(String[] args) throws InterruptedException {

    Thread backgroundThread = new Thread(() -> {
        int i = 0;
        while (!stopRequested) {
			// leaf method
            i++;
        }
    }) ;
    backgroundThread.start();
    TimeUnit.MICROSECONDS.sleep(10);
    stopRequested = true ;
}



但由於你這個迴圈是個leaf method,即沒有調用任何方法,所以在迴圈之中不會有其他線程會觀測到stopRequested值的變化。那麼編譯器就冒進的進行了表達式提升的操作,將stopRequested提升到表達式之外,作為迴圈不變數(loop invariant)處理:

int i = 0;

boolean hoistedStopRequested = stopRequested;// 將stopRequested 提升為局部變數
while (!hoistedStopRequested) {    
	i++;
}



這樣一來,最後將stopRequested賦值為 true 的操作,影響不了提升的hoistedStopRequested的值,自然就無法影響迴圈的執行了,最終導致無法退出。

至於你增加了println之後,迴圈就可以退出的問題。是因為你這行 println 代碼影響了編譯器的優化。println 方法由於最終會調用 FileOutputStream.writeBytes 這個 native 方法,所以無法被內聯優化(inling)。而未被內斂的方法調用從編譯器的角度看是一個“full memory kill”,也就是說副作用不明、必須對記憶體的讀寫操作做保守處理

在這個例子里,下一輪迴圈的stopRequested讀取操作按順序要發生在上一輪迴圈的 println 之後。這裡“保守處理”為:就算上一輪我已經讀取了stopRequested的值,由於經過了一個副作用不明的地方,再到下一次訪問就必須重新讀取了。

所以在你增加了 prinltln 之後,JIT 由於要保守處理,重新讀取,自然就不能做上面的表達式提升優化了。

以上對表達式提升的解釋,總結摘抄自R大知乎回答。R大,行走的 JVM Wiki!

我:“這下明白了吧,這都是 JIT 乾的好事,你要是禁用 JIT 就沒這問題了”

小伙伴:“WK

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1. 常規函數 函數都擁有顯示的類型簽名,其本身也是一種類型。 1.1 函數類型 自由函數 // 自由函數 fn sum(a: i32, b: i32) -> i32 { a+b } fn main() { assert_eq!(3, sum(1, 2)) } 關聯函數與方法 struct A(i3 ...
  • 近幾年來Laravel在PHP領域大放異彩,逐漸成為PHP開發框架中的中流砥柱。 這個系列的文章, 會帶你一起探知Laravel框架底層的實現細節。與其他框架相比,Laravel的設計理念確實更為先進(服務、容器、依賴註入、facade。。。),初讀代碼時會感覺代碼晦澀難懂,而一旦弄清了整套框架的基 ...
  • 1 摘要 通過使用記錄模式來增強Java編程語言,以解構記錄值。記錄模式和類型模式可嵌套使用,從而實現強大、聲明式和可組合的數據導航和處理形式。 2 發展史 由 JEP 405 提出的預覽功能,併在JDK 19發佈,然後由 JEP 432 再次預覽,併在JDK 20發佈。該功能與用於switch的模 ...
  • 來源:麥叔編程 作者:小K 前言 一個好的變數名能讓讀代碼的人(包括寫的人),身心舒暢,但一個“奇葩”的變數名可能會逼瘋一個程式員。 今天是奇葩變數名大賞! 正文 註:以下素材均採集自網路 先上場的是某企業機房的門牌: 我猜這個主任可能是個胡建人。 推薦一個開源免費的 Spring Boot 實戰項 ...
  • 一 背景 C端服務應用升級和重啟,導致耗時瞬時抖動,業務超時,應用監控報警,上游感知明顯,導致用戶體驗變差。 二 應用升級重啟導致抖動的原因 1 C端服務應用升級和重啟的冷啟動階段,它需要重新載入和初始化各種資源,例如資料庫連接、緩存數據等,導致耗時瞬時飆升。 2 應用重啟後,本地緩存失效,應用需要 ...
  • 1 全新併發編程模式 JDK9 後的版本你覺得沒必要折騰,我也認可,但是JDK21有必要關註。因為 JDK21 引入全新的併發編程模式。 一直沽名釣譽的GoLang吹得最厲害的就是協程了。JDK21 中就在這方面做了很大的改進,讓Java併發編程變得更簡單一點,更絲滑一點。 之前寫過JDK21 Fe ...
  • 今天不知為何開始報錯 Entry WEB-INF/classes/classpath.index is a duplicate but no duplicate handling strategy has been set.,大約是由於 我把 Gradle 遷移到了 Kotlin 導致的 經過一番搜 ...
  • 大家好,我是TJ君! 如今在國內運營的各種互聯網應用都有接入IP來源顯示的要求,現在相關API的供應商也很多。今天TJ剛好看到一個不錯的,所以馬上給大家推薦一下。 這款不錯的產品名稱為:IPInfo 產品特性 該IP查詢工具除了傳統的提供地址位置之外,還有很多其他能力,具體的這裡TJ君給大家整理了一 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...