主頁 > 後端開發 > 消失的死鎖:從 JSF 執行緒池滿到 JVM 初始化原理剖析

消失的死鎖:從 JSF 執行緒池滿到 JVM 初始化原理剖析

2023-06-14 10:31:40 後端開發

一、問題描述

在一次上線時,按照正常流程上線后,觀察了線上報文、介面可用率十分鐘以上,未出現例外情況,結果在上線一小時后突然收到jsf執行緒池耗盡的報警,并且該應用一共有30臺機器,只有一臺機器出現該問題,迅速下線該機器的jsf介面,恢復線上,然后開始排查問題,

報錯日志資訊:

[WARN]2023-04-10 18:03:34.847 [ - ][] |[JSF-23002]Task:java.util.concurrent.FutureTask@502cdfa0 has been reject for ThreadPool exhausted! pool:200, active:200, queue:0, taskcnt: 2159[BusinessPool#:][JSF-SEV-WORKER-225-T-8]

二、問題分析

1、出現問題原因:

a)因為只有一臺機器出現執行緒池耗盡,其他機器均正常運行,所以第一時間判斷是否為有大量流量負載不均衡導致;

b)業務代碼存在并發鎖;

c)業務代碼處理時間較長;

d)訪問資料源(如DB、redis)變慢;

排查介面流量UMP監控,按斬訓器緯度看,發現每個機器流量是均衡的,排除a)項;

排查業務量大的介面UMP KEY監控,按斬訓器緯度看,正常機器和例外機器耗時基本一致,并于往常一致,無較大差異,排除c)項;

排查資料庫監控,無慢sql,讀寫均無耗時較長的情況,排除d)項;

綜上,只剩下b)項,確認問題原因是代碼存在并發鎖,故開始排查日志及業務代碼,

2、根據已確認的原因排查思路:

1)down下dump檔案,發現極多JSF執行緒處于RUNNABLE狀態,并且堆疊處于SerializersHelper類

"JSF-BZ-22000-223-T-200" #1251 daemon prio=5 os_prio=0 tid=0x00007fd15005c000 nid=0xef6 in Object.wait() [0x00007fce287ac000]
   java.lang.Thread.State: RUNNABLE
	at com.jd.purchase.utils.serializer.helper.SerializersHelper.ofString(SerializersHelper.java:79)
	at com.jd.ldop.pipe.proxy.OrderMiddlewareCBDExportServiceProxy.getAddress(OrderMiddlewareCBDExportServiceProxy.java:97)
	at com.jd.ldop.pipe.proxy.OrderMiddlewareCBDExportServiceProxy.findOrder(OrderMiddlewareCBDExportServiceProxy.java:211)

根據堆疊資訊排查代碼,發現代碼會初始化一個自定義的序列化工廠類:SerializerFactory

并且此時初始化時會列印日志:

log.info("register: {} , clazz : {}", serializer.getCode(), serializer.getClass().getName());

故根據此日志關鍵字排查初始化加載日志,發現正常機器都加載了兩個序列化物件,只有出問題的那個機器只加載了這一個序列化物件,

于是問題初步定位到出問題的機器初始化ProtoStuffSerializer這個類時失敗,

初始化此類時static代碼塊為:

static {
    STRATEGY = new DefaultIdStrategy(IdStrategy.DEFAULT_FLAGS);
}

2)開始排查為什么初始化這個類會失敗

由于不同機器存在初始化成功和失敗的獨立性,首先考慮jar包是否沖突

a)于是發現這里存在jar沖突,但是將沖突jar排除后,多次重啟機器后發現依然存在此ProtoStuffSerializer初始化失敗情況,

b)存在死鎖,但是正常邏輯下,存在死鎖的話,應該所有機器都會存在此類情況,但是此時大概只有5%的幾率出現死鎖,并且排查jstack發現200個執行緒都卡在獲取ProtoStuffSerializer,

山重水盡疑無路 柳暗花明又一村

3、找到問題

此時排除了所有沒可能的選項,剩下一個可能性再低也是正確選項,

如果存在死鎖情況的話,那jstack的執行緒堆疊資訊肯定會報出來,于是根據jstack執行緒資訊逐個排查每一個執行緒,

最后發現下面這個執行緒的堆疊:

