摘要:源碼實(shí)現(xiàn)如下表示是否有異常如果是使用了偏向鎖,要撤銷偏向鎖在中可以看到方法的實(shí)現(xiàn)。
wait源碼實(shí)現(xiàn)如下
//TRAPS表示是否有異常 void ObjectSynchronizer::wait(Handle obj, jlong millis, TRAPS) { if (UseBiasedLocking) { //如果是使用了偏向鎖,要撤銷偏向鎖 BiasedLocking::revoke_and_rebias(obj, false, THREAD); assert(!obj->mark()->has_bias_pattern(), "biases should be revoked by now"); } ...
在biasedLocking.cpp中可以看到方法的實(shí)現(xiàn)。整體結(jié)構(gòu)劃分如下
BiasedLocking::Condition BiasedLocking::revoke_and_rebias(Handle obj, bool attempt_rebias, TRAPS) { //1:必須在安全點(diǎn) assert(!SafepointSynchronize::is_at_safepoint(), "must not be called while at safepoint"); //2:讀取對(duì)象頭 markOop mark = obj->mark(); if (mark->is_biased_anonymously() && !attempt_rebias) { //3:沒(méi)有線程獲取了偏向鎖 } else if (mark->has_bias_pattern()) { //4:已經(jīng)偏向了 } //5:沒(méi)有執(zhí)行偏向,通過(guò)啟發(fā)式的方式?jīng)Q定到底是執(zhí)行撤銷還是執(zhí)行rebias HeuristicsResult heuristics = update_heuristics(obj(), attempt_rebias); if (heuristics == HR_NOT_BIASED) { //5.1:偏向狀態(tài)改成了不需要偏向 } else if (heuristics == HR_SINGLE_REVOKE) { //5.2:啟發(fā)式?jīng)Q定執(zhí)行單次的撤銷 } //6:等到虛擬機(jī)運(yùn)行到safepoint,實(shí)際就是執(zhí)行 VM_BulkRevokeBias 的doit的 bulk_revoke_or_rebias_at_safepoint方法 VM_BulkRevokeBias bulk_revoke(&obj, (JavaThread*) THREAD, (heuristics == HR_BULK_REBIAS), attempt_rebias); VMThread::execute(&bulk_revoke); return bulk_revoke.status_code(); }沒(méi)有獲取偏向鎖
這里表示還沒(méi)有被偏向,并且不是執(zhí)行rebias
// We are probably trying to revoke the bias of this object due to // an identity hash code computation. Try to revoke the bias // without a safepoint. This is possible if we can successfully // compare-and-exchange an unbiased header into the mark word of // the object, meaning that no other thread has raced to acquire // the bias of the object. markOop biased_value = mark; //prootype本身構(gòu)建的是 markOop( no_hash_in_place | no_lock_in_place ); markOop unbiased_prototype = markOopDesc::prototype()->set_age(mark->age()); //執(zhí)行CAS,如果當(dāng)前對(duì)象的mark沒(méi)有變更,就換成 unbiased_prototype markOop res_mark = (markOop) Atomic::cmpxchg_ptr(unbiased_prototype, obj->mark_addr(), mark); if (res_mark == biased_value) { //如果之前的和現(xiàn)在的一樣,說(shuō)明撤銷成功,BIAS_REVOKED本身是一個(gè)枚舉 return BIAS_REVOKED; }已經(jīng)被其它線程獲取了偏向
//已經(jīng)被線程偏向了,獲取Klass對(duì)象,即類本身的頭,obj則是它的實(shí)例 Klass* k = Klass::cast(obj->klass()); markOop prototype_header = k->prototype_header(); if (!prototype_header->has_bias_pattern()) { //對(duì)象當(dāng)前的偏向狀態(tài)已經(jīng)過(guò)期,并且是不可偏向的,直接設(shè)置成已經(jīng)撤銷偏向即可 // This object has a stale bias from before the bulk revocation // for this data type occurred. It"s pointless to update the // heuristics at this point so simply update the header with a // CAS. If we fail this race, the object"s bias has been revoked // by another thread so we simply return and let the caller deal // with it. markOop biased_value = mark; markOop res_mark = (markOop) Atomic::cmpxchg_ptr(prototype_header, obj->mark_addr(), mark); assert(!(*(obj->mark_addr()))->has_bias_pattern(), "even if we raced, should still be revoked"); return BIAS_REVOKED; } else if (prototype_header->bias_epoch() != mark->bias_epoch()) { //實(shí)例的epoch和類本身的epoch值不一樣,說(shuō)明它已經(jīng)過(guò)期,也就是說(shuō)這個(gè)對(duì)象當(dāng)前處于未偏向但是可偏向的狀態(tài)(rebiasable) // The epoch of this biasing has expired indicating that the // object is effectively unbiased. Depending on whether we need // to rebias or revoke the bias of this object we can do it // efficiently enough with a CAS that we shouldn"t update the // heuristics. This is normally done in the assembly code but we // can reach this point due to various points in the runtime // needing to revoke biases. if (attempt_rebias) { //執(zhí)行rebias wait希望直接撤銷 assert(THREAD->is_Java_thread(), ""); markOop biased_value = mark; markOop rebiased_prototype = markOopDesc::encode((JavaThread*) THREAD, mark->age(), prototype_header->bias_epoch()); markOop res_mark = (markOop) Atomic::cmpxchg_ptr(rebiased_prototype, obj->mark_addr(), mark); if (res_mark == biased_value) { //當(dāng)前線程搶到了這個(gè)對(duì)象的偏向 return BIAS_REVOKED_AND_REBIASED; } } else { markOop biased_value = mark; markOop unbiased_prototype = markOopDesc::prototype()->set_age(mark->age()); //CAS撤銷偏向鎖 markOop res_mark = (markOop) Atomic::cmpxchg_ptr(unbiased_prototype, obj->mark_addr(), mark); if (res_mark == biased_value) { //撤銷了偏向 return BIAS_REVOKED; } } }啟發(fā)式策略
//啟發(fā)式的方式?jīng)Q定要做那種操作static HeuristicsResult update_heuristics(oop o, bool allow_rebias) { markOop mark = o->mark(); if (!mark->has_bias_pattern()) { //不可偏向直接返回 return HR_NOT_BIASED; } //控制撤銷的次數(shù) // Heuristics to attempt to throttle the number of revocations. // Stages: // 1. Revoke the biases of all objects in the heap of this type, // but allow rebiasing of those objects if unlocked. // 2. Revoke the biases of all objects in the heap of this type // and don"t allow rebiasing of these objects. Disable // allocation of objects of that type with the bias bit set. Klass* k = o->blueprint(); jlong cur_time = os::javaTimeMillis(); //獲取上次執(zhí)行bulk revication的時(shí)間 jlong last_bulk_revocation_time = k->last_biased_lock_bulk_revocation_time(); //獲取執(zhí)行bulk revocation的次數(shù) int revocation_count = k->biased_lock_revocation_count(); //定義在globs.hpp,BiasedLockingBulkRebiasThreshold取值為20;BiasedLockingBulkRevokeThreshold取值為40,BiasedLockingDecayTime為25000毫秒 if ((revocation_count >= BiasedLockingBulkRebiasThreshold) && (revocation_count < BiasedLockingBulkRevokeThreshold) && (last_bulk_revocation_time != 0) && (cur_time - last_bulk_revocation_time >= BiasedLockingDecayTime)) { // This is the first revocation we"ve seen in a while of an // object of this type since the last time we performed a bulk // rebiasing operation. The application is allocating objects in // bulk which are biased toward a thread and then handing them // off to another thread. We can cope with this allocation // pattern via the bulk rebiasing mechanism so we reset the // klass"s revocation count rather than allow it to increase // monotonically. If we see the need to perform another bulk // rebias operation later, we will, and if subsequently we see // many more revocation operations in a short period of time we // will completely disable biasing for this type. //在執(zhí)行了一定時(shí)間之內(nèi),執(zhí)行的撤銷次數(shù)沒(méi)有超過(guò)閾值,那么認(rèn)為可以優(yōu)先執(zhí)行bulk rebias,因此將計(jì)數(shù)回歸原始值 k->set_biased_lock_revocation_count(0); revocation_count = 0; } // Make revocation count saturate just beyond BiasedLockingBulkRevokeThreshold if (revocation_count <= BiasedLockingBulkRevokeThreshold) { //計(jì)算執(zhí)行撤銷的次數(shù) revocation_count = k->atomic_incr_biased_lock_revocation_count(); } if (revocation_count == BiasedLockingBulkRevokeThreshold) { //達(dá)到執(zhí)行bulk revoke的閾值,執(zhí)行bulk revoke return HR_BULK_REVOKE; } if (revocation_count == BiasedLockingBulkRebiasThreshold) { //達(dá)到 bulk rebias的閾值,執(zhí)行bulk rebias return HR_BULK_REBIAS; } //默認(rèn)執(zhí)行單次的撤銷 return HR_SINGLE_REVOKE; }bulk_revoke_or_rebias_at_safepoint
bulk revoke的關(guān)鍵在于它會(huì)遍歷所有線程棧的每一幀
static BiasedLocking::Condition bulk_revoke_or_rebias_at_safepoint(oop o, bool bulk_rebias, bool attempt_rebias_of_object, JavaThread* requesting_thread) { … if (bulk_rebias) { ... // Now walk all threads" stacks and adjust epochs of any biased // and locked objects of this data type we encounter //遍歷所有的線程 for (JavaThread* thr = Threads::first(); thr != NULL; thr = thr->next()) { //遍歷線程棧的每一幀,獲取所有的監(jiān)視器 GrowableArray* cached_monitor_info = get_or_compute_monitor_info(thr); for (int i = 0; i < cached_monitor_info->length(); i++) { MonitorInfo* mon_info = cached_monitor_info->at(i); oop owner = mon_info->owner(); markOop mark = owner->mark(); if ((owner->klass() == k_o) && mark->has_bias_pattern()) { // We might have encountered this object already in the case of recursive locking assert(mark->bias_epoch() == prev_epoch || mark->bias_epoch() == cur_epoch, "error in bias epoch adjustment"); //更新所有棧中的有偏向鎖的epoch owner->set_mark(mark->set_bias_epoch(cur_epoch)); } } } ... // At this point we"re done. All we have to do is potentially// adjust the header of the given object to revoke its bias. revoke_bias(o, attempt_rebias_of_object && klass->prototype_header()->has_bias_pattern(), true, requesting_thread); } … if (attempt_rebias_of_object && o->mark()->has_bias_pattern() && klass->prototype_header()->has_bias_pattern()) { //bias_epoch本身則是表示獲取了偏向鎖 markOop new_mark = markOopDesc::encode(requesting_thread, o->mark()->age(), klass->prototype_header()->bias_epoch()); o->set_mark(new_mark); //執(zhí)行rebiase status_code = BiasedLocking::BIAS_REVOKED_AND_REBIASED; ... } }
revoke_bias的執(zhí)行如下
static BiasedLocking::Condition revoke_bias(oop obj, bool allow_rebias, bool is_bulk, JavaThread* requesting_thread) { markOop mark = obj->mark(); … //偏向鎖的頭 markOop biased_prototype = markOopDesc::biased_locking_prototype()->set_age(age); //非偏向鎖的頭 markOop unbiased_prototype = markOopDesc::prototype()->set_age(age); … //獲取偏向的線程 JavaThread* biased_thread = mark->biased_locker(); if (biased_thread == NULL) { // Object is anonymously biased. We can get here if, for // example, we revoke the bias due to an identity hash code // being computed for an object. if (!allow_rebias) { //沒(méi)有線程獲取,又需要執(zhí)行rebias,改掉對(duì)象頭即可 obj->set_mark(unbiased_prototype); } ... //撤銷完畢 return BiasedLocking::BIAS_REVOKED; } … //線程活著 //遍歷棧幀,獲取所有這個(gè)線程的監(jiān)視器,按照最年輕到最老的順序 GrowableArray* cached_monitor_info = get_or_compute_monitor_info(biased_thread); BasicLock* highest_lock = NULL; for (int i = 0; i < cached_monitor_info->length(); i++) { MonitorInfo* mon_info = cached_monitor_info->at(i); if (mon_info->owner() == obj) { ... // Assume recursive case and fix up highest lock later //當(dāng)前棧幀存在了這個(gè)對(duì)象的鎖 markOop mark = markOopDesc::encode((BasicLock*) NULL); highest_lock = mon_info->lock(); //更新棧中的mark為NULL highest_lock->set_displaced_header(mark); } ... } } if (highest_lock != NULL) { // Fix up highest lock to contain displaced header and point // object at it //將最久的那個(gè)lock更新為沒(méi)有偏向 ,棧中 設(shè)置了鎖記錄 highest_lock->set_displaced_header(unbiased_prototype); // Reset object header to point to displaced mark //將對(duì)象頭指向棧中的位置,這樣表示就沒(méi)有偏向了 obj->set_mark(markOopDesc::encode(highest_lock)); assert(!obj->mark()->has_bias_pattern(), "illegal mark state: stack lock used bias bit"); ... } else { ... if (allow_rebias) { obj->set_mark(biased_prototype); } else { // Store the unlocked value into the object"s header. obj->set_mark(unbiased_prototype); } } //撤銷完畢 return BiasedLocking::BIAS_REVOKED; }
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/72752.html
摘要:關(guān)鍵字經(jīng)過(guò)編譯之后,會(huì)在同步塊的前后分別形成和這兩個(gè)字節(jié)碼指令。當(dāng)我們的把字節(jié)碼加載到內(nèi)存的時(shí)候,會(huì)對(duì)這兩個(gè)指令進(jìn)行解析。這兩個(gè)字節(jié)碼都需要一個(gè)類型的參數(shù)來(lái)指明要鎖定和解鎖的對(duì)象。最后喚醒暫停的線程。 文章簡(jiǎn)介 前面我有文章介紹了synchronized的基本原理,這篇文章我會(huì)從jvm源碼分析synchronized的實(shí)現(xiàn)邏輯,希望讓大家有一個(gè)更加深度的認(rèn)識(shí) 內(nèi)容導(dǎo)航 從synchr...
摘要:此時(shí)偏向擁有者會(huì)像輕量級(jí)鎖操作那樣,它的堆棧會(huì)填入鎖記錄,然后對(duì)象本身的會(huì)被更新成指向棧上最老的鎖記錄,然后線程本身在安全點(diǎn)的阻塞會(huì)被釋放如果沒(méi)有被原有的偏向鎖持有者持有,會(huì)撤銷對(duì)象重新回到可偏向但是還沒(méi)有偏向的狀態(tài),然后嘗試重新獲取鎖。 為什么需要偏向鎖 當(dāng)多個(gè)處理器同時(shí)處理的時(shí)候,通常需要處理互斥的問(wèn)題。 一般的解決方式都會(huì)包含acquire和release這個(gè)兩種操作,操作保證...
摘要:實(shí)現(xiàn)原理虛擬機(jī)鎖原理虛擬機(jī)中對(duì)象頭部信息可以看見(jiàn)對(duì)象頭中結(jié)構(gòu)中的成員,允許壓縮。否則,將偏向鎖撤銷,升級(jí)為輕量級(jí)鎖。存在明顯多線程競(jìng)爭(zhēng)的場(chǎng)景下使用偏向鎖是不合適的,例如生產(chǎn)者消費(fèi)者隊(duì)列。 synchronied實(shí)現(xiàn)原理 虛擬機(jī)鎖原理 虛擬機(jī)中對(duì)象頭部信息 /*hotspot/src/share/vm/oops/oop.hpp*/ class oopDesc { friend cla...
摘要:線程安全的概念什么時(shí)候線程不安全怎樣做到線程安全怎么擴(kuò)展線程安全的類對(duì)線程安全的支持對(duì)線程安全支持有哪些中的線程池的使用與中線程池的生命周期與線程中斷中的鎖中常見(jiàn)死鎖與活鎖的實(shí)例線程同步機(jī)制顯示鎖使用與原理原理剖析原理中的與原理偏向鎖狀態(tài) showImg(https://segmentfault.com/img/bVblUE9?w=1354&h=1660); 線程安全的概念 showI...
摘要:而導(dǎo)致這個(gè)問(wèn)題的原因是線程并行執(zhí)行操作并不是原子的,存在線程安全問(wèn)題。表示自旋鎖,由于線程的阻塞和喚醒需要從用戶態(tài)轉(zhuǎn)為核心態(tài),頻繁的阻塞和喚醒對(duì)來(lái)說(shuō)性能開(kāi)銷很大。 文章簡(jiǎn)介 synchronized想必大家都不陌生,用來(lái)解決線程安全問(wèn)題的利器。同時(shí)也是Java高級(jí)程序員面試比較常見(jiàn)的面試題。這篇文正會(huì)帶大家徹底了解synchronized的實(shí)現(xiàn)。 內(nèi)容導(dǎo)航 什么時(shí)候需要用Synchr...
閱讀 683·2023-04-25 18:59
閱讀 1224·2021-09-22 16:00
閱讀 1894·2021-09-22 15:42
閱讀 3602·2021-09-22 15:27
閱讀 1255·2019-08-30 15:54
閱讀 1111·2019-08-30 11:16
閱讀 2457·2019-08-29 16:24
閱讀 834·2019-08-29 12:14