国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

記一次 .NET 某工廠無人車調度系統 線程爆高分析

開發 前端
從卦中可以看到大概有12w的積壓。上面就是我的完整分析思路,最后就是告訴朋友最好的辦法就是去掉多余累贅的 SemaphoreSlim? ,直接用同步的方式執行 Interlocked.Increment(ref serial) 即可,簡單粗暴。

一:背景

1. 講故事

前些天有位朋友找到我,說他程序中的線程數爆高,讓我幫忙看下怎么回事,這種線程數爆高的情況找問題相對比較容易,就讓朋友丟一個dump給我,看看便知。

二:為什么會爆高

1. 查看托管線程

別人說的話不一定是真,得自己拿數據出來說話,可以用  !t 命令觀察一下便知。

0:000> !t
ThreadCount:      4683
UnstartedThread:  0
BackgroundThread: 4663
PendingThread:    0
DeadThread:       19
Hosted Runtime:   no
                                                                                                            Lock  
 DBG   ID     OSID ThreadOBJ           State GC Mode     GC Alloc Context                  Domain           Count Apt Exception
   0    1     cc44 00000268048778C0  202a020 Preemptive  0000000000000000:0000000000000000 00000268048c6d50 -00001 MTA 
   ...
4670 4679     51bc 0000026D143F0420  302b220 Preemptive  0000000000000000:0000000000000000 00000268048c6d50 -00001 MTA (Threadpool Worker) 
4671 4680     3a68 0000026D143F52E0  302b220 Preemptive  0000000000000000:0000000000000000 00000268048c6d50 -00001 MTA (Threadpool Worker) 
4672 4681     337c 0000026D143F1140  302b220 Preemptive  0000026A88AAF5B8:0000026A88AB08D0 00000268048c6d50 -00001 MTA (Threadpool Worker) 
4673 4682    188d4 0000026D143F0AB0  302b220 Preemptive  000002698881A760:000002698881C0B8 00000268048c6d50 -00001 MTA (Threadpool Worker) 
4674 4683     4bcc 0000026D143EF700  302b220 Preemptive  0000026B889C4488:0000026B889C5E18 00000268048c6d50 -00001 MTA (Threadpool Worker)

從卦中信息看確實有 4600+ 的線程,說明確實存在問題,接下來用 ~*e !clrstack 觀察每一個線程都在做什么,線程太多沒法全部輸出完畢,不過很容易的看到有大量的線程卡在 RoutingService.Push 上,截圖如下:

圖片圖片

接下來就是觀察下這個 Push 方法的邏輯,發現卡死在 Result 上,整理后的代碼大概如下:

private readonly SemaphoreSlim slim = new SemaphoreSlim(1, 1);

public void Push(string xxx, xxx xxx)
{
    int num = (xxx.Serial = GetSerial().Result);
}

private async Task<int> GetSerial()
{
    await slim.WaitAsync();
    try
    {
        Interlocked.Increment(ref serial);
    }
    finally
    {
        slim.Release();
    }
    return serial;
}

上面的代碼看起來挺奇葩的,為什么 GetSerial() 中不直接用 Interlocked.Increment() 呢?套一個 SemaphoreSlim 顯得非常多余。

先不管多余不多余,既然 Result 得不到值,就說明這個異步方法得不到完成,那為什么得不到完成呢?

2. 為什么異步得不到完成

熟悉 SemaphoreSlim.WaitAsync() 的朋友應該知道,這里涉及不到異步IO,所以這個是假異步,本質上就是動態生成了一個串聯的 Task<bool>,要想知道得不到完成的根本原因,還得要挖一挖此時的 slim 信號量情況。

0:000> !do 000002690664b5a0
Name:        System.Threading.SemaphoreSlim
MethodTable: 00007ff894e56fc0
EEClass:     00007ff894e3f230
Tracked Type: false
Size:        64(0x40) bytes
File:        D:\xxx\System.Private.CoreLib.dll
Fields:
              MT    Field   Offset                 Type VT     Attr            Value Name