"jcase-jmq-reporter-t-0" #1010 daemon prio=5 os_prio=0 tid=0x00007fd258004800 nid=0x9ba in Object.wait() [0x00007fd10fffd000]
   java.lang.Thread.State: RUNNABLE
	at io.protostuff.runtime.RuntimeEnv.<clinit>(RuntimeEnv.java:229)
	at io.protostuff.runtime.IdStrategy.<clinit>(IdStrategy.java:53)
	at io.protostuff.runtime.ExplicitIdStrategy$Registry.<init>(ExplicitIdStrategy.java:67)
	at com.jd.tp.jcase.util.RecordSerializers$ProtostuffIdRegistry.<init>(RecordSerializers.java:108)
	at com.jd.tp.jcase.util.RecordSerializers.<clinit>(RecordSerializers.java:34)
	at com.jd.tp.jcase.recording.agent.reporter.impl.JmqReporter$ReportRunner.run(JmqReporter.java:106)
	at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
	at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
	at java.lang.Thread.run(Thread.java:748)

發現此執行緒(in Object.wait())也依然處于等待狀態,并且此執行緒的堆疊資訊中包含了protostuff這個關鍵字(由于上面執行緒都等待在初始化protostuffprotostuff導致的!)

于是乎開始分析此執行緒!

從此行堆疊資訊開始排查

at com.jd.tp.jcase.recording.agent.reporter.impl.JmqReporter$ReportRunner.run(JmqReporter.java:106)

xml中存在以下bean:

<bean id="jcaseJmqReporter"  init-method="start" destroy-method="stop">
    <constructor-arg name="name" value="https://www.cnblogs.com/Jcloud/p/jmq"/>
    <constructor-arg name="recorder" ref="jcaseRecorder"/>
    <constructor-arg name="topic" value="https://www.cnblogs.com/Jcloud/p/${jmq.topic.ldopjcasereporter}"/>
    <constructor-arg name="producer" ref="jcaseJmqProducer"/>
    <property name="config" ref="jcaseConfig"/>
</bean>

發現以下代碼符合堆疊資訊:

根據此執行緒的堆疊資訊逐行排查代碼,發現該執行緒執行JmqReporter.run方法時,會初始化RecordSerializers類;并在RecordSerializers中的靜態代碼塊會執行如下代碼:

RecordSerializers.ProtostuffIdRegistry registry = new RecordSerializers.ProtostuffIdRegistry();

于是執行這個類的無參構造時會new出類變數:

于是執行緒開始初始化ExplicitIdStrategy這個類:

開始執行父類的有參構造:

于是開始初始化IdStrategy類,并且執行IdStrategy類的static靜態代碼塊:

于是此處開始初始化RuntimeEnv,并且執行RuntimeEnv的靜態代碼塊;執行緒堆疊資訊就顯示等待在此類了,

排查RuntimeEnv的static代碼塊時發現存在和上一個執行緒使用了相同的類:

new DefaultIdStrategy();

類加載的問題?

首次應該懷疑是類加載的問題,因為除了兩百個執行緒停留在加載protostuffprotostuff(初始化有new DefaultIdStrategy()的代碼)這個類上,此執行緒也處于等待狀態,并且也在加載DefaultIdStrategy()的類上,

然后再分析一下這個執行緒的堆疊資訊,

"jcase-jmq-reporter-t-0" #1010 daemon prio=5 os_prio=0 tid=0x00007fd258004800 nid=0x9ba in Object.wait() [0x00007fd10fffd000]
   java.lang.Thread.State: RUNNABLE
	at io.protostuff.runtime.RuntimeEnv.<clinit>(RuntimeEnv.java:229)
	at io.protostuff.runtime.IdStrategy.<clinit>(IdStrategy.java:53)
	at io.protostuff.runtime.ExplicitIdStrategy$Registry.<init>(ExplicitIdStrategy.java:67)
	at com.jd.tp.jcase.util.RecordSerializers$ProtostuffIdRegistry.<init>(RecordSerializers.java:108)
	at com.jd.tp.jcase.util.RecordSerializers.<clinit>(RecordSerializers.java:34)
	at com.jd.tp.jcase.recording.agent.reporter.impl.JmqReporter$ReportRunner.run(JmqReporter.java:106)
	at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
	at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
	at java.lang.Thread.run(Thread.java:748)

   Locked ownable synchronizers:
	- <0x00000000c81fce28> (a java.util.concurrent.ThreadPoolExecutor$Worker)

可以看到在RuntimeEnv、IdStrategy后都有

從名字上來不難猜到是正在做類的初始化,那我們先來了解下類的初始化程序,

類的初始化程序

當我們第一次主動呼叫某個類的靜態方法就會觸發這個類的初始化,當然還有其他的觸發情況,類的初始化說白了就是在類加載起來之后,在某個合適的時機執行這個類的clinit方法,

