返回顶部
首页 > 资讯 > 精选 >.NET某消防物联网后台服务内存泄漏问题怎么解决
  • 856
分享到

.NET某消防物联网后台服务内存泄漏问题怎么解决

2023-06-30 18:06:19 856人浏览 泡泡鱼
摘要

这篇文章主要讲解了“.net某消防物联网后台服务内存泄漏问题怎么解决”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“.NET某消防物联网后台服务内存泄漏问题怎么解决”吧!一:背景1. 讲故事去

这篇文章主要讲解了“.net某消防物联网后台服务内存泄漏问题怎么解决”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“.NET某消防物联网后台服务内存泄漏问题怎么解决”吧!

一:背景

1. 讲故事

去年十月份有位朋友从微信找到我,说他的程序内存要炸掉了。。。截图如下:

.NET某消防物联网后台服务内存泄漏问题怎么解决

二:Windbg 分析

1. 托管还是非托管

这个是甄别内存问题的第一步,通过 !address -summary 和 !eeheap -GC 两个命令基本就可以断定。

0:000> !address -summary                             Mapping file section regions...Mapping module regions...Mapping PEB regions...Mapping TEB and stack regions...Mapping heap regions...Mapping page heap regions...Mapping other regions...Mapping stack trace database regions...Mapping activation context regions...--- Usage Summary ---------------- RgnCount ----------- Total Size -------- %ofBusy %ofTotalFree                                    237     7ffc`1e222000 ( 127.985 TB)           99.99%<unknown>                               594        3`b9b20000 (  14.901 GB)  95.96%    0.01%Heap                                    370        0`12a2a000 ( 298.164 MB)   1.88%    0.00%Image                                  1248        0`0ee5a000 ( 238.352 MB)   1.50%    0.00%Stack                                   315        0`06780000 ( 103.500 MB)   0.65%    0.00%Other                                    13        0`001d7000 (   1.840 MB)   0.01%    0.00%TEB                                     105        0`000d2000 ( 840.000 kB)   0.01%    0.00%PEB                                       1        0`00001000 (   4.000 kB)   0.00%    0.00%--- Type Summary (for busy) ------ RgnCount ----------- Total Size -------- %ofBusy %ofTotalMEM_PRIVATE                            1178        3`ce03D000 (  15.219 GB)  98.00%    0.01%MEM_IMAGE                              1409        0`0f6fd000 ( 246.988 MB)   1.55%    0.00%MEM_MAPPED                               59        0`04694000 (  70.578 MB)   0.44%    0.00%--- State Summary ---------------- RgnCount ----------- Total Size -------- %ofBusy %ofTotalMEM_FREE                                237     7ffc`1e222000 ( 127.985 TB)           99.99%MEM_COMMIT                             2326        3`c7543000 (  15.115 GB)  97.33%    0.01%MEM_RESERVE                             320        0`1a88b000 ( 424.543 MB)   2.67%    0.00%0:000> !eeheap -gcNumber of GC Heaps: 1generation 0 starts at 0x0000009902B57670generation 1 starts at 0x0000009902A56810generation 2 starts at 0x00000095318C1000ephemeral segment allocation context: (0x0000009902D724A8, 0x0000009902D724C0)         segment             begin         allocated         committed    allocated size    committed size...00000098FFBE0000  00000098FFBE1000  0000009902D724A8  0000009902D7D000  0x31914a8(51975336)  0x319c000(52019200)Large object heap starts at 0x00000095418C1000         segment             begin         allocated         committed    allocated size    committed size00000095418C0000  00000095418C1000  00000095475D8D98  00000095475D9000  0x5d17d98(97615256)  0x5d18000(97615872)Total Allocated Size:              Size: 0x398e6cbc8 (15450164168) bytes.Total Committed Size:              Size: 0x398e7b000 (15450222592) bytes.------------------------------GC Allocated Heap Size:    Size: 0x398e6cbc8 (15450164168) bytes.GC Committed Heap Size:    Size: 0x398e7b000 (15450222592) bytes.

从输出信息看,好家伙。。。进程提交内存是 15G, 托管堆差不多也是 15G,这就说明当前是相对简单的 托管内存泄漏。

2. 到底是什么在泄漏

要想知道到底是什么在泄漏,可以先到托管堆上看看有没有什么异常的对象,使用 !dumpheap -stat 命令。

