Java中線程安全問題
一.線程不安全
多線程的執(zhí)行環(huán)境中,程序的執(zhí)行結(jié)果和預(yù)期的結(jié)果不符合,這就稱為發(fā)生了線程不安全現(xiàn)象
二.那些情況導(dǎo)致了線程不安全?
大致分為以下5種情況:
(1)CPU搶占執(zhí)行 (無法解決);
(2)非原子性 ;
(3)編譯器優(yōu)化(指令重排) 編譯器優(yōu)化在單線程下執(zhí)行沒問題,多線程下優(yōu)化會(huì)發(fā)生混亂;
(4)內(nèi)存的不可見性 ;(volatile輕量級解決)
(5)多個(gè)線程修改了同一個(gè)變量。(方案:讓線程操作自己的變量可以解決該問題,但業(yè)務(wù)場景發(fā)生變化,修改難度變大,通用性不高)
三.Java中解決線程不安全的方案
1.volatile“輕量級”解決線程不安全
volatile的出現(xiàn)可以解決上圖所展現(xiàn)的內(nèi)存不可見問題以及禁止指令重排
實(shí)現(xiàn)原理:工作內(nèi)存中的變量操作結(jié)束后,強(qiáng)制刪除線程工作內(nèi)存中的變量,起到內(nèi)存可見注意事項(xiàng):volatile不能解決原子性問題。volatile可以解決線程不安全問題是錯(cuò)誤的(說法不夠嚴(yán)謹(jǐn))
下面兩種方案通過對關(guān)鍵代碼加鎖,讓cpu排隊(duì)執(zhí)行,鎖操作的步驟為:
1)嘗試獲取鎖,如果拿到則加鎖,否則排隊(duì)等待獲取鎖 2)釋放鎖操作
2.synchronized自動(dòng)加鎖
①synchronized進(jìn)行自動(dòng)加鎖和釋放鎖,是Jvm層面的解決方案
synchronized使用舉例:使用兩個(gè)線程對變量count進(jìn)行一次++ 和 一次- -
沒有使用synchronized上鎖之前,由于非原子問題,兩個(gè)線程進(jìn)行++和- -出現(xiàn)線程不安全問題,通過synchronized關(guān)鍵字的使用,解決了非原子問題,代碼運(yùn)行實(shí)際結(jié)果和預(yù)期結(jié)果一致,保證了線程安全。
②synchronized實(shí)現(xiàn)原理:
1.基于操作系統(tǒng)而言,通過互斥鎖mutex實(shí)現(xiàn)
mutex結(jié)構(gòu)信息:
2.從Jvm層面來看,實(shí)現(xiàn)了一個(gè)監(jiān)視器鎖的加鎖和釋放鎖過程。
3.從Java語言本身來看,存在一個(gè)互斥鎖mutex對象,鎖存在于對象的對象頭中,對象頭中的“偏向線程ID”,表明該鎖被該線程占有,釋放鎖后,偏向線程ID消失。
Owner代表鎖的擁有者,為null時(shí)表示鎖未使用;Nest表示鎖的使用次數(shù),為0表示沒有被使用;此外鎖可以嵌套使用,不會(huì)發(fā)生死鎖情況。
③synchronized鎖升級過程:
沒有線程訪問時(shí)處于無鎖狀態(tài) >> 第一個(gè)線程訪問時(shí),由無鎖狀態(tài)轉(zhuǎn)為偏向鎖 >> 輕量級鎖(其他線程嘗試獲取鎖,鎖處于自旋狀態(tài)) >> 重量級鎖(把沒有拿到鎖的線程放到等待隊(duì)列里面)
3.Lock手動(dòng)上鎖
Lock需要程序員自己手動(dòng)上鎖手動(dòng)釋放鎖;Lock是一個(gè)interface;創(chuàng)建鎖時(shí)可以通過Lock的實(shí)現(xiàn)類ReentrantLock()完成:Lock lock = new ReentrantLock();加鎖操作lock.lock(),釋放鎖操作lock.unlock()
使用Lock需要注意的問題:
一定要把加鎖操作lock()放在try/finally外面如果把lock()放在try中會(huì)導(dǎo)致兩個(gè)問題發(fā)生:
(1)try中代碼出現(xiàn)異常,此時(shí)就會(huì)執(zhí)行finally中釋放鎖的操作,如果try還沒有加鎖就去釋放鎖,勢必是不行的。
(2)try中出現(xiàn)異常后,執(zhí)行finally中釋放鎖操作,線程狀態(tài)異常會(huì)將try中業(yè)務(wù)異常覆蓋掉,增加了排除錯(cuò)誤的成本。
將lock()放在try中第一句可以解決這個(gè)問題
對比發(fā)現(xiàn),這樣做業(yè)務(wù)異常是不會(huì)被線程的狀態(tài)異常覆蓋的,方便了排查錯(cuò)誤?。?!
四.公平鎖與非公平鎖機(jī)制
公平鎖線程按順序執(zhí)行;非公平鎖沒有順序,執(zhí)行效率更高;Java中默認(rèn)鎖策略為非公平鎖機(jī)制synchronized鎖機(jī)制:采用非公平鎖機(jī)制Lock鎖機(jī)制:默認(rèn)采用非公平鎖機(jī)制,但是可以顯示地聲明為公平鎖,比如在創(chuàng)建鎖對象時(shí),在構(gòu)造方法中傳true:Lock lock = new ReentrantLock(true)
五.volatile和synchronized的區(qū)別
volatile可以解決內(nèi)存不可見問題以及禁止指令重排序,但是不能解決非原子性問題;
synchronized可以解決大部分線程的非安全問題,保證關(guān)鍵代碼排隊(duì)執(zhí)行,無論何時(shí)鎖只被一個(gè)線程擁有,可解決非原子性問題
六.synchronized和Lock的區(qū)別
1.synchronized自動(dòng)加鎖和釋放鎖,而Lock需要手動(dòng)加鎖和釋放鎖;
2.synchronized是Jvm層面的實(shí)現(xiàn),Lock是Java語言層面的實(shí)現(xiàn);
3.適用范圍不同:synchronized可以修飾代碼塊(對任意對象加鎖)、修飾靜態(tài)方法(對當(dāng)前的類進(jìn)行加鎖)、修飾普通的方法(對當(dāng)前的實(shí)例對象進(jìn)行加鎖);而Lock只能修飾代碼塊;
4.synchronized只有非公平鎖策略;Lock默認(rèn)采用非公平鎖機(jī)制,但可以顯示聲明為公平鎖;
5.Lock的靈活性更高一些(比如:tryLock)
到此這篇關(guān)于Java中線程安全問題的文章就介紹到這了,更多相關(guān)Java線程安全內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Maven的pom.xml文件結(jié)構(gòu)中的build
本文主要介紹了Maven的pom.xml文件結(jié)構(gòu)中的build,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07以Java代碼為例講解設(shè)計(jì)模式中的簡單工廠模式
簡單來說,工廠模式就是按照需求來返回一個(gè)類型的對象,使用工廠模式的意義就是,如果對象的實(shí)例化與代碼依賴太大的話,不方便進(jìn)行擴(kuò)展和維護(hù),使用工廠的目的就是使對象的實(shí)例化與主程序代碼就行解耦.來具體看一下:2016-05-05springboot實(shí)現(xiàn)圖片大小壓縮功能
這篇文章主要為大家詳細(xì)介紹了springboot實(shí)現(xiàn)圖片大小壓縮功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04攜程Apollo(阿波羅)安裝部署以及java整合實(shí)現(xiàn)
這篇文章主要介紹了攜程Apollo(阿波羅)安裝部署以及java整合實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08spring配置定時(shí)任務(wù)的幾種方式總結(jié)
這篇文章主要介紹了spring配置定時(shí)任務(wù)的幾種方式總結(jié),具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12