Як отримати тип T від члена родового класу чи методу?


675

Скажімо, у мене є загальний член у класі чи методі, тож:

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(), оскільки список може містити нульові елементи. Як я можу це зробити?

Відповіді:


695

Якщо я правильно розумію, у вашому списку є той же параметр типу, що і сам клас контейнерів. Якщо це так, то:

Type typeParameterType = typeof(T);

Якщо вам пощастило мати objectпараметр типу, дивіться відповідь Марка .


4
Лол - так, дуже правда; Я припустив , що ОП був тільки object, IListчи подібне - але це цілком може бути правильну відповідь.
Марк Гравелл

32
Мені подобається читати typeof. Якщо ви хочете знати тип Т, просто використовуйте typeof(T):)
demoncodemonkey

2
Я фактично просто використовував typeof (Type), і він чудово працює.
Антон

1
Однак ви не можете використовувати typeof () із загальним параметром.
Рейневан

2
@Reynevan Звичайно, ви можете використовувати typeof()загальний параметр. Чи є у вас приклад, коли б це не працювало? Або ви плутаєте параметри типу та посилання?
Луань

520

(Примітка: Я припускаю , що всі ви знаєте, objectабо IListчи подібний, і що цей список може бути будь-яким типом під час виконання)

Якщо ви знаєте, що це таке List<T>, то:

Type type = abc.GetType().GetGenericArguments()[0];

Ще один варіант - подивитися індексатор:

Type type = abc.GetType().GetProperty("Item").PropertyType;

Використання нового TypeInfo:

using System.Reflection;
// ...
var type = abc.GetType().GetTypeInfo().GenericTypeArguments[0];

1
Тип типу = abc.GetType (). GetGenericArguments () [0]; ==> Індекс масиву поза межами ...
Патрік Дежардінс,

28
@Daok: тоді це не список <T>
Marc Gravell

Потрібно щось для BindingList або List або будь-який об'єкт, що містить <T>. Що я роблю, використовую користувальницький BindingListView <T>
Патрік Дежардінс,

1
Спробуйте с BindingList <T>, наш BindingListView <T> успадкував від BindingList <T>, і обидва я спробував обидва варіанти, і це не працює. Я можу зробити щось не так ... але я думаю, що це рішення працює для типу List <T>, але не для іншого типу списку.
Патрік Дежардінс

Тип типу = abc.GetType (). GetProperty ("Елемент"). PropertyType; повернути BindingListView <MyObject> замість MyObject ...
Патрік Дежардінс

49

За допомогою наступного методу розширення ви можете піти без роздумів:

public static Type GetListType<T>(this List<T> _)
{
    return typeof(T);
}

Або більш загальне:

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

Використання:

List<string>        list    = new List<string> { "a", "b", "c" };
IEnumerable<string> strings = list;
IEnumerable<object> objects = list;

Type listType    = list.GetListType();           // string
Type stringsType = strings.GetEnumeratedType();  // string
Type objectsType = objects.GetEnumeratedType();  // BEWARE: object

10
Це корисно лише в тому випадку, якщо ви вже знаєте тип Tпід час компіляції. У такому випадку вам зовсім не потрібен код.
рекурсивна

'повернути за замовчуванням (T);'
фантастика

1
@recursive: Це корисно, якщо ви працюєте зі списком анонімного типу.
JJJ

Я робив саме те саме, перш ніж прочитав вашу відповідь, але я зателефонував моємуItemType
вдруге

31

Спробуйте

list.GetType().GetGenericArguments()

7
новий список <int> () .GetType (). GetGenericArguments () повертає System.Type [1] сюди з System.Int32 як запис
Rauhotz

@Rauhotz GetGenericArgumentsметод повертає об'єкт Array Type, з якого вам потрібно потім розібрати потрібне вам положення Generic Type. Такі як Type<TKey, TValue>: вам потрібно буде GetGenericArguments()[0]ввести TKeyтип та GetGenericArguments()[1]отримати TValueтип
GoldBishop

14

Це для мене робота. Де мій список - це якийсь невідомий вид списку.

IEnumerable myEnum = myList as IEnumerable;
Type entryType = myEnum.AsQueryable().ElementType;

1
Я отримую помилку, що для нього потрібен аргумент типу (тобто <T>)
Джозеф Хамфрі

Йосиф та інші, щоб позбутися помилки, це в System.Collections.
користувач2334883

1
Просто мені потрібен другий рядок. A List- це вже реалізація IEnumerable, тому акторський склад, здається, нічого не додає. Але дякую, це вдале рішення.
pipedreambomb

10

Якщо вам не потрібна ціла змінна Type, а ви просто хочете перевірити тип, ви можете легко створити змінну temp і використовувати оператор.

T checkType = default(T);

if (checkType is MyClass)
{}

9

