丝袜人妻一区二区三区_少妇福利无码视频_亚洲理论片在线观看_一级毛片国产A级片

678改動日志看這里!記一次 .NET 某紡織工廠 MES系統(tǒng) API 掛死分析

一、背景

1.講故事

這個月中旬,一個朋友向我求助wx的程序線程份額很高,探討了如何解決。屏幕截圖如下:

說實話,和其他行業(yè)的程序員聊天還是很有趣的,可以交朋友,擴大自己的圈子。朋友說,因為這個bug,項目黃了。

哈哈,顧客好像不買。不能驗收。有害。

。。早找到我,這客戶不就撈回來啦,這也許就是技術(shù)的價值吧!

既然找到我,那就讓這個掛死問題徹底消失吧,上windbg說話。

二:Windbg 分析

1. 查看線程情況

既然朋友說線程高,那就從線程入手,用 !t 命令即可。

0:000> !t ThreadCount: 1006 UnstartedThread: 0 BackgroundThread: 1005 PendingThread: 0 DeadThread: 0 Hosted Runtime: no Lock DBG ID OSID ThreadOBJ State GC Mode GC Alloc Context Domain Count Apt Exception 0 1 10c8 00000000004D89A0 2a020 Preemptive 0000000000000000:0000000000000000 00000000002f1070 0 MTA 2 2 13c0 000000000031FF70 2b220 Preemptive 0000000000000000:0000000000000000 00000000002f1070 0 MTA (Finalizer) 3 3 12cc 000000000032B780 102a220 Preemptive 0000000000000000:0000000000000000 00000000002f1070 0 MTA (Threadpool Worker) 4 5 138c 000000000039E3C0 8029220 Preemptive 00000000B6D3CCA0:00000000B6D3D260 00000000002f1070 0 MTA (Threadpool Completion Port) 6 6 106c 0000000019E562A0 3029220 Preemptive 0000000000000000:0000000000000000 00000000002f1070 0 MTA (Threadpool Worker) 8 11 7f0 0000000019F8F9E0 20220 Preemptive 0000000000000000:0000000000000000 00000000002f1070 0 Ukn 9 1949 323c 000000009AA69E40 8029220 Preemptive 00000000B6BB8AD0:00000000B6BB94E0 00000000002f1070 0 MTA (Threadpool Completion Port) 10 1637 b3c 000000009AA1C260 8029220 Preemptive 00000000B6CD4220:00000000B6CD47E0 00000000002f1070 0 MTA (Threadpool Completion Port) 11 1947 223c 000000009ADB72E0 8029220 Preemptive 00000000B6D88D68:00000000B6D89550 00000000002f1070 0 MTA (Threadpool Completion Port) 12 1968 2e74 000000009AA1E330 8029220 Preemptive 00000000B6A8CD40:00000000B6A8D300 00000000002f1070 0 MTA (Threadpool Completion Port) ... 994 313 1fa4 000000009A81FFC0 8029220 Preemptive 00000000B6BFC1B8:00000000B6BFC410 00000000002f1070 0 MTA (Threadpool Completion Port) 995 1564 18ec 000000009A835510 8029220 Preemptive 00000000B6AC1ED0:00000000B6AC2490 00000000002f1070 0 MTA (Threadpool Completion Port) 996 1581 4ac 000000001C2E36E0 8029220 Preemptive 00000000B6C51500:00000000B6C51AC0 00000000002f1070 0 MTA (Threadpool Completion Port) 997 814 2acc 000000009A73B5E0 8029220 Preemptive 00000000B6D67BF8:00000000B6D683E0 00000000002f1070 0 MTA (Threadpool Completion Port) 998 517 25dc 000000009A838990 8029220 Preemptive 00000000B6D2CA10:00000000B6D2CFD0 00000000002f1070 0 MTA (Threadpool Completion Port) 999 670 2a10 000000001C2E4400 8029220 Preemptive 00000000B6CD0490:00000000B6CD0A50 00000000002f1070 0 MTA (Threadpool Completion Port) 1000 183 1704 000000009A81F930 8029220 Preemptive 00000000B6AE8670:00000000B6AE8C30 00000000002f1070 0 MTA (Threadpool Completion Port) 1001 117 1bcc 000000009A73BC70 8029220 Preemptive 00000000B6B92780:00000000B6B92D40 00000000002f1070 0 MTA (Threadpool Completion Port) 1002 1855 1d68 000000009A81E580 8029220 Preemptive 00000000B6B28460:00000000B6B28A20 00000000002f1070 0 MTA (Threadpool Completion Port) 1003 1070 2ef0 000000009A73C300 8029220 Preemptive 00000000B6B8F640:00000000B6B8FC00 00000000002f1070 0 MTA (Threadpool Completion Port) 1004 1429 210c 000000001C2E4A90 8029220 Preemptive 00000000B6D5F488:00000000B6D5FC70 00000000002f1070 0 MTA (Threadpool Completion Port) 1005 1252 2f38 000000009A838300 8029220 Preemptive 00000000B6A99240:00000000B6A99800 00000000002f1070 0 MTA (Threadpool Completion Port) 1006 1317 3118 000000001C2E5120 8029220 Preemptive 00000000B6DA3A30:00000000B6DA4440 00000000002f1070 0 MTA (Threadpool Completion Port) 1007 1837 3120 000000009A8375E0 8029220 Preemptive 00000000B6D38F10:00000000B6D394D0 00000000002f1070 0 MTA (Threadpool Completion Port) 1009 1964 2f64 000000009A81DEF0 1029220 Preemptive 0000000000000000:0000000000000000 00000000002f1070 0 MTA (Threadpool Worker)

