synchronized是java語言的一個關(guān)鍵字,用來修飾一個方法或者代碼塊,使得目標(biāo)達(dá)到線程同步的目的。
當(dāng)我們希望某個方法或者代碼塊,同一時間只能有一個線程能夠執(zhí)行,即是同一時間只有一個線程能夠進(jìn)入該方法或者代碼塊,其它線程將會被阻塞直接原線程執(zhí)行結(jié)束,此時我們使用該關(guān)鍵字。
由上面的簡介可以可得synchronized的兩種用法,如下:
public synchronized void bar() { ...}public void foo() { synchronized(this) { ... }}當(dāng)用synchronized修飾方法時,此時其功能相當(dāng)于
public void foo() { synchronized(this) { ... }}為此,我還特意看來了一下java生成class文件,發(fā)現(xiàn)對這兩份代碼生成字節(jié)碼并不一樣。那們?yōu)槭裁磿@兩種方式,我個人認(rèn)為原因有二,當(dāng)synchronized(this)的范圍從方法的第一行到最后一行時,直接用synchronized修飾方法就是一種偷懶方式;其次,synchronized(this)可以有控制更小的粒度,既是只要圈住需要同步的代碼可以減少的鎖持有時長,提高TPS。
如果您有更準(zhǔn)確解釋,希望您能在文后評論,謝謝
通過修飾代塊碼方式來實現(xiàn)同步的目標(biāo)時,相比修飾方法有兩個優(yōu)勢。 1. 只需要圈住(保護(hù))應(yīng)該圈住的代碼塊。被圈住的部分才會同步,其它使臨界區(qū)的訪問盡可能的短,從獲得更好的性能。 2. 通引用傳入的變量作為同步的標(biāo)量,它允許同一時刻有多個線程同時進(jìn)入同步塊,當(dāng)它的變量值不同時。反過來,同一時刻且同一個變量值,只允許一個線程進(jìn)入同步塊。
PRivate Object v = new Object;public void foo() { synchronized(v) { System.out.println(Thread.currentThread().getId() + ", enter"); try { TimeUnit.SECONDS.sleep(3); } catch (InterruptedException e) { } System.out.println(Thread.currentThread().getId() + ", leave"); }}public void bar() { new Thread(() -> { foo(); }).start(); new Thread(() -> { foo(); }).start()}它結(jié)果應(yīng)該是(發(fā)生阻塞)0, enter0, leave1, enter1, leave接下來我們稍微變一下,會出現(xiàn)一個不同的結(jié)果
public void bar() { new Thread(() -> { v = "123"; foo(); }).start(); new Thread(() -> { v = "321"; foo(); }).start()}它結(jié)果是不發(fā)生阻塞的v = 1
會怎么這樣呢?當(dāng)引用的是一個靜態(tài)不可變變量(即static final Object v = new Object())?
提示1,Integer a = 1; Integer b = 1;
此時,a == b
為true。當(dāng)Integer在-128~127是全局都是同一個引用變量。 提示2,ClassLoader。
如下這些細(xì)節(jié)找不到引入理由但又感得這些內(nèi)容非常有價值,便用FAQ的方式強(qiáng)行帶入。
能否在synchronized修飾的方法或代碼塊中發(fā)生線程上下文切換? ——能。簡述synchronized具有可重入性。 ——在synchronized的方法或代碼塊內(nèi)可以調(diào)用另一個帶有synchronized的方法或代碼塊,而不發(fā)生死鎖。所有變量v寫操作都發(fā)生synchronized代碼塊里,此時如果讀操作不在synchronized代碼塊里,會怎么樣呢? ——此時變量v為弱一致性。新聞熱點
疑難解答