一. volatile 關鍵字的作用,volatile是線程安全的嗎?
1. 作用
對于可見性,Java 提供了 volatile 關鍵字來保證可見性和禁止指令重排。 volatile 提供 happens-before 的保證,確保一個線程的修改能對其他線程是可見的。當一個共享變量被 volatile 修飾時,它會保證修改的值會立即被更新到主存,當有其他線程需要讀取時,它會去內存中讀取新值。
從實踐角度而言,volatile 的一個重要作用就是和 CAS 結合,保證了原子性,詳細的可以參見 java.util.concurrent.atomic 包下的類,比如 AtomicInteger。
volatile 常用于多線程環境下的單次操作(單次讀或者單次寫)。
2. 線程安全與否
volatile修飾的變量在各個線程的工作內存中不存在一致性的問題(在各個線程工作的內存中,volatile修飾的變量也會存在不一致的情況,但是由于每次使用之前都會先刷新主存中的數據到工作內存,執行引擎看不到不一致的情況,因此可以認為不存在不一致的問題),但是java的運算并非原子性的操作,導致volatile在并發下并非是線程安全的。
二. ThreadLocal 是什么?有哪些使用場景?
ThreadLocal 是一個本地線程副本變量工具類,在每個線程中都創建了一個 ThreadLocalMap 對象,簡單說 ThreadLocal 就是一種以空間換時間的做法,每個線程可以訪問自己內部 ThreadLocalMap 對象內的 value。通過這種方式,避免資源在多線程間共享。
原理:線程局部變量是局限于線程內部的變量,屬于線程自身所有,不在多個線程間共享。Java提供ThreadLocal類來支持線程局部變量,是一種實現線程安全的方式。但是在管理環境下(如 web 服務器)使用線程局部變量的時候要特別小心,在這種情況下,工作線程的生命周期比任何應用變量的生命周期都要長。任何線程局部變量一旦在工作完成后沒有釋放,Java 應用就存在內存泄露的風險。
經典的使用場景是為每個線程分配一個 JDBC 連接 Connection。這樣就可以保證每個線程的都在各自的 Connection 上進行數據庫的操作,不會出現 A 線程關了 B線程正在使用的 Connection; 還有 Session 管理 等問題。
三. 請談談 ThreadLocal 是怎么解決并發安全的?
在java程序中,常用的有兩種機制來解決多線程并發問題,一種是sychronized方式,通過鎖機制,一個線程執行時,讓另一個線程等待,是以時間換空間的方式來讓多線程串行執行。而另外一種方式就是ThreadLocal方式,通過創建線程局部變量,以空間換時間的方式來讓多線程并行執行。兩種方式各有優劣,適用于不同的場景,要根據不同的業務場景來進行選擇。
在spring的源碼中,就使用了ThreadLocal來管理連接,在很多開源項目中,都經常使用ThreadLocal來控制多線程并發問題,因為它足夠的簡單,我們不需要關心是否有線程安全問題,因為變量是每個線程所特有的。
四. 談談你對ThreadLocal的理解,使用ThreadLocal需要注意些什么?
ThreadLocal 變量解決了多線程環境下單個線程中變量的共享問題,使用名為ThreadLocalMap的哈希表進行維護(key為ThreadLocal變量名,value為ThreadLocal變量的值);
使用時需要注意以下幾點:
線程之間的threadLocal變量是互不影響的,
使用private final static進行修飾,防止多實例時內存的泄露問題
線程池環境下使用后將threadLocal變量remove掉或設置成一個初始值
更多關于“Java培訓”的問題,歡迎咨詢千鋒教育在線名師。千鋒已有十余年的培訓經驗,課程大綱更科學更專業,有針對零基礎的就業班,有針對想提升技術的好程序員班,高品質課程助力你實現java程序員夢想。