可以看到當前有 1006 個線程,其中 1000 個是 Threadpool Completion Port,這么多IO線程卡死也是第一次遇到,。

說實話,看到 Threadpool Completion Port 我就想到這是一個異步操作的回調(diào),那為什么會有這么多IO線程被卡死 ? 要想尋找答案,抽個線程看一下。

0:1000> ~1000s ntdll!NtNotifyChangeDirectoryFile+0xa: 00000000`77c7a75a c3 ret 0:1000> !clrstack OS Thread Id: 0x1704 (1000) Child SP IP Call Site 00000000A99FF4C0 0000000077c7a75a [InlinedCallFrame: 00000000a99ff4c0] Interop+Kernel32.ReadDirectoryChangesW, Byte[], Int32, Boolean, Int32, Int32 ByRef, Sy*, IntPtr) 00000000A99FF4C0 000007fe8e87bd20 [InlinedCallFrame: 00000000a99ff4c0] Interop+Kernel32.ReadDirectoryChangesW, Byte[], Int32, Boolean, Int32, Int32 ByRef, Sy*, IntPtr) 00000000A99FF470 000007fe8e87bd20 DomainBoundILS, Byte[], Int32, Boolean, Int32, Int32 ByRef, Sy*, IntPtr) 00000000A99FF560 000007fef19dab6e Sy(AsyncReadState) [E:\A\_work\322\s\corefx\src\Sy\src\System\IO\FileSy @ 141] 00000000A99FF5E0 000007fef19dae6c Sy(UInt32, UInt32, Sy*) [E:\A\_work\322\s\corefx\src\Sy\src\System\IO\FileSy @ 227] 00000000A99FF630 000007feedbe0af9 Sy, Sy, Sy) [E:\A\_work\191\s\src\mscorlib\shared\System\Threading\Execu @ 167] 00000000A99FF6B0 000007feede094dc Sy(UInt32, UInt32, Sy*) [E:\A\_work\191\s\src\mscorlib\src\System\Threading\Overla @ 108] 00000000A99FF7F0 000007feee359ed3 [GCFrame: 00000000a99ff7f0] 00000000A99FF9D0 000007feee359ed3 [DebuggerU2MCatchHandlerFrame: 00000000a99ff9d0]

我去,又見 FileSystemWatcher ,追這個系列的朋友應(yīng)該知道,我上個月分析了一篇 記一次 .NET 某流媒體獨角獸 API 句柄泄漏分析 ,這其中就是 FileSystemWatcher 導(dǎo)致的 文件句柄 爆高,他的形成原因是 定時刷新 appsetttings + reloadOnChange=true 所致,這世界真小,莫不會撞車了。。。 接下來我們重點關(guān)注一下它。

2. 探究 FileSystemWatcher

要想進一步分析,先用 !dso 命令看一下當前的線程棧對象。

0:1000> !dso OS Thread Id: 0x1704 (1000) RSP/REG Object Name 00000000A99FF508 00000000263285d8 Sy[] 00000000A99FF518 00000000242aeb10 Sy 00000000A99FF560 00000000242ae1b0 Micro 00000000A99FF568 00000000242aeaa8 Sy 00000000A99FF578 00000000242aeb10 Sy 00000000A99FF5E0 00000000242a8538 Sy 00000000A99FF5E8 00000000242aea10 Sy+AsyncReadState 00000000A99FF608 00000000242aea10 Sy+AsyncReadState 00000000A99FF610 0000000023206e30 Sy 00000000A99FF618 0000000001032928 Sy 00000000A99FF630 00000000242a8538 Sy 00000000A99FF678 00000000b6a69a40 Sy 00000000A99FF688 00000000242aeb10 Sy 00000000A99FF690 0000000023206e30 Sy 00000000A99FF6C0 0000000021fa55d8 Sy 00000000A99FF6C8 000000002052e6e0 Sy 00000000A99FF7E0 000000000560d2b0 Sy

由于線程棧上的對象是向小擴展的,所以看那個最小地址上的 Sy[] 內(nèi)容就知道當前回調(diào)的是啥啦,截圖如下:

經(jīng)過上一役的分析經(jīng)驗,到這里我基本就搞清楚了,這又是一個不斷構(gòu)建 ConfigurationRoot 時配了 reloadOnChange: true 的經(jīng)典案例,它的后果會導(dǎo)致內(nèi)存中新增大量的 FileSystemWatcher 和 ConfigurationRoot 無法釋放,而誘發(fā)點就是上圖中的日志文件的不斷變更導(dǎo)致的海量回調(diào)函數(shù)觸發(fā)的卡死案例,具體詳情... 請聽我慢慢分解,先驗證下這兩個類在托管堆上的個數(shù)。

0:1000> !dumpheap -stat -type FileSystemWatcher Statistics: MT Count TotalSize Class Name 000007fe8ed5bc90 2 160 Sy`2[[Sy, Sy],[Sy, Sy]] 000007fe8e9f11a0 34480 1930880 Sy+AsyncReadState 000007fe8e9d69c8 34480 4137600 Sy Total 68962 objects 0:1000> !dumpheap -stat -type ConfigurationRoot Statistics: MT Count TotalSize Class Name 000007fe8e9f1e70 34480 827520 Micro;>c__DisplayClass2_0 000007fe8e999560 34480 1103360 Micro Total 68960 objects

