c#在未出现异常情况下查看当前调用堆栈
C#查看堆栈通常是在异常处理中,出现异常之后通过异常的堆栈可以很方便的得到出现这个错误的代码调用路径。这个很有用,是否可以在没有异常出现时使用这种方法排查一些非异常错误呢?答案是肯定的。
起因:
论坛发帖子有几个途径,有可能是新闻系统直接导入的帖子,也有可能是抓取的帖子,还有可能是用户通过正常途径发表。但是这两天出了一个问题,有些帖子的HasImage属性不对。通过几种方法做调试都不能重现问题,没有办法,只有在程序中添加回复的地方添加日志程序来记录堆栈,从而追踪到是哪个途径发帖出现了问题。
代码:
[PostProviderExtension] public class HasImageErrorCheckerPostExtension : IPostProviderExtension { public void BindEvents(PostProviderBase postProvider) { postProvider.Added += new PostChanged(postProvider_Added); } void postProvider_Added(Model.PostInfo post) { try { StackFrame[] stacks = new StackTrace().GetFrames(); if (post.Content.IndexOf( "IMG" ) > -1 && post.HasImage == false ) { StringBuilder sb = new StringBuilder(); sb.AppendLine( "问题出现" ); sb.AppendLine( "stack is:" ); sb.Append(ToString(stacks)); sb.Append( "content=" ); sb.AppendLine(post.Content); sb.Append( "HasImage=" ); sb.AppendLine(UserPostContentProcessor.HasImage(post.Content).ToString()); sb.Append( "createUserID=" ); sb.AppendLine(post.CreateUserID.ToString()); sb.AppendLine( string .Format( "LoginUser={0},Level={1}" ,PageBase.GetLoginUser().ID,PageBase.GetLoginUser().LevelNo)); TextLogWriter.NamedInstance( "\\log\\HasImageErrorCheckerPostExtension\\" ).Write(sb.ToString()); } } catch (Exception ex) { TextLogWriter.NamedInstance( "\\log\\HasImageErrorCheckerPostExtension\\" ).Write(ex); } } private string ToString(StackFrame[] stacks) { string result = string .Empty; foreach (StackFrame stack in stacks) { result += string .Format( "{0} {1} {2} {3}\r\n" , stack.GetFileName(), stack.GetFileLineNumber(), stack.GetFileColumnNumber(), stack.GetMethod().ToString()); } return result; } } |
上面类HasImageErrorCheckerPostExtension继承自IPostProviderExtension并且有PostProviderExtension属性修饰,系统会自动调用它并在发帖时触发这里绑定的事件。这里的核心代码是new StackTrace().GetFrames()通过这个方法可以得到当前程序执行时的堆栈信息。在Release模式下可以得到调用的方法名,在Debug模式下可以得到具体的文件行号,列号。
这个方法是调试中不能重现问题时的一种查找问题的选择方案。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 地球OL攻略 —— 某应届生求职总结
· 提示词工程——AI应用必不可少的技术
· Open-Sora 2.0 重磅开源!
· 周边上新:园子的第一款马克杯温暖上架
2009-05-08 javascript + css 利用div的scroll属性让TAB动感十足
2008-05-08 sql server 中用户定义函数的限制