Как получить тип T от члена универсального класса или метода? - PullRequest
619 голосов
/ 17 февраля 2009

Допустим, у меня есть универсальный член в классе или методе, поэтому:

public class Foo<T>
{
    public List<T> Bar { get; set; }

    public void Baz()
    {
        // get type of T
    }   
}

Когда я создаю экземпляр класса, T становится MyTypeObject1, поэтому у класса есть общее свойство списка: List<MyTypeObject1>. То же самое относится и к универсальному методу в неуниверсальном классе:

public class Foo
{
    public void Bar<T>()
    {
        var baz = new List<T>();

        // get type of T
    }
}

Я хотел бы знать, какой тип объектов содержится в списке моего класса. Таким образом, свойство списка с именем Bar или локальная переменная baz содержит тип T?

Я не могу сделать Bar[0].GetType(), потому что список может содержать ноль элементов. Как я могу это сделать?

Ответы [ 16 ]

4 голосов
/ 16 апреля 2014

Вы можете получить тип «T» из любого типа коллекции, который реализует IEnumerable со следующим:

public static Type GetCollectionItemType(Type collectionType)
{
    var types = collectionType.GetInterfaces()
        .Where(x => x.IsGenericType 
            && x.GetGenericTypeDefinition() == typeof(IEnumerable<>))
        .ToArray();
    // Only support collections that implement IEnumerable<T> once.
    return types.Length == 1 ? types[0].GetGenericArguments()[0] : null;
}

Обратите внимание, что он не поддерживает типы коллекций, которые реализуют IEnumerable дважды, например

public class WierdCustomType : IEnumerable<int>, IEnumerable<string> { ... }

Полагаю, вы могли бы вернуть массив типов, если вам нужно это поддерживать ...

Кроме того, вы также можете кэшировать результат для каждого типа коллекции, если вы делаете это много (например, в цикле).

1 голос
/ 03 июня 2015

Используя решение 3dGrabber:

public static T GetEnumeratedType<T>(this IEnumerable<T> _)
{
    return default(T);
}

//and now 

var list = new Dictionary<string, int>();
var stronglyTypedVar = list.GetEnumeratedType();
1 голос
/ 05 мая 2015
public bool IsCollection<T>(T value){
  var valueType = value.GetType();
  return valueType.IsArray() || typeof(IEnumerable<object>).IsAssignableFrom(valueType) || typeof(IEnumerable<T>).IsAssignableFrom(valuetype);
}
0 голосов
/ 28 июня 2017

Вот как я это сделал

internal static Type GetElementType(this Type type)
{
        //use type.GenericTypeArguments if exist 
        if (type.GenericTypeArguments.Any())
         return type.GenericTypeArguments.First();

         return type.GetRuntimeProperty("Item").PropertyType);
}

Тогда назови это так

var item = Activator.CreateInstance(iListType.GetElementType());

OR

var item = Activator.CreateInstance(Bar.GetType().GetElementType());
0 голосов
/ 13 октября 2016

Если вы хотите узнать базовый тип свойства, попробуйте следующее:

propInfo.PropertyType.UnderlyingSystemType.GenericTypeArguments[0]
0 голосов
/ 07 марта 2010

Тип:

type = list.AsEnumerable().SingleOrDefault().GetType();
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...