00007ff8948094b0  4000c2e       28         System.Int32  1 instance                0 m_currentCount
00007ff8948094b0  4000c2f       2c         System.Int32  1 instance                1 m_maxCount
00007ff8948094b0  4000c30       30         System.Int32  1 instance                0 m_waitCount
00007ff8948094b0  4000c31       34         System.Int32  1 instance                0 m_countOfWaitersPulsedToWake
00007ff8962871e0  4000c32        8 ...Private.CoreLib]]  0 instance 000002690664b5e0 m_lockObjAndDisposed
00007ff894e555f0  4000c33       10 ....ManualResetEvent  0 instance 0000000000000000 m_waitHandle
00007ff894e57870  4000c34       18 ...horeSlim+TaskNode  0 instance 0000026b86919a30 m_asyncHead
00007ff894e57870  4000c35       20 ...horeSlim+TaskNode  0 instance 0000026b889c4378 m_asyncTail
00007ff894a4a1f0  4000c36      888 ...Private.CoreLib]]  0   static 00000268864f83a0 s_cancellationTokenCanceledEventHandler

從卦中看當前的 m_currentCount=0,表明當前的信號量被消費完了,所以其他的線程都在等待就能很好理解,接下來的問題是那個從 1->0 的持有線程為什么不歸還?這個就比較難搞了,可以從如下兩個思路思考:

  • 觀察 Result

首先懷疑是不是 Result 引發的死鎖,用 !eeversion 看了下是 asp.net core ,并沒有所謂的同步上下文,所以這個問題不存在。

0:000> !eeversion
6.0.2023.32017 free
6,0,2023,32017 @Commit: a08d9ce2caf02455c0b825bcdc32974bdf769a80
Server mode with 8 gc heaps
SOS Version: 7.0.8.30602 retail build
  • 觀察代碼

因為 SemaphoreSlim 并不記錄持有線程,windbg 在這里就起不到很好的效果,不過仔細閱讀代碼,發現應該將 await slim.WaitAsync(); 放到 try 中更合理一點,否則無法保證 WaitAsync 和 Release 一定是成雙成對的,截圖如下:

圖片圖片

3. 什么時候開始阻塞的

仔細觀察這個 GetSerial 方法,看看里面的 serial 值就知道大概是進行到哪一步才出的問題。

0:4674> !DumpObj /d 000002690664b258
Name:        xxx.RoutingService
MethodTable: 00007ff895283ed0
EEClass:     00007ff89526ae08
Tracked Type: false
Size:        112(0x70) bytes
File:        D:\xxx\xxx.dll
Fields:
              MT    Field   Offset                 Type VT     Attr            Value Name
...
00007ff894e56fc0  4000214       48 ...ing.SemaphoreSlim  0 instance 000002690664b5a0 slim
00007ff8948094b0  4000215       60         System.Int32  1 instance             9061 serial

從卦中看已經自增到了 9061 ,然后因為某種原因導致wait 和 release 不匹配了,像這種情況線程池也會有大量的任務積壓,可以用 !tp 觀察下。

0:4674> !tp
logStart: 33
logSize: 200
CPU utilization: 22 %
Worker Thread: Total: 4652 Running: 4652 Idle: 0 MaxLimit: 32767 MinLimit: 8
Work Request in Queue: 0
--------------------------------------
Number of Timers: 1
--------------------------------------
Completion Port Thread:Total: 2 Free: 2 MaxFree: 16 CurrentLimit: 2 MaxLimit: 1000 MinLimit: 8

細心的朋友會發現這里的 Work Request in Queue: 0 ,既然是 0 何來積壓?其實這是 sos 的bug,我們需要自己到線程池隊列中提取,從當前的線程棧上尋找 ThreadPoolWorkQueue 對象即可。

0:4674> !dso
OS Thread Id: 0x4bcc (4674)
000000EF384FF5C8 0000026b06544848 System.Threading.ThreadPoolWorkQueue

0:4674> !DumpObj /d 0000026b06544848
Name:        System.Threading.ThreadPoolWorkQueue
MethodTable: 00007ff894e59d80
EEClass:     00007ff894ee01d0
Tracked Type: false
Size:        168(0xa8) bytes
File:        D:\xxx\System.Private.CoreLib.dll
Fields:
              MT    Field   Offset                 Type VT     Attr            Value Name
