草庐IT

c# - 来自动态代码的异常堆栈跟踪中的文件路径和行号错误

coder 2024-05-28 原文

我们使用 System.Reflection.Emit 在运行时从源代码生成代码(是的——就像在编译器中一样)。我们使用 MarkSequencePoint 等向 ILGenerator 提供正确的符号信息,并在 AssemblyBuilder 上启用所有调试标志。程序集在编译它的同一进程中保存在内存中并直接执行。

当使用 Visual Studio 调试器单步执行动态生成代码的源代码时,它实际上运行良好,并且 Visual Studio 似乎完全了解代码在文件和行号方面的来源。

但是 - 当异常生成的代码抛出时,System.Exception 对象包含完全错误的堆栈跟踪。它们指向其他(有效但错误的)文件和行号。它得到了正确的类名和方法名,但指向的文件和行号与异常实际来自的代码路径无关。

指向的文件是如此不相关,似乎无法链接到内联或优化。我能发现的唯一模式是它似乎被某些文件所抵消(在一个假想的按字母顺序排序的源文件列表中,该程序集是从中构建的)。然而,这种模式并不是 100% 一致的,这似乎与问题的根源有关。

如果我从异常构造一个 System.Diagnostics.Debug 对象,它包含相同的错误信息。

我假设 .NET 运行时使用与调试器用于单步执行代码相同的元数据来构造异常堆栈跟踪,在这种情况下,这种行为真的很奇怪。

我正在尝试查明这是否是 .NET 中处理动态内存程序集时的已知错误,或者是否有人在其他领域看到过类似问题。

最佳答案

好吧,我无法弄清楚是什么导致了问题,也无法弄清楚如何使 .NET 正常运行,但至少我找到了一个可能也适用于遇到相同问题的其他人的解决方法。

  1. 在生成 CIL 字节码时,我正在构建一个单独的数据库,将方法名称(完整路径)和 IL 偏移量映射回原始文件名和行号。

  2. 当捕获到异常时,我会检查堆栈跟踪并仅使用来自堆栈帧对象的 GetMethod()GetILOffset() 信息。来自 CLR 的信息恰好是正确的,即使 GetFileName()GetFileLineNumber() 是错误的。

  3. 然后,对于每个堆栈帧,我使用从异常中检索到的方法名称和 IL 偏移量来查找我生成的数据库,以确定我掌握的每个堆栈帧的实际文件名和行号。

  4. 我在数据库中找不到相关信息的帧通常是预编译 .NET 模块中的堆栈帧,从 CLR 检索到的信息实际上是正确的。对于这些帧,我直接在堆栈帧上使用 GetFileName()GetFileLineNumber()

关于c# - 来自动态代码的异常堆栈跟踪中的文件路径和行号错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21092537/

有关c# - 来自动态代码的异常堆栈跟踪中的文件路径和行号错误的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

    我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

  3. ruby - 其他文件中的 Rake 任务 - 2

    我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时

  4. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

    我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

  5. ruby-on-rails - Ruby net/ldap 模块中的内存泄漏 - 2

    作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代

  6. ruby-on-rails - Rails 3 中的多个路由文件 - 2

    Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题

  7. ruby - 将差异补丁应用于字符串/文件 - 2

    对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

  8. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  9. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  10. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

随机推荐