clinit方法是什么?

比如我們在類里宣告一段static代碼塊,或者有靜態屬性,javac會將這些代碼都統一放到一個叫做clinit的方法里,在類初始化的時候來執行這個方法,但是JVM必須要保證這個方法只能被執行一次,如果有其他執行緒并發呼叫觸發了這個類的多次初始化,那只能讓一個執行緒真正執行clinit方法,其他執行緒都必須等待,當clinit方法執行完之后,然后再喚醒其他等待這里的執行緒繼續操作,當然不會再讓它們有機會再執行clinit方法,因為每個類都有一個狀態,這個狀態可以保證這一點,

public static class ClassState {
    public static final InstanceKlass.ClassState ALLOCATED = new InstanceKlass.ClassState("allocated");
    public static final InstanceKlass.ClassState LOADED = new InstanceKlass.ClassState("loaded");
    public static final InstanceKlass.ClassState LINKED = new InstanceKlass.ClassState("linked");
    public static final InstanceKlass.ClassState BEING_INITIALIZED = new InstanceKlass.ClassState("beingInitialized");
    public static final InstanceKlass.ClassState FULLY_INITIALIZED = new InstanceKlass.ClassState("fullyInitialized");
    public static final InstanceKlass.ClassState INITIALIZATION_ERROR = new InstanceKlass.ClassState("initializationError");
    private String value;

    private ClassState(String value) {
        this.value = https://www.cnblogs.com/Jcloud/p/value;
    }

    public String toString() {
        return this.value;
    }
}

當有個執行緒正在執行這個類的clinit方法的時候,就會設定這個類的狀態為being_initialized,當正常執行完之后就馬上設定為fully_initialized,然后才喚醒其他也在等著對其做初始化的執行緒繼續往下走,在繼續走下去之前,會先判斷這個類的狀態,如果已經是fully_initialized了說明有執行緒已經執行完了clinit方法,因此不會再執行clinit方法了

類加載的動作

void TemplateTable::checkcast() {
   ...
   call_VM(rax, CAST_FROM_FN_PTR(address, InterpreterRuntime::quicken_io_cc));
   ...
}

IRT_ENTRY(void, InterpreterRuntime::quicken_io_cc(JavaThread* thread))
 // Force resolving; quicken the bytecode
 int which = get_index_u2(thread, Bytecodes::_checkcast);
 constantPoolOop cpool = method(thread)->constants();
 // We'd expect to assert that we're only here to quicken bytecodes, but in a multithreaded
 // program we might have seen an unquick'd bytecode in the interpreter but have another
 // thread quicken the bytecode before we get here.
 // assert( cpool->tag_at(which).is_unresolved_klass(), "should only come here to quicken bytecodes" );
 klassOop klass = cpool->klass_at(which, CHECK);
 thread->set_vm_result(klass);
IRT_END

klassOop klass_at(int which, TRAPS) {
   constantPoolHandle h_this(THREAD, this);
   return klass_at_impl(h_this, which, CHECK_NULL);
}

klassOop constantPoolOopDesc::klass_at_impl(constantPoolHandle this_oop, int which, TRAPS) {
   ...
   klassOop k_oop = SystemDictionary::resolve_or_fail(name, loader, h_prot, true, THREAD);
   ...
}    

//SystemDictionary::resolve_or_fail最侄訓呼叫到下面這個方法
klassOop SystemDictionary::resolve_instance_class_or_null(Symbol* name, Handle class_loader, Handle protection_domain, TRAPS) {
 ...
 // Class is not in SystemDictionary so we have to do loading.
 // Make sure we are synchronized on the class loader before we proceed
 Handle lockObject = compute_loader_lock_object(class_loader, THREAD);
 check_loader_lock_contention(lockObject, THREAD);
 ObjectLocker ol(lockObject, THREAD, DoObjectLock);
 ...
 //此時會呼叫ClassLoader.loadClass來加載類了
 ...
}

Handle SystemDictionary::compute_loader_lock_object(Handle class_loader, TRAPS) {
 // If class_loader is NULL we synchronize on _system_loader_lock_obj
 if (class_loader.is_null()) {
   return Handle(THREAD, _system_loader_lock_obj);
 } else {
   return class_loader;
 }
}