00007ff89476bf38  4000c61       18       System.Boolean  1 instance                0 loggingEnabled
00007ff89476bf38  4000c62       19       System.Boolean  1 instance                0 _dispatchTimeSensitiveWorkFirst
00007ff89637fc20  4000c63        8 ...Private.CoreLib]]  0 instance 0000026b065448f0 workItems
00007ff89637fe00  4000c64       10 ...Private.CoreLib]]  0 instance 0000026b06544930 timeSensitiveWorkQueue
00007ff894e59d10  4000c65       20 ...acheLineSeparated  1 instance 0000026b06544868 _separated

0:4674> !ext dcq 0000026b065448f0
System.Collections.Concurrent.ConcurrentQueue<System.Object>
   1 - dumpobj 0x0000026806c782f8
...
119419 - dumpobj 0x000002690a097658
119420 - dumpobj 0x000002690a097810
119421 - dumpobj 0x000002690a0981a8
---------------------------------------------
119421 items

從卦中可以看到大概有12w的積壓。上面就是我的完整分析思路,最后就是告訴朋友最好的辦法就是去掉多余累贅的 SemaphoreSlim ,直接用同步的方式執行 Interlocked.Increment(ref serial) 即可,簡單粗暴。

三:總結

這次線程爆高的事故原因還是挺有意思的,用了一個雙同步來獲取 serial 值,感覺像是一種聰明反被聰明誤,代碼一定要簡單粗暴,代碼越少bug越少。

責任編輯:武曉燕 來源: 一線碼農聊技術
相關推薦

2024-08-08 11:21:01

2021-10-27 07:30:32

.NETCPU論壇

2023-05-12 17:42:22

CPUMES系統

2022-10-24 07:48:37

.NETCPUGC

2024-03-15 15:15:53

.NETCPU系統

2021-05-17 07:43:06

Web站 CPU.NET

2023-07-31 22:29:20

CPU.NETAPI

2024-12-31 09:36:06

2021-04-21 07:38:41

CPU游戲站程序

2024-06-13 17:09:55

2023-05-08 08:25:52

2024-03-28 12:56:36

2023-04-06 10:52:18

2023-03-26 20:24:50

ERP網站系統

2024-03-26 00:44:53

.NETCIM系統

2024-07-01 13:00:24

.NET網絡邊緣計算

2024-07-09 11:51:20

Windows線程池源碼

2022-01-17 21:28:36

管理系統.NET

2025-10-29 01:11:00

.NET系統windows

2023-06-29 17:55:00

.NET日志WinDbg
點贊
收藏

51CTO技術棧公眾號

