4

简短版本: 我希望我的程序能够在运行时(只读)访问它自己的源代码。有没有办法在编译时自动将源代码打包到dll中?

长版:

背景是当发生异常时,我想自动创建一个文件,其中包含发生的一些细节。该文件应包括导致问题的函数的源代码。我想通过电子邮件将此文件发送给其他人,而接收者很可能没有(或不想安装)Visual Studio,所以任何使用符号服务器等的东西都是不可能的。它必须是纯文本文件。

理想情况下,我会在某处找到相关的源代码文件,然后复制相关行。您可以放心地假设,只要我有一个包含整个源代码的文件夹,我就能找到我想要的文件和行。

到目前为止我能想到的最好的想法——我没有详细研究它,因为它看起来很混乱——是修改 MSBuild 文件以在编译期间创建源的 .zip,并要求.dll 和 .zip 驻留在同一文件夹中。

我发现的关于 stackoverflow 的大多数听起来相似的问题似乎都涉及反编译.dll 文件,这不是我想要做的。我有源代码,我想以方便的方式将它与 .dll 一起提供。

更新:真正的长版本

似乎有些人在认真质疑我为什么要这样做,所以这是答案:我的软件的主要目的是测试其他一些软件。其他软件有一个 GUI。举个简单的例子,假设其他软件是标准的 Windows 计算器,那么我的测试用例可能看起来像这样:

calculator.Open();
calculator.EnterValue(13);
calculator.PressButtonPlus();
calculator.EnterValue(38);
calculator.PressButtonEnter();
int value = calculator.GetDisplayedValue();
Assert.That(value == 51);
calculator.Close();

这些测试是有意以人类可读的方式编写的。

当出现问题时,我想做的是给计算器的开发人员详细描述如何重现问题,以一种他可以在没有我的软件的情况下手动重现的方式。(在这个例子中,他会打开计算器,输入 13,按加号,等等。)

也许另一种选择是让每个函数calculator.Something()在日志中写出一条信息行,但这会 a) 做更多的工作,b) 只包括测试运行到它中止的点,c) 承担一些风险在一个函数中忘记了编写该行,从而对所做的事情给出了错误的表示。但我对复制源代码以外的其他解决方案持开放态度。

4

3 回答 3

2

看看这个问题:C# 中是否存在 __LINE__ __FILE__ 等价物?

C++ 提供了宏(__LINE____FILE__等),它们在编译时替换为表示信息。这意味着如果你写这样的东西:

throw new CException(__FILE__);

它会编译成这样的:

throw new CException("test.cpp");

产生一个硬编码的值。C# 编译器不提供这些宏,因此您必须使用反射来获取有关引发异常的位置的信息。上面链接的问题中描述了您可以做到的方式。

如果您无法提供.pdbException.ToString()符号,则(or )的默认行为StackTrace.ToString()不会返回行号,而是返回失败操作的 MSIL 偏移量。据我所知,您可以使用 ReSharper 的Stack Trace Explorer导航到表示代码(不是 100% 确定这一点,但在 stackoverflow 上也有一个问题提到了这个事实)。

于 2013-01-09T14:01:30.967 回答
1

您可以包含源文件的副本作为资源。

  • 在项目文件夹中,创建一个名为Resources. 在那里复制源文件。
  • 在项目中创建一个资源文件,然后将您制作的源副本包含在其中。
  • 设置预构建事件以将实际源文件复制到Resources文件夹,因此您始终拥有更新的副本。在我创建的示例中,效果很好:

    copy $(ProjectDir)*.cs $(ProjectDir)Resources
    
  • 在您的代码中,现在您可以像这样获取文件的内容(我想资源文件的名称是Resource1.resx

    // Get the source of the Program.cs file.
    string contents = Resource1.Program;
    

该项目最终是这样的:

项目树

于 2013-01-09T14:49:12.327 回答
0

是的,我还建议在 MSBuild 期间将源文件打包成 .zip 或其他任何文件,并将该 .zip 文件与您的应用程序/dll 一起打包。在运行时,当发生异常时,您会获得 Aschratt 描述的文件和方法名称,从 .zip 中提取文件并在其中找到方法。

于 2013-01-09T14:45:33.503 回答