Java monitor機(jī)制使用方法解析
monitor概念
管程,監(jiān)視器。在操作系統(tǒng)中,存在著semaphore和mutex,即信號(hào)量和互斥量,使用基本的mutex進(jìn)行開發(fā)時(shí),需要小心的使用mutex的down和up操作,否則容易引發(fā)死鎖問題。為了更好的編寫并發(fā)程序,在mutex和semaphore基礎(chǔ)上,提出了更高層次的同步原語,實(shí)際上,monitor屬于編程語言的范疇,C語言不支持monitor,而java支持monitor機(jī)制。
一個(gè)重要特點(diǎn)是,在同一時(shí)間,只有一個(gè)線程/進(jìn)程能進(jìn)入monitor所定義的臨界區(qū),這使得monitor能夠?qū)崿F(xiàn)互斥的效果。無法進(jìn)入monitor的臨界區(qū)的進(jìn)程/線程,應(yīng)該被阻塞,并且在適當(dāng)?shù)臅r(shí)候被喚醒。顯然,monitor作為一個(gè)同步工具,也應(yīng)該提供這樣管理線程/進(jìn)程的機(jī)制。
monitor這個(gè)機(jī)制之所以被稱為:更高級(jí)的原語,它不可避免的需要對(duì)外屏蔽這些機(jī)制,并且在內(nèi)部實(shí)現(xiàn)這些機(jī)制,使得monitor成為一個(gè)簡(jiǎn)潔易用的借口。
monitor基本元素
- 臨界區(qū)
- monitor對(duì)象和鎖
- 條件變量以及定義在monitor對(duì)象上的wait,signal操作
使用monitor主要是為了互斥進(jìn)入臨界區(qū),為了能夠阻塞無法進(jìn)入臨界區(qū)的進(jìn)程,線程,需要一個(gè)monitor object來協(xié)助,這個(gè)object內(nèi)部會(huì)有相應(yīng)的數(shù)據(jù)結(jié)構(gòu),例如列表,用來保存被阻塞的線程;同時(shí)由于monitor機(jī)制本質(zhì)是基于mutex原語的,所以object必須維護(hù)一個(gè)基于mutex的鎖。
此外,為了在適當(dāng)?shù)臅r(shí)候能夠阻塞和喚醒 進(jìn)程/線程,還需要引入一個(gè)條件變量,這個(gè)條件變量用來決定什么時(shí)候是“適當(dāng)?shù)臅r(shí)候”,這個(gè)條件可以來自程序代碼的邏輯,也可以是在 monitor object 的內(nèi)部,總而言之,程序員對(duì)條件變量的定義有很大的自主性。不過,由于 monitor object 內(nèi)部采用了數(shù)據(jù)結(jié)構(gòu)來保存被阻塞的隊(duì)列,因此它也必須對(duì)外提供兩個(gè) API 來讓線程進(jìn)入阻塞狀態(tài)以及之后被喚醒,分別是 wait 和 notify。
monitor在java中的實(shí)現(xiàn)
臨界區(qū)的圈定
被synchronized關(guān)鍵字修飾的方法,代碼塊,就是monitor機(jī)制的臨界區(qū)
monitor object
在上述synchronized關(guān)鍵字被使用時(shí),往往需要指定一個(gè)對(duì)象與之關(guān)聯(lián),例如synchronized(this),總之,synchronized需要管理一個(gè)對(duì)象,這個(gè)對(duì)象就是monitor object。
monitor機(jī)制中,monitor 我不檢測(cè)題充當(dāng)著維護(hù)mutex和wait, signalAPI來管理線程的阻塞和喚醒。
Java 對(duì)象存儲(chǔ)在內(nèi)存中,分別分為三個(gè)部分,即對(duì)象頭、實(shí)例數(shù)據(jù)和對(duì)齊填充,而在其對(duì)象頭中,保存了鎖標(biāo)識(shí);同時(shí),java.lang.Object 類定義了 wait(),notify(),notifyAll() 方法,這些方法的具體實(shí)現(xiàn),依賴于一個(gè)叫 ObjectMonitor 模式的實(shí)現(xiàn),這是 JVM 內(nèi)部基于 C++ 實(shí)現(xiàn)的一套機(jī)制,基本原理如下所示:
當(dāng)一個(gè)線程需要獲取 Object 的鎖時(shí),會(huì)被放入 EntrySet 中進(jìn)行等待,如果該線程獲取到了鎖,成為當(dāng)前鎖的 owner。如果根據(jù)程序邏輯,一個(gè)已經(jīng)獲得了鎖的線程缺少某些外部條件,而無法繼續(xù)進(jìn)行下去(例如生產(chǎn)者發(fā)現(xiàn)隊(duì)列已滿或者消費(fèi)者發(fā)現(xiàn)隊(duì)列為空),那么該線程可以通過調(diào)用 wait 方法將鎖釋放,進(jìn)入 wait set 中阻塞進(jìn)行等待,其它線程在這個(gè)時(shí)候有機(jī)會(huì)獲得鎖,去干其它的事情,從而使得之前不成立的外部條件成立,這樣先前被阻塞的線程就可以重新進(jìn)入 EntrySet 去競(jìng)爭(zhēng)鎖。這個(gè)外部條件在 monitor 機(jī)制中稱為條件變量。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
實(shí)現(xiàn)分布式WebSocket集群的方法
本文總結(jié)出了幾個(gè)實(shí)現(xiàn)分布式WebSocket集群的辦法,從zuul到spring cloud gateway的不同嘗試,總結(jié)出了這篇文章,希望能幫助到某些人,并且能一起分享這方面的想法與研究2022-03-03java實(shí)現(xiàn)文件斷點(diǎn)續(xù)傳下載功能
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)文件斷點(diǎn)續(xù)傳下載功能的具體代碼,感興趣的小伙伴們可以參考一下2016-05-05通過Java實(shí)現(xiàn)中文分詞與文本關(guān)鍵詞提取
這篇文章主要為大家詳細(xì)介紹了如何利用Java實(shí)現(xiàn)中文分詞以及文本關(guān)鍵詞提取功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)學(xué)習(xí)2023-06-06java 文件目錄讀寫刪除操作詳細(xì)實(shí)現(xiàn)代碼
這篇文章主要介紹了java 文件讀寫刪操作詳細(xì)實(shí)現(xiàn)代碼,需要的朋友可以參考下2017-09-09詳解SpringBoot與SpringCloud的版本對(duì)應(yīng)詳細(xì)版
這篇文章主要介紹了詳解SpringBoot與SpringCloud的版本對(duì)應(yīng)詳細(xì)版,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09Spring Boot + thymeleaf 實(shí)現(xiàn)文件上傳下載功能
最近同事問我有沒有有關(guān)于技術(shù)的電子書,我打開電腦上的小書庫(kù),但是郵件發(fā)給他太大了,公司又禁止用文件夾共享,于是花半天時(shí)間寫了個(gè)小的文件上傳程序,部署在自己的Linux機(jī)器上,需要的朋友可以參考下2018-01-01如何在 Linux 上搭建 java 部署環(huán)境(安裝jdk/tomcat/mys
這篇文章主要介紹了如何在 Linux 上搭建 java 部署環(huán)境(安裝jdk/tomcat/mysql) + 將程序部署到云服務(wù)器上的操作),本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-01-01Java?spring?boot發(fā)送郵箱實(shí)現(xiàn)過程記錄
我們?cè)?站上注冊(cè)賬號(hào)的時(shí)候?般需要獲取驗(yàn)證碼,?這個(gè)驗(yàn)證碼?般發(fā)送在你的?機(jī)號(hào)上還有的是發(fā)送在你的郵箱中,這篇文章主要給大家介紹了關(guān)于Java?spring?boot發(fā)送郵箱實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下2024-01-01JAVA實(shí)現(xiàn)簡(jiǎn)單系統(tǒng)登陸注冊(cè)模塊
這篇文章主要介紹了一個(gè)簡(jiǎn)單完整的登陸注冊(cè)模塊的實(shí)現(xiàn)過程,文章條理清晰,在實(shí)現(xiàn)過程中加深了對(duì)相關(guān)概念的理解,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2015-07-07