3

首先让我说,我想做的是获取泛型类中的属性值,该值可能被继承自它的类覆盖。在基类中将其视为默认值,类的继承者可以覆盖以设置自己的默认值。

我尝试使用 System.Reflection.FieldInfo.GetValue 直接在类型上使用反射,但这不适用于具有泛型类型的类。所以我认为我需要实例化类才能看到值是什么。

我通过读取 bin 中的 Dll 并使用反射查找从我的接口继承的类型来检索“类型”。

我正在使用 .NET 4.5

这里的文档似乎准确地解释了我需要做什么 http://msdn.microsoft.com/en-us/library/b8ytshk6.aspx

在本文档中,我能看到的唯一区别是我们是如何获得类型的,我在 bin 中查找类型,它们简单地称为 typeof(),由于类型非常复杂,看起来这可能是不匹配的,但我看不到缺少什么(如果有的话)

foreach (var item in types)
{
    var ts = item.GetField("DefaultTimeToExpire");
    Type[] typeArguments = item.GetGenericArguments();
    if (ts != null)
    {
        var t = item.MakeGenericType(typeArguments);
        var obj = Activator.CreateInstance(t);
        var timespan = obj.DefaultTimeToExpire;
        subscriberInfos.Add(new Tuple<string, Type, TimeSpan>(item.Name, item, timespan));
    }
}

到目前为止,我正在调用 GetField 来查找具有“DefaultTimeToExpire”字段的项目,这部分可以很好地找到我需要的类型。接下来我调用 GetGenericArguments,它返回一个 Arguments 类型的预期数组。然后我调用 MakeGenericType 最后创建实例给我错误消息

“无法创建 BusinessLogic.TestSubscriberXXX`1[Message] 的实例,因为 Type.ContainsGenericParameters 为真。”

这看起来正是我应该做的。谢谢

4

1 回答 1

4

为了实例化泛型类型,您需要知道应该替换其类型参数的实际值(类型)。该GetGenericArguments()方法作为一种反射形式,只为您提供类型参数,而不是它们的实际值。值由您决定……这就是泛型的全部意义所在。

If itemis a type List<T>thenitem.GetGenericArguments()将返回一个数组,其中包含一个表示类型参数的假“类型” T(其IsGenericParameter属性设置为 true)。因此,将该参数类型传递回item.MakeGenericType()将简单地创建另一个与原始类型等效的开放泛型类型。要关闭泛型类型以便对其进行实例化,您需要提供一个实际(非参数)类型参数,例如int.

例如,typeof(List<>).MakeGenericType(typeof(int))将返回typeof(List<int>),而 typeof(List<>).MakeGenericType(typeof(List<>).GetGenericArguments())将简单地typeof(List<>)再次返回。这就是您的代码中发生的事情。

对不起,如果这有点不透明,我不知道该怎么解释。最重要的是,like 类型List<T>仅在您想要替换的类型时才有用T

于 2012-08-29T14:04:38.077 回答