使用反射实例化泛型类

时间:2013-09-24 18:16:12

标签: c# generics

[注意:我不相信这个问题与上面链接的问题重复,正如我在下面的UPDATE中解释的那样。]

有没有办法使用反射来定义/实例化泛型类?

所以我有一堆类,每个类都拥有一个泛型类的实例,它共享其所有者的类型:

public class GenericClass<T>
{
    T Owner { get; set; }
    public GenericClass(T owner) { Owner = owner; }
}
public class MyClass
{
    private GenericClass<MyClass> myGenericObject;
    public MyClass() { myGenericObject = new GenericClass<MyClass>(this); }
}

这是有效的,但当然我必须明确指定“MyClass”作为GenericClass定义中的参数。我希望能够做到这样的事情:

private GenericClass<typeof(this)> myGenericObject; // Error: invalid token

有没有在编译时根据包含的类动态指定泛型对象的类型?


更新:在阅读these questions的答案后,我了解到我可以像这样实例化一个局部变量:

var myGenericObject = Activator.CreateInstance(typeof(GenericClass<>).MakeGenericType(this.GetType()));

但是,当然,this关键字只能在方法中使用(例如,我可以将这行代码放在MyClass的构造函数中)。但我无法使用此方法在上面的代码中定义实例变量(即myGenericObject)。有没有办法动态指定通用实例变量?

3 个答案:

答案 0 :(得分:2)

关于您的更新,您可以将Type传递给MakeGenericType。例如,以下内容也有效:

var myObject = new MyClass();

var myGenericObject = Activator.CreateInstance(typeof(GenericClass<>).MakeGenericType(typeof(MyClass)), myObject);

Console.WriteLine(myGenericObject.GetType());

输出:

ConsoleApplication1.GenericClass`1[ConsoleApplication1.MyClass]

myObject.GetType()也做了同样的事情:

var myGenericObject = Activator.CreateInstance(typeof(GenericClass<>).MakeGenericType(myObject.GetType()), myObject);

答案 1 :(得分:0)

有静态内置构造:

Activator.CreateInstance()

查看重载。

更新

public Type FakeType { get; private set; }
public T CreateInstance<T>() where T : SomeEntityBase
{
     return (T) Activator.CreateInstance(FakeType);
}

答案 2 :(得分:0)

不确定这是你想要的,但可以尝试继承:

public class GenericClass<T>
{
    T Owner { get; set; }

    public GenericClass(T owner) { Owner = owner; }
}

public abstract class MyClassBase<T> where T : MyClassBase<T>
{
    protected GenericClass<T> MyGenericObject { get; private set; }

    protected MyClassBase() { MyGenericObject = new GenericClass<T>((T)this); }
}

public class MyClass1 : MyClassBase<MyClass1>
{
    public MyClass1() { }
}

public class MyClass2 : MyClassBase<MyClass2>
{
    public MyClass2() { }
}