Visual Studio 高级调试-Dump分析

news2025/1/10 6:03:10

Dump

dump指转储,一般用来创建进程快照。它可以在不停止应用的情况下,直接将模块列表、线程列表、堆栈信息、异常信息、句柄信息等所有内存信息保存下来,帮助开发者分析生产环境问题等。

这篇博客主要介绍dotnet-dump的使用以及如何在Visual Studio中进行dump分析。

dotnet-dump

dotnet-dump是用于收集转储的跨平台命令行工具, Visual Studio和windbg也具有转储收集功能。

使用命令行安装dotnet-dump。

dotnet tool install --global dotnet-dump

dotnet-dump ps

dotnet-dump 只能访问托管代码, 使用dotnet-dump ps命令可以查看可以收集转储的 dotnet 进程。

dotnet-dump collect

使用dotnet-dump collect收集转储信息并保存到本地。

dotnet-dump collect -p 16036 -o C:\Users\mahua\Desktop\dump\test.dmp --type full

其中type按照收集数据范围由小到大分为:mini,triage, heap,full。

triage和heap转储数据范围一致,但triage去除了个人用户信息,如路径密码等。

如果在Windows环境转储时提示访问输出路径被拒绝,则需要提升dotnet-tools的运行权限,默认是到C:User\%User%\.dotnet\路径下,提升用户对tools文件夹的访问权限。

dotnet-dump analyze

dotnet-dump analyze命令非常繁琐,建议放弃,我们可以直接把dump文件拖到Visual Studio里进行分析。

Dump分析

为了方便模拟各种应用场景,我们可以使用dotnet官方提供的代码示例:DiagnosticScenarios · dotnet/samples。代码中包含了几个简单api模拟了死锁,内存飙升,内存泄漏,CPU飙升等场景。

我们将程序运行起来,通过swagger页面调用相应的接口来测试dump分析。

死锁分析

★运行诊断分析★

通过swagger页面调用死锁接口,接口会因为死锁而不返回结果,抓取dump文件,拖入Visual Studio进行分析。

点击运行诊断分析,可以看到Visual Studio直接给出了分析结果:检测到已死锁线程。

★使用调试窗口★

此时Visual Studio处于调试状态,而且dump文件包含了当时的堆栈、线程等信息,所以我们可以使用调试模式下的各种监视窗口来分析问题,详细信息可以看我上一篇博客:Visual Studio 高级调试-代码调试,这里我们主要查看并行堆栈窗口。

可以看到从左边起,前3个线程254,11,35都在等待锁的持有者25180,而线程25180则在等待另一个锁的持有者线程6248,线程6248也在等待锁的持有者25180。

那么问题就找出来了:线程25180和线程6248各持有一个锁,并且都在等待对方将锁释放出来,因此产生了死锁。

将鼠标放置在堆栈帧上查看提示,双击图示堆栈帧将来到线程6248发生死锁的位置,可以看到这里线程6248已经获得了锁o1,正在请求锁: o2。

此时锁o2应该是被线程25180所持有,而它正在请求锁o1,我们点击倒数第2个堆栈帧,可以来到线程25180死锁的地方,从而证明这一点。

堆栈信息遵守"后来者居上"原则,最近发生的会放在最上面。

★死锁的堆栈分析原理★

