這篇文章將為大家詳細講解有關(guān)Java中synchronized關(guān)鍵字的作用是什么,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個參考,希望大家閱讀完這篇文章后對相關(guān)知識有一定的了解。
創(chuàng)新互聯(lián)建站服務(wù)項目包括容城網(wǎng)站建設(shè)、容城網(wǎng)站制作、容城網(wǎng)頁制作以及容城網(wǎng)絡(luò)營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢、行業(yè)經(jīng)驗、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,容城網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到容城省份的部分城市,未來相信會繼續(xù)擴大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!
我們將其理解為同步鎖,可以實現(xiàn)共享資源的同步訪問,解決線程并發(fā)的安全問題。1.1 怎么使用的
- 修飾實例方法,作用于當(dāng)前對象實例加鎖,進入同步代碼前要獲得當(dāng)前對象實例的鎖
- 修飾靜態(tài)方法,作用于當(dāng)前類對象加鎖,進入同步代碼前要獲得當(dāng)前類對象的鎖 。
也就是給當(dāng)前類加鎖,會作用于類的所有對象實例,因為靜態(tài)成員不屬于任何一個實例對象,是類成員( static 表明這是該類的一個靜態(tài)資源,不管new了多少個對象,只有一份,所以對該類的所有對象都加了鎖)。
所以如果一個線程A調(diào)用一個實例對象的非靜態(tài) synchronized 方法,而線程B需要調(diào)用這個實例對象所屬類的靜態(tài) synchronized 方法,是允許的,不會發(fā)生互斥現(xiàn)象,因為訪問靜態(tài) synchronized 方法占用的鎖是當(dāng)前類的鎖,而訪問非靜態(tài) synchronized 方法占用的鎖是當(dāng)前實例對象鎖。
- 修飾代碼塊,指定加鎖對象,對給定對象加鎖,進入同步代碼塊前要獲得給定對象的鎖。
和 synchronized 方法一樣,synchronized(this)代碼塊也是鎖定當(dāng)前對象的。
synchronized 關(guān)鍵字加到 static 靜態(tài)方法和 synchronized(class)代碼塊上都是是給 Class 類上鎖。
這里再提一下:
synchronized關(guān)鍵字加到非 static 靜態(tài)方法上是給對象實例上鎖。
另外需要注意的是:
盡量不要使用 synchronized(String a) 因為JVM中,字符串常量池具有緩沖功能!
2.早期的synchronized
JDK1.6之前屬于重量級鎖,依賴于操作系統(tǒng)的Mutex Lock,Java的線程映射到操作系統(tǒng)的原生線程,需要操作系統(tǒng)申請互斥量,操作系統(tǒng)對線程的切換,需要從用戶態(tài)切換到內(nèi)核態(tài),比較耗時,效率底下。3.對synchronized的優(yōu)化
JDK1.6之后在JVM層面對synchronized底層做了很多的優(yōu)化,包括偏向鎖,輕量級鎖,自旋鎖,自適應(yīng)自旋鎖,鎖消除,鎖粗化等優(yōu)化技術(shù)。3.1 偏向鎖
目的:在沒有線程競爭的情況下,減少傳統(tǒng)的重量級鎖使用操作系統(tǒng)互斥量的開銷,提升性能。- 如果在接下來的執(zhí)行中偏向鎖沒有被其他線程獲取,那么擁有該鎖的線程就不需要同步
變化:在鎖競爭激烈的場合,偏向鎖失效。原因是,在此情況下,極有可能每次申請鎖的線程不是同一個線程,所以此時不應(yīng)該使用偏向鎖,否則得不償失。But,偏向鎖失效后,并不會立即膨脹為重量級鎖,而是首先升級為輕量級鎖。關(guān)于偏向鎖的原理可以查看《深入理解Java虛擬機:JVM高級特性與最佳實踐》第二版的13章第三節(jié)鎖優(yōu)化。3.2 輕量級鎖
當(dāng)偏向鎖失效,JVM不會立即升級為重量級鎖,而是試圖使用輕量級鎖的優(yōu)化手段(JDK1.6之后加入的)。輕量級鎖不是為了替代重量級鎖,它的本意是是在沒有線程競爭的情況下,減少傳統(tǒng)的重量級鎖使用操作系統(tǒng)互斥量的開銷,提升性能。- 和偏向鎖不同,輕量級鎖使用CAS操作代替重量級鎖。
變化:對于大多數(shù)鎖來說,在整個同步周期都不存在競爭,這來自經(jīng)驗數(shù)據(jù)。如果沒有競爭,輕量級鎖使用CAS操作,避免了使用互斥鎖的開銷。如果存在競爭,除了互斥鎖的開銷,還會有額外的CAS操作,所以如果存在鎖競爭,輕量級鎖比重量級鎖更慢。如果競爭激烈輕量級鎖會迅速膨脹為重量級鎖。關(guān)于輕量級鎖的原理可以查看《深入理解Java虛擬機:JVM高級特性與最佳實踐》第二版的13章第三節(jié)鎖優(yōu)化。3.3 自旋鎖和自適應(yīng)自旋鎖
輕量級鎖失效后,JVM避免線程真的在操作系統(tǒng)層面掛起,還會進行一項成為自旋鎖的優(yōu)化手段。在JDK1.6之前就有這項技術(shù)了,只是他是默認(rèn)關(guān)閉的,可以通過參數(shù)--XX:+UseSpinning開啟。JDK1.6之后默認(rèn)開啟。自旋不能完全替代阻塞,因為它還要占用處理器的時間。如果鎖被占用的時間短,那么自旋鎖的效果就好;否則,反之。自旋等待的時間必須固定,如果超過限定的次數(shù),仍然沒有獲取到鎖,就掛起線程。自旋默認(rèn)10次,可以使用參數(shù)--XX:PreBlockSpin修改。3.3.1 為什么會有自旋鎖
互斥同步對性能最大的影響是阻塞的實現(xiàn),因為線程的掛起和恢復(fù)都需要轉(zhuǎn)入內(nèi)核態(tài)去完成(用戶態(tài)到內(nèi)核態(tài)的轉(zhuǎn)換將會耗費一定的時間)。而一般線程持有鎖的時間并不會太長,如果僅僅為了這一點時間而掛起或恢復(fù)線程將會得不償失。所以JVM團隊就想:"
能否讓后面來的請求獲取鎖的線程等待一會兒而不被掛起?看看持有鎖的線程是否很快就會釋放鎖"。目的:為了減少線程的掛起和恢復(fù),減少帶來的系統(tǒng)開銷,引入自旋鎖。3.3.2 如何實現(xiàn)自旋
為了讓一個線程等待,我們只需要讓線程執(zhí)行一個忙循環(huán)(自旋),這項技術(shù)就叫做
自旋。3.3.3 自旋的特點
- JDK1.6之前默認(rèn)關(guān)閉,之后默認(rèn)打開
3.3.4 自適應(yīng)自旋鎖
另外,在JDK1.6時候引入了自適應(yīng)自旋鎖。改進:自旋次數(shù)不是固定的。根據(jù)上次同一個鎖的自旋次數(shù)和鎖的擁有者的狀態(tài)來確定自旋次數(shù)。JVM變得越來越聰明了。與自旋鎖的區(qū)別就是自旋次數(shù)不固定。3.4 鎖消除
即使JVM正在運行,如果檢測到共享數(shù)據(jù)不可能存在競爭,將會執(zhí)行鎖消除操作。這將會節(jié)省毫無意義的請求鎖的時間。3.5 鎖粗化
原則上我們寫代碼,總是建議將Synchronized代碼塊的作用范圍限制的盡量小,只在共享數(shù)據(jù)的實際作用域才進行同步,使需要同步的操作數(shù)盡量小,如果存在競爭,等待線程也會盡快拿到鎖。大部分情況下,上面的原則沒有問題,但是如果一些列的連續(xù)操作都對同一個對象反復(fù)加鎖解鎖,會帶來很多不必要的性能消耗。4.Synchronized和ReenTrantLock對比
兩者都是可重入鎖。
“可重入鎖”概念是:
自己可以再次獲取自己的內(nèi)部鎖。
比如一個線程獲得了某個對象的鎖,此時這個對象鎖還沒有釋放,當(dāng)其再次想要獲取這個對象的鎖的時候還是可以獲取的,如果不可鎖重入的話,就會造成死鎖。
同一個線程每次獲取鎖,鎖的計數(shù)器都自增1,所以要等到鎖的計數(shù)器下降為0時才能釋放鎖。
② synchronized 依賴于 JVM 而 ReenTrantLock 依賴于 API
synchronized 是依賴于 JVM 實現(xiàn)的,前面我們也講到了 虛擬機團隊在 JDK1.6 為 synchronized 關(guān)鍵字進行了很多優(yōu)化,但是這些優(yōu)化都是在虛擬機層面實現(xiàn)的,并沒有直接暴露給我們。
ReenTrantLock 是 JDK 層面實現(xiàn)的(也就是 API 層面,需要 lock() 和 unlock 方法配合 try/finally 語句塊來完成),所以我們可以通過查看它的源代碼,來看它是如何實現(xiàn)的。
③ ReenTrantLock 比 synchronized 增加了一些高級功能
相比synchronized,ReenTrantLock增加了一些高級功能。
主要來說主要有三點:
a.等待可中斷;b.可實現(xiàn)公平鎖;c.可實現(xiàn)選擇性通知(鎖可以綁定多個條件)
- ReenTrantLock提供了一種能夠中斷等待鎖的線程的機制,通過lock.lockInterruptibly()來實現(xiàn)這個機制。
也就是說正在等待的線程可以選擇放棄等待,改為處理其他事情。
- ReenTrantLock可以指定是公平鎖還是非公平鎖。而synchronized只能是非公平鎖。所謂的公平鎖就是先等待的線程先獲得鎖。ReenTrantLock默認(rèn)情況是非公平的,可以通過 ReenTrantLock類的
ReentrantLock(boolean fair)
構(gòu)造方法來制定是否是公平的。 - synchronized關(guān)鍵字與wait()和notify/notifyAll()方法相結(jié)合可以實現(xiàn)等待/通知機制,ReentrantLock類當(dāng)然也可以實現(xiàn),但是需要借助于Condition接口與newCondition() 方法。
Condition是JDK1.5之后才有的,它具有很好的靈活性,比如可以實現(xiàn)多路通知功能也就是在一個Lock對象中可以創(chuàng)建多個Condition實例(即對象監(jiān)視器),線程對象可以注冊在指定的Condition中,從而可以有選擇性的進行線程通知,在調(diào)度線程上更加靈活。在使用notify/notifyAll()方法進行通知時,被通知的線程是由 JVM 選擇的,用ReentrantLock類結(jié)合Condition實例可以實現(xiàn)“選擇性通知” ,這個功能非常重要,而且是Condition接口默認(rèn)提供的。
而synchronized關(guān)鍵字就相當(dāng)于整個Lock對象中只有一個Condition實例,所有的線程都注冊在它一個身上。
如果執(zhí)行notifyAll()方法的話就會通知所有處于等待狀態(tài)的線程這樣會造成很大的效率問題,而Condition實例的signalAll()方法 只會喚醒注冊在該Condition實例中的所有等待線程。
如果你想使用上述功能,那么選擇ReenTrantLock是一個不錯的選擇。
在JDK1.6之前,synchronized 的性能是比 ReenTrantLock 差很多。
具體表示為:
synchronized 關(guān)鍵字吞吐量隨線程數(shù)的增加,下降得非常嚴(yán)重。
而ReenTrantLock 基本保持一個比較穩(wěn)定的水平。
我覺得這也側(cè)面反映了, synchronized 關(guān)鍵字還有非常大的優(yōu)化余地。
后續(xù)的技術(shù)發(fā)展也證明了這一點,我們上面也講了在 JDK1.6 之后 JVM 團隊對 synchronized 關(guān)鍵字做了很多優(yōu)化。
JDK1.6 之后,synchronized 和 ReenTrantLock 的性能基本是持平了。所以網(wǎng)上那些說因為性能才選擇 ReenTrantLock 的文章都是錯的!JDK1.6之后,性能已經(jīng)不是選擇synchronized和ReenTrantLock的影響因素了!而且虛擬機在未來的性能改進中會更偏向于原生的synchronized,所以還是提倡在synchronized能滿足你的需求的情況下,優(yōu)先考慮使用synchronized關(guān)鍵字來進行同步!優(yōu)化后的synchronized和ReenTrantLock一樣,在很多地方都是用到了CAS操作。
關(guān)于Java中synchronized關(guān)鍵字的作用是什么就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
網(wǎng)站標(biāo)題:Java中synchronized關(guān)鍵字的作用是什么
標(biāo)題來源:
http://weahome.cn/article/iighsj.html