草庐IT

c# - 一个奇怪的 Visual Studio 2010 调试器案例(它不能命中断点)

coder 2024-05-20 原文

一个奇怪的 Visual Studio 2010 调试器案例(它不能命中断点)

这是重现问题的代码:

class Program {
  static void Main(string[] args) {
    bool b = false;

    if (b) {
        List<string> list = new List<string>();
        foreach (var item in list) {

        }
    } else {
        Console.WriteLine("1");
    }
    Console.WriteLine("2");//add a break point here in VS2010
  }
  //1. configuration: release
  //2. platform target: x64 or Any Cpu
  //3. debug info: pdb only or full
  //4. OS: Win7 x64
  //5. optimize code: enabled
}

在代码的最后一句打断点,然后在vs2010中调试,会发现打不到断点。

要重现这个奇怪的案例,您需要满足以下条件:

  1. 操作系统:windows 7 x64;
  2. VS构建配置:发布;
  3. VS 构建平台目标:x64 或任何 Cpu;
  4. VS 构建调试信息:仅 pdb 或完整;
  5. VS 构建优化代码:已启用;

我不确定这些条件是否足以重现它,但当我发现这个问题时,我的机器就是这样配置的。

为什么调试器无法命中断点?

提前致谢!

如果您可以重现此问题,请考虑在 this post. 上投票

最佳答案

当提供的示例在 Release模式下构建,然后通过 JIT 转换为 64 位机器代码时,它不包含足够的信息供调试器将断点与任何特定机器指令相关联。这就是为什么调试器在执行 JIT 机器代码期间永远不会在此断点处停止。它只是不知道在哪里停下来。这可能是某种不当行为,甚至是 64 位 CLR 调试器中的错误,因为它只有在 JIT 编译为 64 位机器代码而不是 32 位机器代码时才可重现。

当调试器在您的代码中发现断点时,它会尝试在 JIT 代码中找出与断点标记的位置相对应的机器指令。首先,它需要找到与 C# 代码中的断点位置对应的 IL 指令。然后它需要找到与IL命令对应的机器指令。然后它在找到的机器指令上设置一个真正的断点并开始执行该方法。在您的情况下,调试器似乎只是忽略了一个断点,因为它无法将其映射到特定的机器指令。

调试器无法找到紧跟在 if…else 语句之后的机器指令的地址。 if...else 语句和其中的代码以某种方式导致了这种行为。 if...else 后面的语句无关紧要。您可以将 Console.WriteLine(“2”) 语句替换为其他语句,您仍然能够重现该问题。

如果您使用 Reflector 反汇编生成的程序集,您将看到 C# 编译器围绕读取列表的逻辑发出一个 try…catch block 。它是 C# 编译器的文档化功能。您可以在 The foreach statement 阅读更多相关信息。

try…catch…finally block 对 JIT 代码具有相当大的侵入性影响。它在后台使用 Windows SEH 机制并严重重写您的代码。我现在找不到好文章的链接,但我相信如果您有兴趣,可以在那里找到一篇。

这就是这里发生的事情。 try...finally block 在 if...else 语句内部导致调试器打嗝。您可以使用非常简单的代码重现您的问题。

bool b = false;
if (b)
{
    try
    {
        b = true;
    }
    finally
    {
        b = true;
    }
}
else
{
    b = true;
}
b = true;

这段代码不调用任何外部函数(它消除了其中一个答案提出的方法内联的影响),它直接编译到 IL 中,而不需要 C# 编译器添加任何额外的代码。

它只能在 Release模式下重现,因为在 Debug模式下,编译器会为 C# 代码的每一行发出 IL NOP 指令。 IL NOP 指令什么也不做,它被同样什么都不做的JITer 直接编译成CPU NOP 指令。这条指令的用处在于它可以被调试器用作断点的 anchor ,即使其余代码被 JITer 严重重写也是如此。