虽然Visual Studio非常智能的给出了结论:存在死锁。但我们仍有必要知道是如何通过堆栈分析得出这一结论的。

  1. 首先发现大多数线程共享一个公共调用堆栈

  1. 该调用堆栈调用了某方法,而该方法又调用了 Monitor.ReliableEnter。这表示线程正等待获取某个锁。

  1. 查看同步块表(不懂同步块的同学看这里C#引用类型实现原理),确定该锁已经被其他线程所持有。

  1. 进而分析得出两个线程分别持有一个锁,并且都在等待获取对方的锁,因此判定存在死锁。

内存飙升分析

调用接口模拟20秒内,内存使用升高的场景。同样的,我们使用dump转储并拖入Visual Studio进行分析。

诊断分析

对于内存飙升这种情况,诊断分析不会直接给出结论,但我们可以注意到其中大对象的创建,点击发现一个存放Customer的数组使用了1670万字节的空间。我们需要使用内存分析,进一步查找详细信息。

调试托管内存

我们可以选择调试托管内存,得到托管堆视图。看到这里基本可以断定,应用在该时刻创建了大量的Customer对象,并且没有及时释放,导致了内存占用升高。

查看并行堆栈

那么我们能否定位到代码位置呢?此时可以查看并行堆栈,它会显示dump时正在运行的线程信息,我们可以查看是哪些线程正在执行创建Customer的程序,并且根据pdb定位到代码位置。

再次dump对比

我们可以再次dump,并且进行内存分析,可以看到Customer对象以及大部分被回收了。

内存泄漏分析

调用接口,模拟20万kb内存泄漏发生,内存泄漏的显著特点是内存占用持续升高,并且没有减小的趋势。同样得到dump文件之后拖入Visual Studio进行分析。

按照理论,我们应该先分析内存整体状态,找出占用内存较多的对象,这里大多数对象是 StringCustomer 对象。然后再通过对System.String 实例使用 gcroot 命令,以查看对象的根方式和原因。

但是,Visual Studio已经帮我们做好了一切,甚至已经把嫌疑最大的对象标记出来了。

查看对象引用关系

这里的内存泄漏是由静态根引起的,不了解静态根和GC原理的同学可以看这里:CLR内存管理机制

我们可以看到testwebapi.Controllers.Customer对象被创建了100万个,并且没有被回收,双击查看对象实例,发现它们最终都间接的被testwebapi.Controllers.Processor这个静态变量所引用,因此无法被释放。

总结

好了,关于dump分析的内容就写到这里了,dump主要用在无法直接进行调试的生产环境或CI环境。dump分析命令非常复杂且低效,借助Visual Studio我们可以更加高效的分析dump所记录的数据。

下篇我们将介绍Visual Studio调试时功能强大的程序诊断工具以及企业版的特有的IntelliTrace等内容。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/357652.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot2.x实战专题——SpringBoot2 多配置文件【开发环境、测试环境、生产环境】(内含教学视频+源代码)

SpringBoot2.x实战专题——SpringBoot2 多配置文件【开发环境、测试环境、生产环境】(内含教学视频源代码) 教学视频源代码下载链接地址:https://download.csdn.net/download/weixin_46411355/87463492 目录SpringBoot2.x实战专题——SpringBoot2 多配置文件【开发…

【MySQL】Java连接MySQL数据库(封装版只需会MySQL)

一、准备普通项目如果创建的是普通的Java项目,我们需要去maven仓库下载jdbc驱动包然导入项目中就能使用,具体步骤详见MySQL数据库之Java中如何使用数据库【JDBC编程】maven项目如果创建的项目是maven项目,我们只需在pom.xml文件里引入一组依赖…

适合初学者的超详细实用调试技巧(上)

我们日常写代码的时候,常常会遇到bug的情况,这个时候像我这样的初学者就会像无头苍蝇一样这里改改那里删删,为了根除这种情况,我最近系统学习了调试的技巧,我想要十分详细地讲解,所以大概不会一篇文章写完。…

Django入门与构建

文章目录Django入门与构建Django介绍conda切换环境(看需要)安装虚拟环境基于 virtualenv基于venvDjango项目构建Django项目安装:创建应用Django运行测试总结Django入门与构建 官网:https://www.djangoproject.com/ 博客:https://www.liujian…

Spring的IOC,注入问题,包括自动注入的条件和前提

1. 前言 Spring的核心技术IOC(Intorol of Converse控制反转)的实现途径是DI(dependency Insert依赖注入)。而依赖注入(DI)的实现方式又有两种,xml方式和注解方式。如下图所示: 什么是属性注入 属性注入就是在实例化对象时,同时向对象中的…

小米12s ultra,索尼xperia1 iv,数码相机 拍照对比

首先说明所有的测试结果和拍摄数据我放到百度网盘了(地址在结尾) 为什么做这个测试 我一直想知道现在的手机和相机差距有多大,到底差在哪儿? 先说结论: 1.1英寸的手机cmos(2022年) 6年前(2016)的入门款相机(m43画幅) 2.手机 不能换镜头,只能在特定的拍摄距离才能发挥出全…

网络安全-域名相关收集

网络安全-域名相关收集 前言 一,我也是初学者记录的笔记 二,可能有错误的地方,请谨慎 三,欢迎各路大神指教 四,任何文章仅作为学习使用 五,学习网络安全知识请勿适用于违法行为 学习网络安全知识请勿适用于…

【C#基础】C# 常用语句讲解

序号系列文章3【C#基础】C# 数据类型总结4【C#基础】C# 变量和常量的使用5【C#基础】C# 运算符总结文章目录前言语句概念1,迭代语句1.1 for 语句1.2 foreach 语句1.3 while 语句1.4 do 语句2,选择语句2.1,if 语句2.2,else 语句2.3…

Android 基础知识4-2.9 FrameLayout(帧布局)详解

一、FrameLayout(帧布局)概述 FrameLayout又称作帧布局,它相比于LinearLayout和RelativeLayout要简单很多,因为它的应用场景也少了很多。这种布局没有方便的定位方式,所有的控件都会默认摆放在布局的左上角。 示例1代…

算法队列与栈—Java版

队列、双向队列、栈 — ArrayDeque 使用双向队列ArrayDeque可以完成以上三种数据结构。队列的操作包括:入队、出队、返回队首元素、返回队尾元素、删除队首元素、删除队尾元素、判断空、返回队列长度。双向队列包括:首尾入队、首尾出队、返回首尾元素、…

【挣值分析】

名称解释 拼写解释PV计划费用,预估预算EV挣值,实际预估预算AC实际费用,实际花费CV成本偏差 (EV - AC)SV进度偏差(EV - PV)CPI成本绩效指数 (EV / AC)SPI进度绩效指数 &a…

【Kubernetes】【十】Pod详解 Pod介绍与配置

第五章 Pod详解 本章节将详细介绍Pod资源的各种配置(yaml)和原理。 Pod介绍 Pod结构 每个Pod中都可以包含一个或者多个容器,这些容器可以分为两类: 用户程序所在的容器,数量可多可少 Pause容器,这是每个…

Mybatis(4)之跟着老杜做一个简单的银行转账会话

这是个MVC项目,我不一定可以完整的实现这个项目,但力求把这个复现出来,尽量的复现细节。 第一步:创建数据库 表 创建表如下: 我们使用 int 是为了方便 然后采用 demcial,精确度较高 添加两个用户 然后…

JVM学习笔记二:运行时数据区概述及JVM线程

目录 概述 JVM各区域划分 JVM线程 线程的内存空间 JVM系统线程 概述 运行时数据区是JVM非常重要的组成部分,这一篇主要介绍运行时数据区各个区域的划分,以及JVM中的线程。 JVM各区域划分 学习JVM,必须牢记下图各个区域的分类&#xff…

【5G RRC】5G系统消息SIB2介绍

博主未授权任何人或组织机构转载博主任何原创文章,感谢各位对原创的支持! 博主链接 本人就职于国际知名终端厂商,负责modem芯片研发。 在5G早期负责终端数据业务层、核心网相关的开发工作,目前牵头6G算力网络技术标准研究。 博客…

九龙证券|动力锂离子电池的能量密度可达多少?

现在,我国车用动力锂电池技能道路挑选的是与美国相同的磷酸铁锂道路。毫无疑问,磷酸铁锂离子电池有着许多优势:稳定性高、价格相对便宜,然而其能量密度(能量密度更大,才能更好地解决路程问题)提…

RabbitMQ 入门到应用 ( 五 ) 基本应用

6.更多应用 6.1.AmqpAdmin 工具类 可以通过Spring的Autowired 注入 AmqpAdmin 工具类 , 通过这个工具类创建 队列, 交换机及绑定 import org.springframework.amqp.core.AmqpAdmin; import org.springframework.amqp.core.Binding; import org.springframework.amqp.core.Di…

linux xargs 删除名字中包含某字符串的文件

xargs的作用 格式化输出 可以把多行文本变成一行,或者指定行数和列数。每一列用空格作分隔符号。 test.txt中的内容 例子1: 用xargs格式化输出后,多行变成了一行,而且多个空格变成了一个空格。 cat test|xargs例子2: 当然也可…

go size class 内存块思考

浏览到的一篇文章,让我也有机会反思一下 go 内存管理。网络上,go 内存管理方面的介绍挺多的,面试的时候,偶尔也会被问到内存管理。 而且,从 go1.15 到 go1.16 在 size class 上引入了新的内存块,能直观的看…

信奥一本通1365

1365:FBI树(fbi) 时间限制: 1000 ms 内存限制: 65536 KB 提交数: 6443 通过数: 4366 【题目描述】 我们可以把由“0”和“1”组成的字符串分为三类:全“0”串称为B串,全“1”串称为I串,既含“0”又含“1”的串则称为…