我正在尝试将两个方法合并在一起(一个来自我的项目,一个来自 dll(并将它们写入 dll。原始代码如下所示:
.method public hidebysig static class Storage.DataMap
Create() cil managed
{
.maxstack 8
// [22 7 - 22 70]
IL_0000: call unmanaged stdcall native int Storage.DataMapPlugin::DataMap_CreateEmptyDataMap()
IL_0005: newobj instance void Storage.DataMap::.ctor(native int)
IL_000a: ret
} // end of method DataMap::Create
我想在这个代码中添加一个简单的 Console.WriteLine,我最终得到以下内容:(来自反编译器(
.method public hidebysig static class Storage.DataMap
Create() cil managed
{
.maxstack 8
// [22 7 - 22 44]
IL_0000: ldstr "Hello World"
IL_0005: call void [mscorlib]System.Console::WriteLine(string)
// [23 7 - 23 70]
IL_000a: call unmanaged stdcall native int Storage.DataMapPlugin::DataMap_CreateEmptyDataMap()
IL_000f: newobj instance void Storage.DataMap::.ctor(native int)
IL_0014: ret
} // end of method DataMap::Create
我的反编译器无法反编译它(dotPeek(,所以我假设我生成的IL无效。但是,我不确定我生成的 IL 出了什么问题。
我将 dnlib
与 C# 一起使用来获取该方法的两个 IL 代码,然后将两个代码块连接在一起(我剥离了 nop
和 ret
语句(最后一个除外((。为了重新编译代码,我使用 ModuleDef#Write(String)
函数,我在编写自己的 IL 语句时取得了成功,但在合并现有语句时却没有成功。
(以下 IL 全部从控制台打印,而不是从反编译器打印(
原始部分 1 IL:
IL_0000: nop
IL_0001: ldstr "Hello World"
IL_0006: call System.Void System.Console::WriteLine(System.String)
IL_000B: nop
IL_000C: ret
原始部分 2 IL:
IL_0000: call System.IntPtr Storage.DataMapPlugin::DataMap_CreateEmptyDataMap()
IL_0005: newobj System.Void Storage.DataMap::.ctor(System.IntPtr)
IL_000A: ret
合并的 IL:
IL_0001: ldstr "Hello World"
IL_0006: call System.Void System.Console::WriteLine(System.String)
IL_0000: call System.IntPtr Storage.DataMapPlugin::DataMap_CreateEmptyDataMap()
IL_0005: newobj System.Void Storage.DataMap::.ctor(System.IntPtr)
IL_000A: ret
我解决了自己的问题。正如多个人在评论中正确指出的那样,IL 是正确的。该问题与我的代码或生成的 IL 无关,而是与反编译器 dotPeek 有关。
发生的事情如下:我第一次生成 DLL,但 IL 是错误的(它有 2 个返回语句(,使用 dotPeek 对其进行反编译,发现 IL 格式不正确。我再次执行此操作,但使用正确的 IL(只有 1 个返回语句(并删除了旧 DLL,将新 DLL 重命名为旧的、已删除的 DLL。然而,DotPeek不喜欢这样,它会向我显示更新的IL,但不会显示更新的代码。
经过多次重新启动并删除/重新添加程序集后,我发现情况确实如此。 dnSpy似乎比dotPeek工作得更好,所以我将来会使用它。