我对以下代码有一个非常奇怪的问题:
using System;
using System.Linq;
namespace ConsoleApp1
{
public class Program
{
public static void Main(string[] args)
{
var testTuples1 = GimeStringTuples("foo", "bar");
var testTuples2 = GimeTuples("foo", "bar");
Console.ReadKey();
}
public static object GimeStringTuples(params string[] values)
{
Type genericType = Type.GetType("System.Tuple`" + values.Length);
Type[] typeArgs = values.Select(_ => typeof(string)).ToArray();
Type specificType = genericType.MakeGenericType(typeArgs);
return Activator.CreateInstance(specificType, values);
}
public static object GimeTuples<T>(params T[] values)
{
Type genericType = Type.GetType("System.Tuple`" + values.Length);
Type[] typeArgs = values.Select(_ => typeof(T)).ToArray();
Type specificType = genericType.MakeGenericType(typeArgs);
dynamic result;
string[] testArgs = { "foo", "bar" };
result = Activator.CreateInstance(specificType, testArgs);
result = Activator.CreateInstance(specificType, values);
return result;
}
}
}
它在第二行到最后一行失败了:
result = Activator.CreateInstance(specificType, values);
这很奇怪,因为它基本上与之前执行的行相同:
result = Activator.CreateInstance(specificType, testArgs);
在这两种情况下,相同的参数作为specificType
参数传递,而string[2]
作为第二个参数传递。
GimeStringTuples
方法工作得很好......虽然没有涉及泛型 - 这可能是提示。
任何人都能解释这种不寻常的行为吗?
答案 0 :(得分:1)
我可以告诉你会发生什么。查看生成的程序集,您可以看到编译器引入了object[]
包装values
参数:
Activator.CreateInstance(specificType, new string[] { "foo", "bar" });
Activator.CreateInstance(specificType, new object[] { values });
现在无法找到正确的超载。 如果你添加一个演员表,你将得到预期的结果,代码再次起作用:
Activator.CreateInstance(specificType, values as string[])
但我无法告诉你为什么会发生这种情况,也许它可以从规格中挖掘出来。
答案 1 :(得分:0)
好的解决方案(感谢@thehennyy):
using System;
using System.Linq;
namespace ConsoleApp1
{
public class Program
{
public static void Main(string[] args)
{
var testTuple = GimeTuples("foo", "bar");
Console.WriteLine(testTuple);
Console.ReadKey();
}
public static object GimeTuples<T>(params T[] values)
{
Type genericType = Type.GetType("System.Tuple`" + values.Length);
Type[] typeArgs = values.Select(_ => typeof(T)).ToArray();
Type specificType = genericType.MakeGenericType(typeArgs);
object[] constructorArguments = values.Cast<object>().ToArray();
return Activator.CreateInstance(specificType, constructorArguments);
}
}
}
这会强制编译器将通用数组元素作为参数传递,而不是将整个数组作为单个参数传递。