一个奇怪的 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中调试,会发现打不到断点。
要重现这个奇怪的案例,您需要满足以下条件:
我不确定这些条件是否足以重现它,但当我发现这个问题时,我的机器就是这样配置的。
为什么调试器无法命中断点?
提前致谢!
如果您可以重现此问题,请考虑在 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/
使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta
我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何
我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>
如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象
关闭。这个问题需要detailsorclarity.它目前不接受答案。想改进这个问题吗?通过editingthispost添加细节并澄清问题.关闭8年前。Improvethisquestion为什么SecureRandom.uuid创建一个唯一的字符串?SecureRandom.uuid#=>"35cb4e30-54e1-49f9-b5ce-4134799eb2c0"SecureRandom.uuid方法创建的字符串从不重复?
我有一个正在构建的应用程序,我需要一个模型来创建另一个模型的实例。我希望每辆车都有4个轮胎。汽车模型classCar轮胎模型classTire但是,在make_tires内部有一个错误,如果我为Tire尝试它,则没有用于创建或新建的activerecord方法。当我检查轮胎时,它没有这些方法。我该如何补救?错误是这样的:未定义的方法'create'forActiveRecord::AttributeMethods::Serialization::Tire::Module我测试了两个环境:测试和开发,它们都因相同的错误而失败。 最佳答案
GivenIamadumbprogrammerandIamusingrspecandIamusingsporkandIwanttodebug...mmm...let'ssaaay,aspecforPhone.那么,我应该把“require'ruby-debug'”行放在哪里,以便在phone_spec.rb的特定点停止处理?(我所要求的只是一个大而粗的箭头,即使是一个有挑战性的程序员也能看到:-3)我已经尝试了很多位置,除非我没有正确测试它们,否则会发生一些奇怪的事情:在spec_helper.rb中的以下位置:require'rubygems'require'spork'
如何在ruby中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL
我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b
我想让一个yaml对象引用另一个,如下所示:intro:"Hello,dearuser."registration:$introThanksforregistering!new_message:$introYouhaveanewmessage!上面的语法只是它如何工作的一个例子(这也是它在thiscpanmodule中的工作方式。)我正在使用标准的rubyyaml解析器。这可能吗? 最佳答案 一些yaml对象确实引用了其他对象:irb>require'yaml'#=>trueirb>str="hello"#=>"hello"ir