- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
为了实验,我尝试从源类型读取方法体(使用 GetILAsByteArray())并将其添加到新类型(使用 CreateMethodBody() ).
我的源码类就是这样
public class FullClass
{
public string Test(string data)
{
return data;
}
public string Test2(string data)
{
return data;
}
public string Test5(string data, string data1)
{
return data + data1;
}
}
为此代码生成的 IL(使用反射器获取)
.method public hidebysig instance string Test(string data) cil managed
{
.maxstack 1
.locals init (
[0] string CS$1$0000)
L_0000: nop
L_0001: ldarg.1
L_0002: stloc.0
L_0003: br.s L_0005
L_0005: ldloc.0
L_0006: ret
}
但是从我的新类型生成的 IL 看起来像这样
.method public hidebysig virtual instance string Test(string) cil managed
{
.maxstack 0
L_0000: nop
L_0001: ldarg.1
L_0002: stloc.0
L_0003: br.s L_0005
L_0005: ldloc.0
L_0006: ret
}
不同之处在于 maxstack 值和 .locals 指令。我不明白为什么我的实际类生成局部变量,尽管它没有任何局部变量??
为什么 .maxstack 值不同,因为我使用来自源的相同 IL 来创建新类型。?
由于这个原因,在调用该方法时出现错误“公共(public)语言运行时检测到无效程序”。
我创建动态类型的代码如下所示
public static class Mixin<Target>
{
public static Target compose<TSource>()
{
Type newType = null;
AppDomain currentDom = Thread.GetDomain();
AssemblyName DAssembly = new AssemblyName();
DAssembly.Name = "DynamicTypesAssembly";
AssemblyBuilder DAssemblyBldr = currentDom.DefineDynamicAssembly(
DAssembly,
AssemblyBuilderAccess.RunAndSave);
ModuleBuilder DModuleBldr = DAssemblyBldr.DefineDynamicModule(DAssembly.Name, DAssembly.Name + ".dll", false);
// var DInterface = EmitInterface(DModuleBldr);
TypeBuilder TypeBldr = DModuleBldr.DefineType("WorkOut.DType",
TypeAttributes.Public | TypeAttributes.BeforeFieldInit | TypeAttributes.Serializable
,typeof(object), new[] { typeof(Target) });
//TypeBldr.AddInterfaceImplementation(typeof(DInterface));
var methodCol = typeof(Target).GetMethods(BindingFlags.Public| BindingFlags.Instance);
foreach (var ms in methodCol)
{
var paramCol = ms.GetParameters();
var paramTypeArray = paramCol.Select(x => x.ParameterType).ToArray();
var paramNameArray = paramCol.Select(x=>x.Name).ToArray();
MethodBuilder MthdBldr = TypeBldr.DefineMethod(ms.Name,
MethodAttributes.Public | MethodAttributes.Virtual | MethodAttributes.HideBySig,
ms.ReturnType,
paramTypeArray);
for(int i=0;i<paramCol.Count();i++)
{
MthdBldr.DefineParameter(i+1, ParameterAttributes.None, paramNameArray[i]);
}
MethodInfo[] methodInfos = typeof(TSource).GetMethods(BindingFlags.Public | BindingFlags.NonPublic |
BindingFlags.Static | BindingFlags.Instance);
for (int i = 0; i < methodInfos.Count(); i++)
{
var paramSrc = methodInfos[i].GetParameters();
var paramSrcTypeArray = paramSrc.Select(x => x.ParameterType).ToArray();
if (methodInfos[i].Name == ms.Name && methodInfos[i].ReturnType == ms.ReturnType && paramSrc.Count() == paramCol.Count() && paramTypeArray.SequenceEqual(paramSrcTypeArray))
{
var ILcodes = methodInfos[i].GetMethodBody().GetILAsByteArray();
var ilGen = MthdBldr.GetILGenerator();
//ilGen.Emit(OpCodes.Ldarg_0); //Load the 'this' reference onto the evaluation stack
//ilGen.Emit(OpCodes.Initobj);
MthdBldr.CreateMethodBody(ILcodes, ILcodes.Length);
//ilGen.Emit(OpCodes.Ret);
break;
}
}
}
newType = TypeBldr.CreateType();
DAssemblyBldr.Save("a.dll");
return (Target)Activator.CreateInstance(newType);
}
调用它的代码是
var resMix = Mixin<ITest>.compose<FullClass>();
var returned1 = resMix.Test("sam");
编辑: ITest(目标)接口(interface)是
public interface ITest
{
string Test(string data);
}
编辑:
评论这一行时
//var ilGen = MthdBldr.GetILGenerator();
maxstack 变为 .maxstack 16
我用 PEverify 工具对新的 dll 进行了检查,这给出了以下错误
WorkOut.DType::Test][offset 0x00000002] 无法识别的局部变量号。
非常感谢任何帮助.... :)
最佳答案
作为关于 CreateMethodBody 的 MSDN 页面说,这没有得到完全支持。
很可能实现没有将 IL 解析为字节数组,因此它突然将 maxstack 设置为 16。
如果您为该方法创建一个 ILGenerator,它会将方法 maxstack 设置为零。当您使用不同的 Emit 重载时,ILGenerator 会递增它。由于您没有这样做,而是使用 CreateMethodBody,它会保持为零。这解释了差异。
CreateMethodBody 对于涉及除简单代码之外的任何内容的场景肯定是有问题的。每个带有元数据 token 的操作码都将不可用,因为在创建字节数组时您不知道模块范围内的有限 token 。而且它不允许您发出异常处理程序。
长话短说,CreateMethodBody 本身毫无意义。
如果你想继续实验,我建议你使用我的reflection IL reader获取方法指令的表示,然后使用 ILGenerator 在方法构建器中重现方法主体。
关于c# - 动态类型创建中的 MethodBuilder.CreateMethodBody() 问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4460859/
为了实验,我尝试从源类型读取方法体(使用 GetILAsByteArray())并将其添加到新类型(使用 CreateMethodBody() ). 我的源码类就是这样 public class Fu
我是一名优秀的程序员,十分优秀!