SystemDictionary::resolve_instance_class_or_null這個方法非常關鍵了,在里面我們看到會獲取一把鎖ObjectLocker,其相當于我們java代碼里的synchronized關鍵字,而物件對應的是lockObject,這個物件是上面的SystemDictionary::compute_loader_lock_object方法回傳的,從代碼可知只要不是bootstrapClassloader加載的類就會回傳當前classloader物件,也就是說當我們在加載一個類的時候其實是會持有當前類加載物件的鎖的,在獲取了這把鎖之后就會呼叫ClassLoader.loadClass來加載類了,

小結

看到這里是否能解釋了我們線上為什么會有那么多執行緒會卡在某一個地方了?因為這個類的狀態是being_initialized,所以只能等了,

這個類加載的鎖,不過遺憾的是因為這把鎖不是java層面來顯示加載的,因此我們在jstack執行緒dump的輸出里居然看不到這把鎖的存在,

從dump來看確實是死鎖了,那這個場景當時是怎么發生的呢?

如圖所示,最后A、B執行緒均在等待對方初始化完成,然后C、D、E等兩百個執行緒需要使用ProtoStuffSerializer時,就在等待A執行緒初始化ProtoStuffSerializer完成,因此造成了JSF執行緒池爆滿,

"JSF-BZ-22000-223-T-1" #980 daemon prio=5 os_prio=0 tid=0x00007fd164002000 nid=0x99a in Object.wait() [0x00007fd1de8b7000]
   java.lang.Thread.State: RUNNABLE
	at com.jd.purchase.utils.serializer.impl.ProtoStuffSerializer.<clinit>(ProtoStuffSerializer.java:42)
	at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
	at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:62)
	at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
	at java.lang.reflect.Constructor.newInstance(Constructor.java:423)

只有此執行緒獲取到了ProtoStuffSerializer的初始化鎖也間接證明了這一點,

三、解決方案

了解到是由于A、B執行緒互相爭奪對方的初始化鎖后,那么為了打破這一點,就讓其中某一個執行緒提前初始化這些類就可以了,

這里選擇提前加載這個bean:初始化業務所使用到的類

四、Demo驗證

Demo代碼:

public class JVMTest {
    public static void main(String[] args) {
        new Thread(){
            public void run(){
                B.test();
            }
        }.start();

        new Thread(){
            public void run(){
                A.test();
            }
        }.start();
    }

}
class A{
    static{
        int a=0;
        System.out.println(a);
        B.test();
    }
    static void test(){
        System.out.println("呼叫了A的test方法");
    }
}
class B{
    static{
        int b=0;
        System.out.println(b);
        A.test();
    }
    static void test(){
        System.out.println("呼叫了B的test方法");
    }
}

結果:

Demo現象解釋

我們Demo里的那兩個執行緒,從dump來看確實是死鎖了,那這個場景當時是怎么發生的呢?

執行緒1首先執行B.test(),于是會對B類做初始化,設定B的類狀態為being_initialized,接著去執行B的clinit方法,但是在clinit方法里要去呼叫A.test方法,理論上此時會對A做初始化并呼叫其test方法,但是就在設定完B的類狀態之后,執行其clinit里的A.test方法之前;

執行緒2卻執行了A.test方法,此時執行緒2會優先負責對A的初始化作業,即設定A類的狀態為being_initialized,然后再去執行A的clinit方法,此時執行緒1發現A的類狀態是being_initialized了,那執行緒1就認為有執行緒對A類正在做初始化,于是就等待了,而執行緒2同樣發現B的類狀態也是being_initialized,于是也開始等待,這樣就形成了兩個執行緒都在等待另一個執行緒完成初始化的情況,造成了類死鎖的現象,

五、總結

類加載的死鎖很隱蔽了,但是類初始化的死鎖更隱蔽,所以大家要謹記在類的初始化代碼里產生回圈依賴,另外對于jdk8的defalut特性也要謹慎,因為這會直接觸發介面的初始化導致更隱蔽的回圈依賴,

推薦閱讀:

JDK的sql設計不合理導致的驅動類初始化死鎖問題:https://blog.csdn.net/xichenguan/article/details/39578401

java虛擬機規范—初始化:https://blog.csdn.net/weixin_38233104/article/details/125251345

JVM常用命令:https://zhuanlan.zhihu.com/p/401563061

作者:京東物流 李鍵嶼

來源:京東云開發者社區

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/555152.html

標籤:Java

上一篇:驅動開發:內核ShellCode執行緒注入

下一篇:返回列表

