Для эксперимента я пытаюсь прочитать тело методы (с использованием GetILAsByteArray()) от типа источника и добавить его к новому типу (с помощью CreateMethodBody()).MethodBuilder.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;
}
}
Ил генерируется для этого кода (взятый с помощью отражателя)
.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 из источника, чтобы создать новый тип.?
Из-за этого я получаю сообщение об ошибке «Common Language Runtime обнаружил недопустимую программу» при вызове метода.
Мой код создания динамического типа выглядит следующим образом
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");
Edit: И интерфейс ITest (Target) является
public interface ITest
{
string Test(string data);
}
EDIT:
, комментируя эту линию
//var ilGen = MthdBldr.GetILGenerator();
maxstack становится .maxstack 16
Я провел проверку против новой библиотеки DLL против PEverify инструмент, это дает следующие ошибки
WorkOut.DType :: Test] [offset 0x00000002] Нераспознанный номер локальной переменной.
Любая помощь действительно оценили .... :)
Привет, Рамеш, можете ли вы запросить дополнительную информацию или принять ответ? – 2011-01-07 11:05:12
@Jb, я использую ваш читатель IL только ... забыл обновить .. изменит его .. – RameshVel 2011-01-07 11:39:28