9

在 C# 中是否可以在运行时创建一个从泛型类继承的类型,其中基类的模板参数是当前正在构造的类?这将编译得很好:

// I have this class:
public class OtherClass<T>
    where T : OtherClass<T>
{ }

// I want to create this at runtime:
public class MyClass : OtherClass<MyClass>
{ }

但我不确定如何创建MyClassusing System.Reflection.Emit.ModuleBuilder.TypeBuilder

AssemblyName asn = new AssemblyName("test.dll");
AssemblyBuilder asb = AppDomain.CurrentDomain.DefineDynamicAssembly(
    asn, AssemblyBuilderAccess.RunAndSave, @"D:\test_assemblies");

ModuleBuilder = modb = asb.DefineDynamicModule("test", "test.dll");

TypeBuilder tb = modb.DefineType(
    "test",
    TypeAttributes.Public | TypeAttributes.Class,
    typeof(MyClass)); // How to specify inheritance?

// populate properties, fields, methods, etc., emit...

tb.CreateType();

这可能吗?

编辑 - 根据到目前为止的回复,我已经尝试过:

public class OtherClass<T>
    where T : OtherClass<T>
{ }

public static void CreateSimple()
{
    AssemblyName asn = new AssemblyName("test");
    AssemblyBuilder asb = AppDomain.CurrentDomain.DefineDynamicAssembly(
        asn, AssemblyBuilderAccess.RunAndSave, @"D:\test_asms");
    ModuleBuilder modb = asb.DefineDynamicModule("test", "test.dll");

    try
    {
        TypeBuilder tb = modb.DefineType(
            "MyClass",
            TypeAttributes.Public | TypeAttributes.Class);

        Type my = tb.CreateType();
        tb.SetParent(typeof(OtherClass<>).MakeGenericType(my));

        my = tb.CreateType();
    }
    catch (Exception e)
    {
        throw;
    }
}

但得到这个例外:

GenericArguments[0], 'MyClass', on 'MyProject.Factory+OtherClass`1[T]'
violates the constraint of type 'T'.
4

3 回答 3

8

编辑:这是我的最终工作答案:

        AssemblyName asn = new AssemblyName("test.dll");
        AssemblyBuilder asb = AppDomain.CurrentDomain.DefineDynamicAssembly(
            asn, AssemblyBuilderAccess.RunAndSave, @"D:\test_assemblies");

        ModuleBuilder modb = asb.DefineDynamicModule("test", "test.dll");

        TypeBuilder tb = modb.DefineType(
            "test",
            TypeAttributes.Public | TypeAttributes.Class);
        // Typebuilder is a sub class of Type
        tb.SetParent(typeof(OtherClass<>).MakeGenericType(tb));
        var t2 = tb.CreateType();
        var i = Activator.CreateInstance(t2);

诀窍是使用参数化的泛型类型调用 SetParent,参数是正在构造的类型本身的类型构建器。


使用TypeBuilder.SetParent(Type parent)方法。

使用时要小心,异常抛出推迟到CreateType调用:

如果 parent 为 null,则 Object 用作​​基本类型。

在 .NET Framework 1.0 和 1.1 版本中,如果 parent 是接口类型,则不会引发异常,但在调用 CreateType 方法时会引发 TypeLoadException。

SetParent 方法不检查大多数无效的父类型。例如,当当前类型有默认构造函数时,不拒绝没有默认构造函数的父类型,不拒绝密封类型,也不拒绝 Delegate 类型。在所有这些情况下,CreateType 方法都会引发异常。

要构建您的泛型类型OtherClass<T>,请使用MakeGenericType方法:

var genericType = typeof(OtherClass<>).MakeGenericType(typeof(MyClass));
于 2012-11-07T19:16:20.317 回答
2

您可以在之后设置基本类型。

TypeBuilder.SetParent(类型)

于 2012-11-07T19:08:10.920 回答
1

是的。这是可能的。请参阅Reflection.Emit命名空间和 TypeBuilder 的方法。

于 2012-11-07T19:09:25.163 回答