標籤雲
其他(160965) Python(38226) JavaScript(25495) Java(18236) C(15237) 區塊鏈(8270) C#(7972) AI(7469) 爪哇(7425) MySQL(7251) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5875) 数组(5741) R(5409) Linux(5347) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4593) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2435) ASP.NET(2404) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1984) 功能(1967) HtmlCss(1966) Web開發(1951) C++(1940) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1881) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • 消失的死鎖:從 JSF 執行緒池滿到 JVM 初始化原理剖析

    在一次上線時,按照正常流程上線后,觀察了線上報文、介面可用率十分鐘以上,未出現例外情況,結果在上線一小時后突然收到jsf執行緒池耗盡的報警,并且該應用一共有30臺機器,只有一臺機器出現該問題,迅速下線該機器的jsf介面,恢復線上。然后開始排查問題。 ......

    uj5u.com 2023-06-14 10:31:40 more
  • 驅動開發:內核ShellCode執行緒注入

    還記得`《驅動開發:內核LoadLibrary實作DLL注入》`中所使用的注入技術嗎,我們通過`RtlCreateUserThread`函式呼叫實作了注入DLL到應用層并執行,本章將繼續探索一個簡單的問題,如何注入`ShellCode`代碼實作反彈Shell,這里需要注意一般情況下`RtlCreat... ......

    uj5u.com 2023-06-14 10:25:37 more
  • vs2022 wxWidgets

    # 下載 https://www.wxwidgets.org/downloads/ 下載壓縮包即可 ![image](https://img2023.cnblogs.com/blog/916065/202306/916065-20230614040303993-2082032985.png) # 編 ......

    uj5u.com 2023-06-14 08:34:12 more
  • Nexus3 重置 admin 賬號密碼

    # 問題背景 nexus3 的 admin 賬號密碼忘記了,需要重置。 # 環境說明 ``` nexus 基于 docker-compose 部署,版本 nexus3.26 docker 鏡像 sonatype/nexus3:3.26.1 ``` # 操作步驟 參考: https://support ......

    uj5u.com 2023-06-14 08:33:43 more
  • OpenFoam——自定義求解器

    ## 1、求解器 ### 1.1 復制原始碼 本案例以icoFoam為例,復制【openFOAM/OpenFOAM-9/applications/solvers/incompressible/icoFoam】檔案夾至run檔案夾下(我的是【openFOAM/mtl-9/run/solvers/inco ......

    uj5u.com 2023-06-14 08:33:09 more
  • C++面試八股文:什么是RAII?

    某日二師兄參加XXX科技公司的C++工程師開發崗位第13面: > 面試官:什么是`RAII`? > > 二師兄:`RAII`是`Resource Acquisition Is Initialization`的縮寫。翻譯成中文是資源獲取即初始化。 > > 面試官:`RAII`有什么特點和優勢? > > ......

    uj5u.com 2023-06-14 08:32:46 more
  • SpringBoot啟動類@SpringBootApplication注解

    在springBoot的啟動類中,提供了一個mai函式的程式入口,來啟動加載SpringBoot程式,那么注解@SpringBootApplication,通過原始碼可以看到,它相當于@ComponentScan + @EnableAutoConfiguration + @SpringBootConf ......

    uj5u.com 2023-06-14 08:32:35 more
  • 現代C++學習指南-方向篇

    C++是一門有著四十年歷史的語言,先后經歷過四次版本大升級(誕生、98、11、17(20),14算小升級)。每次升級都是很多問題和解決方案的取舍。了解這些歷史,能更好地幫助我們理清語言的發展脈絡。所以接下來我將借它的發展歷程,談一談我對它的理解,最后給出我認為比較合理的學習路線指南。 ### C++ ......

    uj5u.com 2023-06-14 08:32:28 more
  • celery筆記三之task和task的呼叫

    > 本文首發于公眾號:Hunter后端 > 原文鏈接:[celery筆記三之task和task的呼叫](https://mp.weixin.qq.com/s/AIobDZVDWV3r_XauvmkVKA) 這一篇筆記介紹 task 和 task 的呼叫。 以下是本篇筆記目錄: 1. 基礎的 task ......

    uj5u.com 2023-06-14 08:27:13 more
  • 現代C++學習指南-型別系統

    > 在前一篇,我們提供了一個方向性的指南,但是學什么,怎么學卻沒有詳細展開。本篇將在前文的基礎上,著重介紹下怎樣學習C++的型別系統。 ### 寫在前面 在進入型別系統之前,我們應該先達成一項共識——盡可能使用C++的現代語法。眾所周知,出于兼容性的考慮,C++中很多語法都是合法的。但是隨著新版本的 ......

    uj5u.com 2023-06-14 08:16:26 more