将IEnumerable强制转换为IEnumerable<;T>;反射

本文关键字:IEnumerable gt 反射 lt 转换 | 更新日期: 2023-09-27 18:27:59

所以,我需要调用一个第三方方法,它有一个像这样的签名

ThirdPartyMethod<T>(IEnumerable<T> items)

我的问题是在编译时我不知道对象的类型。

在编译时,我有这个

IEnumerable myObject;
Type typeOfEnumerableIHave;

Sooo。。反思能以某种方式帮助我吗?

为了简单起见,假设我有一个类似的方法

void DoWork(IEnumerable items, Type type)
{
   //In here I have to call ThirdPartyMethod<T>(IEnumerable<T> items);
}

将IEnumerable强制转换为IEnumerable<;T>;反射

由于您有IEnumerable myObject;和类似ThirdPartyMethod<T>(IEnumerable<T> items)的签名,因此可以使用Cast():

ThirdPartyMethod(myObject.Cast<T>())

如果在编译时不知道T类型,则应该在运行时提供它。

假设你的第三方库看起来像这个

public static class External
{
    public static void ThirdPartyMethod<T>(IEnumerable<T> items)
    {
        Console.WriteLine(typeof(T).Name);
    }
}

如果你有以下

Type theType = typeof(int); 
IEnumerable myObject = new object[0];

您可以在运行时中获得通用方法ThirdPartyMethodCast

var targetMethod = typeof(External).GetMethod("ThirdPartyMethod", BindingFlags.Static | BindingFlags.Public);
var targetGenericMethod = targetMethod.MakeGenericMethod(new Type[] { theType });
var castMethod = typeof(Enumerable).GetMethod("Cast", BindingFlags.Static | BindingFlags.Public);
var caxtGenericMethod = castMethod.MakeGenericMethod(new Type[] { theType });

最后调用方法:

targetGenericMethod.Invoke(null, new object[] { caxtGenericMethod.Invoke(null, new object[] { myObject }) });

你可以试试这样的东西:

void DoWork(IEnumerable items, Type type)
    {
        // instance of object you want to call
        var thirdPartyObject = new ThirdPartyObject();
        // create a list with type "type"
        var typeOfList = typeof(List<>).MakeGenericType(type);
        // create an instance of the list and set items 
        // as constructor parameter
        var listInstance = Activator.CreateInstance(listOfTypes, items);
        // call the 3. party method via reflection, make it generic and
        // provide our list instance as parameter
        thirdPartyObject.GetType().GetMethod("ThirdPartyMethod")
            .MakeGenericMethod(type)
            .Invoke(thirdPartyObject, new []{listInstance});            
    }

该代码创建泛型类型"type"的列表实例(通过使用MakeGenericType)。然后将项元素复制到列表中,并通过相关操作调用第三方方法(请注意"MakeGenericMethod"调用,以确保该方法具有与方法参数相同的类型参数。