Ви можете використовувати цей для повернення загального списку:

public string ListType<T>(T value)
{
    var valueType = value.GetType().GenericTypeArguments[0].FullName;
    return valueType;
}

8

Врахуйте це: я використовую його, щоб експортувати 20 типових списків таким же чином:

private void Generate<T>()
{
    T item = (T)Activator.CreateInstance(typeof(T));

    ((T)item as DemomigrItemList).Initialize();

    Type type = ((T)item as DemomigrItemList).AsEnumerable().FirstOrDefault().GetType();
    if (type == null) return;
    if (type != typeof(account)) //account is listitem in List<account>
    {
        ((T)item as DemomigrItemList).CreateCSV(type);
    }
}

1
Це не працює, якщо T абстрактний надклас фактично доданих об'єктів. Не кажучи вже про те, new T();що робив би те саме, що і (T)Activator.CreateInstance(typeof(T));. Для цього потрібно додати where T : new()визначення класу / функції, але якщо ви хочете зробити об'єкти, це все одно потрібно зробити.
Nyerguds

Також ви дзвоните GetTypeза FirstOrDefaultзаписом, що призводить до потенційного виключення з нульовою посиланням. Якщо ви впевнені, що він поверне хоча б один предмет, чому б не використати Firstнатомість?
Mathias Lykkegaard Lorenzen

6

Я використовую цей метод розширення, щоб здійснити щось подібне:

public static string GetFriendlyTypeName(this Type t)
{
    var typeName = t.Name.StripStartingWith("`");
    var genericArgs = t.GetGenericArguments();
    if (genericArgs.Length > 0)
    {
        typeName += "<";
        foreach (var genericArg in genericArgs)
        {
            typeName += genericArg.GetFriendlyTypeName() + ", ";
        }
        typeName = typeName.TrimEnd(',', ' ') + ">";
    }
    return typeName;
}

public static string StripStartingWith(this string s, string stripAfter)
{
    if (s == null)
    {
        return null;
    }
    var indexOf = s.IndexOf(stripAfter, StringComparison.Ordinal);
    if (indexOf > -1)
    {
        return s.Substring(0, indexOf);
    }
    return s;
}

Ви використовуєте його так:

[TestMethod]
public void GetFriendlyTypeName_ShouldHandleReallyComplexTypes()
{
    typeof(Dictionary<string, Dictionary<string, object>>).GetFriendlyTypeName()
        .ShouldEqual("Dictionary<String, Dictionary<String, Object>>");
}

Це не зовсім те, що ви шукаєте, але це корисно для демонстрації прийомів.


Привіт, дякую за вашу відповідь, чи можете ви додати розширення для "StripStartingWith"?
Седрік

1
@CedricArnould - додано.
Кен Сміт

5

GetGenericArgument()Метод повинен бути встановлений на базовий тип примірника (клас якого є загальний клас myClass<T>). В іншому випадку він повертає тип [0]

Приклад:

Myclass<T> instance = new Myclass<T>();
Type[] listTypes = typeof(instance).BaseType.GetGenericArguments();

5

Ви можете отримати тип "T" з будь-якого типу колекції, який реалізує IEnumerable <T> із наступним:

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 <T> двічі, наприклад

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

Я думаю, ви могли б повернути масив типів, якщо вам це було потрібно для підтримки ...

Також ви можете також захотіти кешувати результат за типом колекції, якщо ви робите це багато (наприклад, у циклі).


1
public bool IsCollection<T>(T value){
  var valueType = value.GetType();
  return valueType.IsArray() || typeof(IEnumerable<object>).IsAssignableFrom(valueType) || typeof(IEnumerable<T>).IsAssignableFrom(valuetype);
}

1
Це, мабуть, вирішує питання про те, чи є тип подібним до списку, але питання стосується того, як визначити, з яким загальним параметром типу тип, який, як відомо, вже був ініціалізований.
Натан Туггі

1

Використання рішення 3dGrabber:

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

//and now 

var list = new Dictionary<string, int>();
var stronglyTypedVar = list.GetEnumeratedType();

0

Якщо ви хочете знати базовий тип ресурсу, спробуйте:

propInfo.PropertyType.UnderlyingSystemType.GenericTypeArguments[0]

0

Ось як я це зробив

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());

АБО

var item = Activator.CreateInstance(Bar.GetType().GetElementType());

-9

Тип:

type = list.AsEnumerable().SingleOrDefault().GetType();

1
Це б кинуло NullReferenceException, якщо в списку немає елементів всередині нього, щоб він міг перевірити.
rossisdead

1
SingleOrDefault()також кидає, InvalidOperationExceptionколи є два або більше елементів.
devgeezer

Ця відповідь є неправильною, як правильно вказали \ @rossisdead та \ @devgeezer.
Олівер
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.