通过在 if…else 之后的语句之前放置一条 NOP 指令,我能够使调试器正常工作。

您可以在此处阅读有关 NOP 操作和调试器映射过程的更多信息 Debugging IL

您可以尝试使用 WinDbg 和 SOS 扩展来检查该方法的 JIT 版本。您可以尝试检查 JIT-er 生成的机器代码,并尝试理解为什么它不能将该机器代码映射回特定的 C# 行。

这里是关于使用 WinDbg 中断托管代码和获取 JIT-ed 方法的内存地址的几个链接。我相信您应该能够找到一种方法来从那里获取方法的 JIT 代码:Setting a breakpoint in WinDbg for Managed Code , SOS Cheat Sheet (.NET 2.0/3.0/3.5) .

您也可以尝试向 Microsoft 报告问题。这可能是 CLR 调试器错误。

感谢您提出有趣的问题。

关于c# - 一个奇怪的 Visual Studio 2010 调试器案例(它不能命中断点),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5744506/

有关c# - 一个奇怪的 Visual Studio 2010 调试器案例(它不能命中断点)的更多相关文章

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

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

  2. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  3. ruby-on-rails - 渲染另一个 Controller 的 View - 2

    我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>

  4. ruby-on-rails - 如果 Object::try 被发送到一个 nil 对象,为什么它会起作用? - 2

    如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象

  5. ruby - 为什么 SecureRandom.uuid 创建一个唯一的字符串? - 2

    关闭。这个问题需要detailsorclarity.它目前不接受答案。想改进这个问题吗?通过editingthispost添加细节并澄清问题.关闭8年前。Improvethisquestion为什么SecureRandom.uuid创建一个唯一的字符串?SecureRandom.uuid#=>"35cb4e30-54e1-49f9-b5ce-4134799eb2c0"SecureRandom.uuid方法创建的字符串从不重复?

  6. ruby-on-rails - Rails - 从另一个模型中创建一个模型的实例 - 2

    我有一个正在构建的应用程序,我需要一个模型来创建另一个模型的实例。我希望每辆车都有4个轮胎。汽车模型classCar轮胎模型classTire但是,在make_tires内部有一个错误,如果我为Tire尝试它,则没有用于创建或新建的activerecord方法。当我检查轮胎时,它没有这些方法。我该如何补救?错误是这样的:未定义的方法'create'forActiveRecord::AttributeMethods::Serialization::Tire::Module我测试了两个环境:测试和开发,它们都因相同的错误而失败。 最佳答案

  7. ruby-on-rails - 无法让 rspec、spork 和调试器正常运行 - 2

    GivenIamadumbprogrammerandIamusingrspecandIamusingsporkandIwanttodebug...mmm...let'ssaaay,aspecforPhone.那么,我应该把“require'ruby-debug'”行放在哪里,以便在phone_spec.rb的特定点停止处理?(我所要求的只是一个大而粗的箭头,即使是一个有挑战性的程序员也能看到:-3)我已经尝试了很多位置,除非我没有正确测试它们,否则会发生一些奇怪的事情:在spec_helper.rb中的以下位置:require'rubygems'require'spork'

  8. c# - 如何在 ruby​​ 中调用 C# dll? - 2

    如何在ruby​​中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL

  9. ruby - 用 Ruby 编写一个简单的网络服务器 - 2

    我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b

  10. ruby - 一个 YAML 对象可以引用另一个吗? - 2

    我想让一个yaml对象引用另一个,如下所示:intro:"Hello,dearuser."registration:$introThanksforregistering!new_message:$introYouhaveanewmessage!上面的语法只是它如何工作的一个例子(这也是它在thiscpanmodule中的工作方式。)我正在使用标准的ruby​​yaml解析器。这可能吗? 最佳答案 一些yaml对象确实引用了其他对象:irb>require'yaml'#=>trueirb>str="hello"#=>"hello"ir

随机推荐