8

目前我正在这样做(//编辑:这完全是愚蠢的,因为有一个数组以与我昨天做我的字典相同的方式对其项目进行编号):

    Dictionary<int, Type> genericMap = new Dictionary<int, Type>
        {
            { 0, typeof(Action) },
            { 1, typeof(Action<>) },
            { 2, typeof(Action<,>) },
            { 3, typeof(Action<,,>) },
            { 4, typeof(Action<,,,>) },
            { 5, typeof(Action<,,,,>) },
            { 6, typeof(Action<,,,,,>) },
            { 7, typeof(Action<,,,,,,>) },
            { 8, typeof(Action<,,,,,,,>) },
        };

还有别的地方...

var parms = meth.GetParameters();
dType = genericMap[parms.Length].MakeGenericType(parms.Select(p => p.ParameterType).ToArray());

其中 meth 是 MethodInfo。

有没有更优雅的方法来做到这一点?或者我是否必须定义这样的映射才能动态获取与参数计数相对应的正确 Action<> 类型?

4

2 回答 2

12

编辑

顺便说一句,你知道Expression.GetActionType吗?

创建一个表示具有特定类型参数的泛型 System.Action 委托类型的 Type 对象。

使用该方法,您可以执行以下操作:

dType = Expression.GetActionType( parms.Select(p => p.ParameterType).ToArray() );

其他方法:

var parms = meth.GetParameters();

int numArgs = parms.Length;

if(numArgs == 0)
{
    dType = typeof(Action);
}    
else 
{
   var rawType = Type.GetType("System.Action`" + numArgs);
   dType = rawType.MakeGenericType(parms.Select(p => p.ParameterType).ToArray());
}

但是你的方法真的没有错。在反思方面,尽可能明确通常是更好的选择。

此外,这是构建地图的另一种奇特方式:

typeof(Action).Assembly
              .GetExportedTypes()
              .Where(type => type.FullName.StartsWith("System.Action") 
                                  && type.IsSubclassOf(typeof(Delegate)))
              .ToDictionary(type => type.GetGenericArguments().Length)
于 2013-04-06T13:31:55.263 回答
1

您可以使用数组而不是字典,它实际上从 0 到 8 编号,只需使用类型数组并使用索引访问它们。

于 2013-04-06T14:58:34.370 回答