果不其然,托管堆有 3.4w 的 FileSystemWatcher 和 ConfigurationRoot,接下來就得和朋友溝通了。

3. 到底是什么代碼引起的?

詢問朋友為什么會有 3.4w 的 ConfigurationRoot 對象,理論上一個程序只會有一個,根據(jù)這些信息,朋友很快就找到了問題代碼,截圖如下:

就是因為上圖中的 reloadOnChange: true 讓底層構(gòu)建了 FileSystemWatcher 對 a 的實時監(jiān)控,從而導(dǎo)致內(nèi)存出現(xiàn)了 3.4w 的對象無法釋放。

三:為什么日志變更會造成程序卡死

1. 當初的困惑

其實我當初分析到這里的時候也是很困惑的,就算內(nèi)存中有 3.4w 的 FileSystemWatcher,那也只是對 a 的監(jiān)控,只要這個文件不變動就不會觸發(fā) 3.4w 的回調(diào),不是嗎? 可當我分析完 ConfigurationRoot 源碼之后,我發(fā)現(xiàn)自己真tmd的天真。

2. 從源碼中尋找答案

首先我們看看 FileSystemWatcher 到底監(jiān)控的是啥? 可以在它的構(gòu)造函數(shù)中設(shè)一個斷點,如下圖所示:

很明顯的看到,它 watch 的是程序根目錄,這就能解釋為什么日志文件有變更就會觸發(fā)文件變更的回調(diào)函數(shù),為了驗證,我可以在 ReadDirectoryChangesCallback 方法中下一個斷點,再丟一個日志文件到根目錄,看是否能觸發(fā)就知道了。。。 截圖如下:

回到本案例,也就是說一旦有日志變動,就會觸發(fā) 3.4w 個回調(diào)函數(shù),如果變動100次,就會觸發(fā) 340w 次回調(diào),而日志變更不停止,自然就會因為海量的回調(diào)把程序搞死。。。對吧。。。

四:總結(jié)

本次事故可能是由于朋友偷懶,沒有將 Configuration 或 IOptions 注下去,而是采用重新構(gòu)建 ConfigurationRoot 的方式獲取 ConnectionString,并錯誤的配置 reloadOnChange: true, 導(dǎo)致IO線程無法及時處理由于日志文件的變更導(dǎo)致的海量回調(diào)函數(shù),進而導(dǎo)致程序掛死。

知道整個來龍去脈之后,優(yōu)化措施就很簡單了,提供兩種方法。

  1. 將 reloadOnChange: true 改成 reloadOnChange: false 。
  2. 想辦法將 Configuration 注入到 DataBaseConfig 類中,做成靜態(tài)變量也行 。