0:000> !dumpheap -statStatistics:              MT    Count    TotalSize Class Name...00007ff8815d0f88  7260233    290409320 System.Collections.ArrayList00007ff8815e6830  7313696    326240826 System.String000000952fbbd2b0 12141398    509369998      Free00007ff880685cf0  7254983    928637824 System.Diagnostics.ProcessInfo00007ff88065f7d0  7256845   2031916600 System.Diagnostics.Process00007ff8815e6ea8  7391338   2230744600 System.Object[]00007ff88068fa70 186800748   8966435904 System.Diagnostics.ThreadInfo

从卦象上来看,真的太奇怪了,如果大家了解 Process 类,应该知道 ProcessInfo 和 ThreadInfo 都是挂在 Process 下的,而且 ThreadInfo 对象高达 1.8亿,真的太????????了,看样子程序是在不断的做 Process.Start 操作吧。

接下来要探究的问题是 ThreadInfo 到底正在被谁持有??? 可以挑几个看看它们的 !gcroot, 首尾法查了几个,都是没有引用根,如下所示:

0:000> !gcroot 0000009531e8f760Found 0 unique roots (run '!GCRoot -all' to see all roots).0:000> !gcroot 0000009531e8f670 Found 0 unique roots (run '!GCRoot -all' to see all roots).0:000> !gcroot 0000009531e8f378Found 0 unique roots (run '!GCRoot -all' to see all roots).

3. 无引用根为什么不被回收

既然对象没有引用根,为什么 GC 不回收它呢? 这里就需要谈经验了,在我之前分析的很多关于内存泄漏 的dump中,我都是从 生产端 找问题,貌似还没有出现一个从 消费端 找问题的案例,参考模型如下:

.NET某消防物联网后台服务内存泄漏问题怎么解决

生产者既然没问题,那消费端能有什么问题呢?大家可以想一想,托管堆的消费端起码有两个角色。

GC 肯定是不会出问题的,那就只能怀疑 Finalizer 线程出了什么问题,可以用 !t 命令把所有线程调出来看看。

0:000> !tThreadCount:      9566UnstartedThread:  0BackgroundThread: 88PendingThread:    0DeadThread:       9471Hosted Runtime:   no                                                                                                        Lock         ID OSID ThreadOBJ           State GC Mode     GC Alloc Context                  Domain           Count Apt Exception   0    1 40e18 000000952fbdfe50    26020 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     STA    2    2 41ce8 000000952fc0e4d0    2b220 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     MTA (Finalizer)    4    4 41cb4 000000954c324970    27220 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     STA    5    5 41cb8 000000954c36d1e0  2027220 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     STA    6    6 41c58 000000954c33f070  2027220 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     STA    7    7 41c38 000000954c33f840    27220 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     STA    8    8 41e0c 000000954c333580    27220 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     STA    9    9 41e2c 000000954c354440    27220 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     STA   10   10 41f24 000000954c355840    27220 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     STA ...XXXX 9446    0 000000957a233410  8039820 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     MTA (Threadpool Completion Port) XXXX 9447    0 0000009579f83e30  8039820 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     MTA (Threadpool Completion Port) XXXX 9450    0 000000957a46dcf0  8039820 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     MTA (Threadpool Completion Port) XXXX 9449    0 000000957967c6e0  8039820 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     MTA (Threadpool Completion Port) XXXX 9448    0 000000957aee0010  8039820 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     MTA (Threadpool Completion Port) XXXX 9452    0 00000095796824a0  8039820 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     MTA (Threadpool Completion Port) XXXX 9451    0 000000957af05df0  8039820 Preemptive  0000000000000000:0000000000000000 000000952fbd2e40 0     MTA (Threadpool Completion Port)

这不看不知道,一看吓一跳,当前进程有 9566 个,死线程高达 9471 个 ,以过往经验,这小子不用线程池,用 new Thread 咯, ???????????? ,吐槽结束,再看下 Finalizer 线程正在做什么,使用 ~2s & !dumpstack