久久综合视频网| 亚洲少妇中出一区| 久久国产精品99国产精| 天堂社区在线视频| 国产精品av久久久久久麻豆网| 精品国产伦一区二区三区免费 | 日本1区2区| 国产精品久久777777毛茸茸| xxx成人少妇69| 色呦呦呦在线观看| 亚洲视频小说图片| 久久av秘一区二区三区| 精品久久综合| 色综合五月天导航| 4438x成人网全国最大| 国产精品久久久久影院色老大| 日本公妇乱淫免费视频一区三区| 精品国产影院| 亚洲国产欧美自拍| 欧美美乳在线| 99re热视频精品| 欧美日韩国产免费一区二区三区| 一区二区三区视频免费视频观看网站 | 国内自拍视频一区| 免费看欧美美女黄的网站| 久久免费精品日本久久中文字幕| 日本无删减在线| 一区二区三区在线高清| 亚洲精品蜜桃久久久久久| 亚洲神马久久| 国产精品一区二区性色av| 精品自拍视频| 欧美v国产在线一区二区三区| 一级视频在线免费观看| 国产欧美日韩综合精品一区二区| 视频一区二区在线| 欧美777四色影| 情事1991在线| 国产精品亲子伦av一区二区三区| 日韩欧美黄色影院| 91caoporn在线| 午夜精品久久久久久久| 99re在线视频播放| 国产视频不卡一区| 中国黄色录像片| 亚洲欧美日韩国产一区| 91精品免费| 午夜影院欧美| 成人免费淫片视频软件| 欧美日韩中文一区二区| 91po在线观看91精品国产性色| 免费观看欧美大片| 亚洲美女性视频| 男人av在线播放| 亚洲国产精品中文| 3344国产永久在线观看视频| 精品欧美一区二区三区精品久久| 欧洲美女少妇精品| 欧美日韩另类一区| 首播影院在线观看免费观看电视| 亚洲少妇30p| av免费看大片| 一区二区三区丝袜| 性感美女激情视频在线观看| 五月天视频一区| 日韩精品系列| 欧美精品自拍偷拍| 变态调教一区二区三区| 亚洲级视频在线观看免费1级| 91超碰国产在线| 337p日本欧洲亚洲大胆色噜噜| 色呦呦呦在线观看| 精品视频在线导航| julia一区二区三区中文字幕| 高清不卡一区二区三区| 日韩欧美在线免费观看视频| 国产乱对白刺激视频不卡| 999一区二区三区| 不卡在线视频中文字幕| 蜜臀av无码一区二区三区| 欧美寡妇偷汉性猛交| 亚洲欧美偷拍三级| 亚洲成人最新网站| 日本高清在线观看| 日本美女高潮视频| 在线观看一级片| a级影片在线| 国产九九在线| 欧美3p视频在线观看| 一级特黄妇女高潮| 国产视频久久久久| 亚洲电影av| 成人黄色免费观看| 免费99热在线观看| 91免费看国产| 久久免费精品国产久精品久久久久| 国产美女免费观看| 加勒比海盗1在线观看免费国语版| 欧美激情xxxx性bbbb| 日本一区二区三区视频视频| 精品日本12videosex| 免费亚洲电影| 成人动漫在线观看视频| 国产精品青青在线观看爽香蕉| 精品日本美女福利在线观看| 久久久久久久久一| 免费日韩av片| 国产精品伦理| 黄色av资源| 国产成人精品在线视频| 一区二区三区黄色| 精品久久久久久久大神国产| 国产91对白在线观看九色| 伊人网在线播放| 九色91国产| 亚洲男人天堂手机在线| 99精品久久只有精品| 狂野欧美xxxx韩国少妇| 97cao在线| 免费的一级黄色片| 一级一片免费播放| 中文字幕精品网| 99re亚洲国产精品| 亚洲人成免费| 精品一区二区三区在线| 啊啊啊国产视频| 亚洲欧美另类在线观看| 夜夜嗨网站十八久久| 麻豆传媒视频在线| 97视频在线观看视频免费视频 | 中文字幕久热精品视频在线| 99久久精品99国产精品| 69堂免费精品视频在线播放| 在线国产1区| 日本午夜激情视频| 国产一区视频观看| 国产成人精品免费视频| 在线视频一区二区| 日韩视频在线永久播放| 亚洲欧洲综合另类| 波多野结衣亚洲一区| 黄色免费成人| 日本.亚洲电影| 午夜av在线播放| 在线观看国产麻豆| 亚洲精品无人区| 欧美一区二区三区不卡| 亚洲天堂精品在线观看| 国产日产欧美精品一区二区三区| 久久久久国产精品一区二区| 任你躁在线精品免费| 久久av网站| 欧美一级片网址| 欧美影院精品| 97一区二区国产好的精华液| 日本蜜桃在线观看视频| 一区二区三区短视频| 日本在线观看高清完整版| av黄色在线| 欧美色图天堂| 色屁屁www国产馆在线观看| 最新国产在线观看| 黄视频免费在线看| 亚洲a∨精品一区二区三区导航| 污片视频在线免费观看| 污影院在线观看| 亚洲欧洲自拍| www 久久久| 国产精品探花在线观看| 亚洲大全视频| 99成人精品| 日韩成人av影视| 国产日产欧产精品推荐色| 日韩美女视频一区| 欧美性视频一区二区三区| 亚洲精品一区二区在线观看| 一个人www欧美| 91久久中文字幕| 色综合久久av| 美女网站视频黄色| 人操人视频在线观看| 日韩另类在线| 开心激情综合| 在线不卡欧美| 久久精品一区二区三区四区| 富二代精品短视频| 亚洲国产第一页| 国产精品亚洲自拍| 亚洲欧美日产图| av三级影院| 国精产品一区一区三区mba下载| 亚洲免费一区| 日本人妖一区二区| 亚洲一区二区三区中文字幕在线 | 精品国产一区二区三区四区四| 日韩视频―中文字幕| 国产精品精品软件视频| 日韩欧美在线电影| 老司机午夜激情| 欧美特大特白屁股xxxx|