最后上一個彩蛋,朋友太客氣了。

1.《678改動日志看這里!記一次 .NET 某紡織工廠 MES系統(tǒng) API 掛死分析》援引自互聯(lián)網(wǎng),旨在傳遞更多網(wǎng)絡(luò)信息知識,僅代表作者本人觀點,與本網(wǎng)站無關(guān),侵刪請聯(lián)系頁腳下方聯(lián)系方式。

2.《678改動日志看這里!記一次 .NET 某紡織工廠 MES系統(tǒng) API 掛死分析》僅供讀者參考,本網(wǎng)站未對該內(nèi)容進行證實,對其原創(chuàng)性、真實性、完整性、及時性不作任何保證。

3.文章轉(zhuǎn)載時請保留本站內(nèi)容來源地址,http://f99ss.com/gl/2072880.html

上一篇

【678改動日志】Flink 最佳實踐之使用 Canal 同步 MySQL 數(shù)據(jù)至 TiDB

下一篇

dnf2019春節(jié)套專題之DNF:金秋裝扮男職業(yè)時裝居然這么帥

【678改動日志】Flink 最佳實踐之使用 Canal 同步 MySQL 數(shù)據(jù)至 TiDB

【678改動日志】Flink 最佳實踐之使用 Canal 同步 MySQL 數(shù)據(jù)至 TiDB

678改動日志相關(guān)介紹,一.背景介紹 本文介紹了將MySQL中的數(shù)據(jù)通過Binlog Canal導(dǎo)入Kafka,然后由Flink消耗的情況。 為了能夠快速的驗證整套流程的功能性,所有的組件都以單機的形式部署。如果手上的物理資...

【678改動日志】專題形象一落千丈的徐睿知 韓國演藝圈歷代勁爆 爭議連環(huán)爆

【678改動日志】專題形象一落千丈的徐睿知 韓國演藝圈歷代勁爆 爭議連環(huán)爆

678改動日志相關(guān)介紹,安靜的外表,聲音低,有磁性的她有望成為一線女演員。 她就是最近爭議連環(huán)爆的徐睿知。1990年4月6日出生于首爾過去曾在街上被星探看中。但因為覺得自己聲音低沉比較適合當主播,因此拒絕了試鏡邀請,之后她到...

【678改動日志】2月8日·貴州要聞及抗擊肺炎快報

【678改動日志】2月8日·貴州要聞及抗擊肺炎快報

678改動日志相關(guān)介紹,每天廣播 貴州:2020年2月7日12-24時,全省新型新冠病毒感染肺炎新增確診病例8例,出院患者0例。 其中:新增確診病例中,遵義市3例,畢節(jié)市4例,黔西南州1例。 截至2月7日24時,全省累計報告...

【678改動日志】專題網(wǎng)絡(luò)流量與Agent_Drable惡意程序深度分析

【678改動日志】專題網(wǎng)絡(luò)流量與Agent_Drable惡意程序深度分析

678改動日志相關(guān)介紹,在回顧最近的一些網(wǎng)絡(luò)異常時,發(fā)現(xiàn)了使用DNS隧道與C2通信的攻擊組織,并將其命名為“Cold River”。破譯受害者和C2的流量,發(fā)現(xiàn)攻擊者使用的復(fù)雜誘餌文件,與其他未知樣本連接,并發(fā)現(xiàn)攻擊者使用的...

678改動日志專題之必收藏 | 寬帶錯誤碼含義及處理方法大全

678改動日志專題之必收藏 | 寬帶錯誤碼含義及處理方法大全

678改動日志相關(guān)介紹,文章有點長。但是很耐用! 更多寬帶信息,微信搜索“寬帶軍”,寬帶軍洗干凈了等著你! 最近發(fā)現(xiàn)很多小朋友給我留言 不是為了找對象 在找對象的路上。 已經(jīng)有錯誤了。 . . . 寬帶錯誤 今天寬帶軍給大家...

【678改動日志】專題連接寬帶上網(wǎng)時,你遇到過這些錯誤代碼嗎?

【678改動日志】專題連接寬帶上網(wǎng)時,你遇到過這些錯誤代碼嗎?

678改動日志相關(guān)介紹,想打開電腦瀏覽器上網(wǎng),無法打開未知網(wǎng)頁,顯示601這樣的數(shù)字,或者寬帶連接瀏覽器也可以顯示這樣的數(shù)字。不要著急。確認寬帶連接是否正常,除了不欠費的情況外,今天筆者將告訴大家常用錯誤代碼的解決方案。 6...