0:002> ~2sntdll!NtWaitForSingleObject+0xa:00007ff8`8e220c8a c3              ret0:002> !dumpstackOS Thread Id: 0x41ce8 (2)Current frame: ntdll!NtWaitForSingleObject+0xaChild-SP         RetAddr          Caller, Callee0000009549e4e120 00007ff88b591118 KERNELBASE!WaitForSingleObjectEx+0x94, calling ntdll!NtWaitForSingleObject0000009549e4e1c0 00007ff88da3e334 combase!MTAThreadWaitForCall+0x54 [d:\9147\com\combase\dcomrem\channelb.cxx:5657], calling KERNELBASE!WaitForSingleObject0000009549e4e210 00007ff88d8fe089 combase!MTAThreadDispatchCrossApartmentCall+0x75 [d:\9147\com\combase\dcomrem\chancont.cxx:193], calling combase!MTAThreadWaitForCall [d:\9147\com\combase\dcomrem\channelb.cxx:5619]0000009549e4e240 00007ff88da3e13d combase!CrpcChannelBuffer::SendReceive2+0x64b [d:\9147\com\combase\dcomrem\channelb.cxx:4796], calling combase!MTAThreadDispatchCrossApartmentCall [d:\9147\com\combase\dcomrem\chancont.cxx:156]0000009549e4e2b0 00007ff88e1bb6f7 ntdll!RtlAllocateHeap+0xd7, calling ntdll!RtlpLowFragHeapAllocFromContext...0000009549e4f5d0 00007ff8827d79cd clr!ManagedThreadBase_DispatchOuter+0x75, calling clr!ManagedThreadBase_DispatchMiddle0000009549e4f5e0 00007ff8828601af clr!EEConfig::GetConfigDWord_DontUse_+0x3b, calling clr!EEConfig::GetConfiguration_DontUse_0000009549e4f660 00007ff8828574fa clr!FinalizerThread::FinalizerThreadStart+0x10a, calling clr!ManagedThreadBase_DispatchOuter0000009549e4f6a0 00007ff8827d55b9 clr!EEHeapFreeInProcessheap+0x45, calling kernel32!HeapFreeStub0000009549e4f700 00007ff882882e8f clr!Thread::intermediateThreadProc+0x860000009549e4f780 00007ff882882e6f clr!Thread::intermediateThreadProc+0x66, calling clr!_chkstk0000009549e4f7c0 00007ff88dcd13d2 kernel32!BaseThreadInitThunk+0x220000009549e4f7f0 00007ff88e2003c4 ntdll!RtlUserThreadStart+0x34

从堆栈信息看,原来是终结器线程卡死了,从 MTAThreadDispatchCrossApartmentCall方法看,貌似是 MTA 向 STA 做了一个调用,到这里有经验的朋友应该知道,这和 com 组件有关系了,也就是说 Finalizer 线程无法释放由 STA 线程创建的 COM 对象,那到底是哪一个线程创建了没有被合理释放的 COM 组件呢?

4. 寻找创建 COM 组件的线程

说实话,这个对COM组件不了解的话,很难找出答案,但天无绝人之路,当我回头看 线程列表 的时候,发现居然有 38 个 STA线程,截图如下:

.NET某消防物联网后台服务内存泄漏问题怎么解决

这里面肯定有问题,接下来抽一个线程看看调用栈如何。

0:004> !clrstack OS Thread Id: 0x41cb4 (4)        Child SP               IP Call Site000000954da1ee38 00007ff88e220c8a [HelperMethodFrame: 000000954da1ee38] System.Threading.Thread.SleepInternal(Int32)000000954da1ef30 00007ff88138c20a *** WARNING: Unable to verify checksum for mscorlib.ni.dllSystem.Threading.Thread.Sleep(Int32)000000954da1ef60 00007ff82322437f xxx.CFileLogTask.DoWork()000000954da1f160 00007ff8232234d6 xxx.CTask.InitStart()000000954da1f240 00007ff8813c31d3 System.Threading.ExecutionContext.RunInternal(System.Threading.ExecutionContext, System.Threading.ContextCallback, System.Object, Boolean)000000954da1f310 00007ff8813c3064 System.Threading.ExecutionContext.Run(System.Threading.ExecutionContext, System.Threading.ContextCallback, System.Object, Boolean)000000954da1f340 00007ff8813c3032 System.Threading.ExecutionContext.Run(System.Threading.ExecutionContext, System.Threading.ContextCallback, System.Object)000000954da1f390 00007ff8813bc812 System.Threading.ThreadHelper.ThreadStart()000000954da1f5e8 00007ff8827d6bb3 [GCFrame: 000000954da1f5e8] 000000954da1f938 00007ff8827d6bb3 [DebuggerU2MCatchHandlerFrame: 000000954da1f938]

接下来反编译下 xxx.CFileLogTask.DoWork() 方法看看它是如何被 Thread 执行的。

.NET某消防物联网后台服务内存泄漏问题怎么解决

到这里终于水落石出,罪魁祸首在 CurrThread.SetApartmentState(ApartmentState.STA); 这一句上,我也不知道为啥开个 Thread 还要给个 STA。。。

感谢各位的阅读,以上就是“.NET某消防物联网后台服务内存泄漏问题怎么解决”的内容了,经过本文的学习后,相信大家对.NET某消防物联网后台服务内存泄漏问题怎么解决这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

--结束END--

本文标题: .NET某消防物联网后台服务内存泄漏问题怎么解决

本文链接: https://lsjlt.com/news/330780.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • .NET某消防物联网后台服务内存泄漏问题怎么解决
    这篇文章主要讲解了“.NET某消防物联网后台服务内存泄漏问题怎么解决”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“.NET某消防物联网后台服务内存泄漏问题怎么解决”吧!一:背景1. 讲故事去...
    99+
    2023-06-30
  • .NET某消防物联网后台服务内存泄漏分析
    目录一:背景1. 讲故事二:Windbg 分析1. 托管还是非托管2. 到底是什么在泄漏3. 无引用根为什么不被回收4. 寻找创建 COM 组件的线程三:总结一:背景 1. 讲故事 ...
    99+
    2024-04-02
  • 怎么解决内存泄漏问题
    本篇内容介绍了“怎么解决内存泄漏问题”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!问题排查首先确定内存泄漏问题出现的时间,发现在该时间点的上...
    99+
    2023-06-16
  • drawimage内存泄漏问题怎么解决
    解决drawImage内存泄漏问题的方法如下:1. 及时释放资源:使用完image对象后,可以调用`image = null;`来手...
    99+
    2023-09-05
    drawimage
  • java内存泄漏问题怎么解决
    这篇文章主要介绍“java内存泄漏问题怎么解决”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“java内存泄漏问题怎么解决”文章能帮助大家解决问题。1、概念Java中的内存泄露是指不再使用的对象的内存...
    99+
    2023-06-30
  • Java的内存泄漏问题怎么解决
    本篇内容介绍了“Java的内存泄漏问题怎么解决”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一问题的提出Java的一个重要优点就是通过垃圾收...
    99+
    2023-06-03
  • android内存溢出和内存泄漏问题怎么解决
    Android内存溢出和内存泄漏是常见的问题,可以通过以下方法来解决:1. 使用内存分析工具:可以使用Android Studio自...
    99+
    2023-08-26
    android
  • 解决阿里云服务器内存升级后无法访问网页的问题
    1. 检查服务器配置 首先,确保你的服务器配置正确。检查服务器的内存升级是否成功,并确保服务器已经重新启动。如果服务器没有正确启动,你可以尝试重新启动服务器。 2. 检查网络连接 如果服务器配置正确,但仍然无法访问网页,那么可能是网络连接...
    99+
    2023-10-28
    阿里 无法访问 内存
  • 腾讯云服务器无法访问互联网怎么解决方法问题
    DNS 劫持 - 如果DNS 设置不正确,腾讯云服务器可能无法正常访问互联网。这可能是由于攻击者尝试通过劫持域名来访问腾讯云服务器的IP地址,从而导致服务器无法正常响应请求。通过更改DNS设置或联系腾讯云客服,可以解决这个问题。 网络连接...
    99+
    2023-10-27
    互联网 腾讯 解决方法
  • 云服务器内网穿透绑定域名怎么解决问题
    首先,要确保云服务器的安全设置得当。如果云服务器的安全设置不当,可能会导致用户的数据和文件被窃取或丢失。因此,要保证云服务器的安全设置得当,需要在云服务器的设置页面中设置一些安全选项,例如SSL证书、防火墙、数据加密等。 其次,要确保用户...
    99+
    2023-10-28
    解决问题 绑定 内网
  • 云服务器内网映射是什么意思啊怎么解决问题
    云服务器内网映射指的是在使用云服务器的客户端(例如Web App)上安装一个或多个远程服务器,通过该服务器提供的网络服务,可以使用互联网访问该服务器。 对于使用云服务器的客户端来说,只需要在安装云服务器的客户端时选择“内网映射”选项即可,...
    99+
    2023-10-27
    解决问题 内网 服务器
  • 云服务器内网映射是什么意思呀怎么解决问题
    云服务器内网映射指的是在使用云服务器的客户端(例如Web App)上安装一个或多个远程服务器(VPS),可以通过云服务器来存储、管理、安全地访问和使用其所存储的数据。 要解决云服务器内网映射的问题,可以考虑以下几个方面: 确认云服务器的...
    99+
    2023-10-27
    解决问题 内网 服务器
  • 云服务器内网映射是什么意思啊怎么解决的问题
    内网映射是云服务器的重要功能之一,它可以提高云服务器的可用性和性能。通过内网映射,云服务器可以将其虚拟网络拓扑映射到内部网络中,以便访问其他云服务器的内部资源。内网映射可以有效地减少云服务器网络的延迟,提高云服务器的响应速度和可用性。 云...
    99+
    2023-10-28
    内网 服务器
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作