Java的synchronized關(guān)鍵字深入解析
概念
在并發(fā)編程中,多線程同時(shí)并發(fā)訪問(wèn)的資源叫做臨界資源,當(dāng)多個(gè)線程同時(shí)訪問(wèn)對(duì)象并要求操作相同資源時(shí),分割了原子操作就有可能出現(xiàn)數(shù)據(jù)的不一致或數(shù)據(jù)不完整的情況,為避免這種情況的發(fā)生,我們會(huì)采取同步機(jī)制,以確保在某一時(shí)刻,方法內(nèi)只允許有一個(gè)線程。
sychronized 用于 修飾 代碼塊、類(lèi)的實(shí)例方法 & 靜態(tài)方法
是利用鎖的機(jī)制來(lái)實(shí)現(xiàn)同步的。
鎖機(jī)制有如下兩種特性:
**互斥性:**即在同一時(shí)間只允許一個(gè)線程持有某個(gè)對(duì)象鎖,通過(guò)這種特性來(lái)實(shí)現(xiàn)多線程中的協(xié)調(diào)機(jī)制,這樣在同一時(shí)間只有一個(gè)線程對(duì)需同步的代碼塊(復(fù)合操作)進(jìn)行訪問(wèn)?;コ庑晕覀円餐Q(chēng)為操作的原子性。
**可見(jiàn)性:**必須確保在鎖被釋放之前,對(duì)共享變量所做的修改,對(duì)于隨后獲得該鎖的另一個(gè)線程是可見(jiàn)的(即在獲得鎖時(shí)應(yīng)獲得最新共享變量的值),否則另一個(gè)線程可能是在本地緩存的某個(gè)副本上繼續(xù)操作從而引起不一致。
synchronized的用法
代碼如下:
/** * 對(duì)象鎖 */ public class Test{ // 對(duì)象鎖:形式1(方法鎖) public synchronized void Method1(){ System.out.println("我是對(duì)象鎖也是方法鎖"); try{ Thread.sleep(500); } catch (InterruptedException e){ e.printStackTrace(); } } // 對(duì)象鎖:形式2(代碼塊形式) public void Method2(){ synchronized (this){ System.out.println("我是對(duì)象鎖"); try{ Thread.sleep(500); } catch (InterruptedException e){ e.printStackTrace(); } } } /** * 方法鎖(即對(duì)象鎖中的形式1) */ public synchronized void Method1(){ System.out.println("我是對(duì)象鎖也是方法鎖"); try{ Thread.sleep(500); } catch (InterruptedException e){ e.printStackTrace(); } } /** * 類(lèi)鎖 */ // 類(lèi)鎖:形式1 :鎖靜態(tài)方法 public static synchronized void Method1(){ System.out.println("鎖靜態(tài)方法"); try{ Thread.sleep(500); } catch (InterruptedException e){ e.printStackTrace(); } } // 類(lèi)鎖:形式2 :鎖靜態(tài)代碼塊 public void Method2(){ synchronized (Test.class){ System.out.println("鎖靜態(tài)代碼塊"); try{ Thread.sleep(500); } catch (InterruptedException e){ e.printStackTrace(); } } } }
使用synchronized注意的問(wèn)題
- 與moniter關(guān)聯(lián)的對(duì)象不能為空
- 盡量減小synchronized作用域范圍
- 不能使用不同的monitor鎖相同的方法
- 防止多個(gè)鎖的交叉導(dǎo)致死鎖
synchronized原理
采用 synchronized 修飾符實(shí)現(xiàn)的同步機(jī)制叫做互斥鎖機(jī)制,它所獲得的鎖叫做互斥鎖。每個(gè)對(duì)象都有一個(gè) monitor (鎖標(biāo)記),當(dāng)線程擁有這個(gè)鎖標(biāo)記時(shí)才能訪問(wèn)這個(gè)資源,沒(méi)有鎖標(biāo)記便進(jìn)入鎖池。任何一個(gè)對(duì)象系統(tǒng)都會(huì)為其創(chuàng)建一個(gè)互斥鎖,這個(gè)鎖是為了分配給線程的,防止打斷原子操作。每個(gè)對(duì)象的鎖只能分配給一個(gè)線程,因此叫做互斥鎖。
在 Java 中,每個(gè)對(duì)象有一把鎖,叫對(duì)象鎖,針對(duì)每個(gè)類(lèi)也有一個(gè)鎖,可以稱(chēng)為“類(lèi)鎖”,類(lèi)鎖實(shí)際上是通過(guò)對(duì)象鎖實(shí)現(xiàn)的,即類(lèi)的 Class 對(duì)象鎖。每個(gè)類(lèi)只有一個(gè) Class 對(duì)象,所以每個(gè)類(lèi)只有一個(gè)類(lèi)鎖。
底層依賴(lài)于 JVM ,通過(guò)一個(gè)監(jiān)視器對(duì)象monitor完成, wait、notify 等方法也依賴(lài)于 monitor 對(duì)象監(jiān)視器鎖(monitor的本質(zhì) 依賴(lài)于 底層操作系統(tǒng)的互斥鎖Mutex Lock實(shí)現(xiàn))
在 Java 中,每個(gè)對(duì)象都會(huì)有一個(gè) monitor 對(duì)象,監(jiān)視器。
- 某一線程占有這個(gè)對(duì)象的時(shí)候,先monitor 的計(jì)數(shù)器是不是0,如果是0還沒(méi)有線程占有,這個(gè)時(shí)候線程占有這個(gè)對(duì)象,并且對(duì)這個(gè)對(duì)象的monitor+1;如果不為0,表示這個(gè)線程已經(jīng)被其他線程占有,這個(gè)線程等待。當(dāng)線程釋放占有權(quán)的時(shí)候,monitor-1;
- 同一線程可以對(duì)同一對(duì)象進(jìn)行多次加鎖,+1,+1,重入性
我們先看一段簡(jiǎn)單的代碼:
public class TestSynchronized { private int count; @Test public void test() throws InterruptedException { synchronized (this){ count++; } } }
現(xiàn)在我們反編譯一下:
通過(guò)反編譯后的代碼我們可以知道synchronized修飾的代碼塊的加鎖其實(shí)是通過(guò)monitorenter和monitorExit匯編指令配合使用的。在執(zhí)行 count++ 指令之前,編譯器加了一條 monitorenter 指令,count++ 指令執(zhí)行結(jié)束時(shí)又加了一條 monitorexit 指令。準(zhǔn)確意義上來(lái)說(shuō),這就是兩條加鎖的釋放鎖的指令。
除此之外,我們的 synchronized 方法在反編譯后并沒(méi)有這兩條指令,但是編譯器卻在方法表的 flags 屬性中設(shè)置了一個(gè)標(biāo)志位 ACC_SYNCHRONIZED。這樣,每個(gè)線程在調(diào)用該方法之前都會(huì)檢查這個(gè)狀態(tài)位是否為 1,如果狀態(tài)為 1 說(shuō)明這是一個(gè)同步方法,需要首先執(zhí)行 monitorenter 指令去嘗試獲取當(dāng)前實(shí)例對(duì)象的內(nèi)置鎖,并在方法執(zhí)行結(jié)束執(zhí)行 monitorexit 指令去釋放鎖。
JVM對(duì)synchronized的優(yōu)化
synchronized鎖住的代碼塊:同一時(shí)刻只能由一個(gè)線程訪問(wèn),屬于悲觀鎖
對(duì)象頭與monitor
一個(gè)對(duì)象實(shí)例包含:對(duì)象頭、實(shí)例變量、填充數(shù)據(jù)
偏向鎖
在對(duì)象第一次被某一線程占有的時(shí)候,是否偏向鎖置1,鎖表01,寫(xiě)入線程號(hào),當(dāng)其他的線 程訪問(wèn)的時(shí)候,競(jìng)爭(zhēng),失敗則升級(jí)為輕量級(jí)鎖
CAS(campany and set)
競(jìng)爭(zhēng)失敗的時(shí)候,不是馬上轉(zhuǎn)化級(jí)別,而是執(zhí)行幾次空循環(huán)5 10
鎖消除
JIT在編譯的時(shí)候吧不必要的鎖去掉
到此這篇關(guān)于Java的synchronized關(guān)鍵字深入解析的文章就介紹到這了,更多相關(guān)synchronized關(guān)鍵字內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring?Boot對(duì)接Oracle數(shù)據(jù)庫(kù)具體流程
這篇文章主要給大家介紹了關(guān)于Spring?Boot對(duì)接Oracle數(shù)據(jù)庫(kù)的具體流程,本文將介紹如何在Spring Boot中連接Oracle數(shù)據(jù)庫(kù)的基本配置,包括添加依賴(lài)、配置數(shù)據(jù)源、配置JPA等,需要的朋友可以參考下2023-11-11一篇文章帶你了解JAVA面對(duì)對(duì)象應(yīng)用
Java是一門(mén)面向?qū)ο蟮恼Z(yǔ)言。對(duì)象是Java程序中的基本實(shí)體。除了對(duì)象之外Java程序同樣處理基本數(shù)據(jù)。下面這篇文章主要給大家總結(jié)了關(guān)于Java中面向?qū)ο蟮闹R(shí)點(diǎn),需要的朋友可以參考借鑒,下面來(lái)一起看看吧2021-08-08Java實(shí)現(xiàn)獲取某年某月第一天/最后一天的方法
這篇文章主要介紹了Java實(shí)現(xiàn)獲取某年某月第一天/最后一天的方法,涉及java日期運(yùn)算相關(guān)操作技巧,需要的朋友可以參考下2018-02-02Java實(shí)現(xiàn)紀(jì)元秒和本地日期時(shí)間互換的方法【經(jīng)典實(shí)例】
這篇文章主要介紹了Java實(shí)現(xiàn)紀(jì)元秒和本地日期時(shí)間互換的方法,結(jié)合具體實(shí)例形式分析了Java日期時(shí)間相關(guān)操作技巧,需要的朋友可以參考下2017-04-04SpringBoot多數(shù)據(jù)源配置并通過(guò)注解實(shí)現(xiàn)動(dòng)態(tài)切換數(shù)據(jù)源
本文主要介紹了SpringBoot多數(shù)據(jù)源配置并通過(guò)注解實(shí)現(xiàn)動(dòng)態(tài)切換數(shù)據(jù)源,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08詳解使用spring aop實(shí)現(xiàn)業(yè)務(wù)層mysql 讀寫(xiě)分離
本篇文章主要介紹了使用spring aop實(shí)現(xiàn)業(yè)務(wù)層mysql 讀寫(xiě)分離,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-01-01springcloud中RabbitMQ死信隊(duì)列與延遲交換機(jī)實(shí)現(xiàn)方法
死信隊(duì)列是消息隊(duì)列中非常重要的概念,同時(shí)我們需要業(yè)務(wù)場(chǎng)景中都需要延遲發(fā)送的概念,比如12306中的30分鐘后未支付訂單取消,那么本期,我們就來(lái)講解死信隊(duì)列,以及如何通過(guò)延遲交換機(jī)來(lái)實(shí)現(xiàn)延遲發(fā)送的需求,感興趣的朋友一起看看吧2022-05-05