3

我正在查看优秀的 Dapper micro-orm,在Dapper.Rainbow项目中,有一些代码使用 IL 创建表 ctor。我希望有人可以向我解释这段代码在做什么 - 显然它是动态发出代码来创建构造函数(或者是调用构造函数的代码,或两者兼而有之),但是实际创建时发出的代码是什么,以及为什么需要这个吗?

这是代码

protected Action<TDatabase> CreateTableConstructor(Type tableType)
{
    var dm = new DynamicMethod("ConstructInstances", null, new Type[] { typeof(TDatabase) }, true);
    var il = dm.GetILGenerator();

    var setters = GetType().GetProperties()
        .Where(p => p.PropertyType.IsGenericType && p.PropertyType.GetGenericTypeDefinition() == tableType)
        .Select(p => Tuple.Create(
                p.GetSetMethod(true),
                p.PropertyType.GetConstructor(new Type[] { typeof(TDatabase), typeof(string) }),
                p.Name,
                p.DeclaringType
         ));

    foreach (var setter in setters)
    {
        il.Emit(OpCodes.Ldarg_0);
        // [db]

        il.Emit(OpCodes.Ldstr, setter.Item3);
        // [db, likelyname]

        il.Emit(OpCodes.Newobj, setter.Item2);
        // [table]

        var table = il.DeclareLocal(setter.Item2.DeclaringType);
        il.Emit(OpCodes.Stloc, table);
        // []

        il.Emit(OpCodes.Ldarg_0);
        // [db]

        il.Emit(OpCodes.Castclass, setter.Item4);
        // [db cast to container]

        il.Emit(OpCodes.Ldloc, table);
        // [db cast to container, table]

        il.Emit(OpCodes.Callvirt, setter.Item1);
        // []
    }

    il.Emit(OpCodes.Ret);
    return (Action<TDatabase>)dm.CreateDelegate(typeof(Action<TDatabase>));
}
4

1 回答 1

3

基本上,它采用当前类型(派生自Database<T>)的所有属性,如下所示:

public Table<SomeEntity> SomeTable { get; private set; }

并为每个此类属性创建一个执行如下代码的委托:

var table = new Table<SomeEntity>(db, "SomeTable");
db.SomeTable = table;
于 2013-04-10T19:57:50.670 回答