สมมติว่าฉันมีสมาชิกทั่วไปในคลาสหรือเมธอด เช่น:

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
    }
}

ฉันต้องการทราบว่ารายการในชั้นเรียนของฉันประกอบด้วยวัตถุประเภทใด ดังนั้นสิ่งที่ประเภทของการTไม่ทรัพย์สินรายการที่เรียกว่าBarหรือตัวแปรท้องถิ่นbazมี?

ฉันทำไม่ได้Bar[0].GetType()เพราะรายการอาจมีองค์ประกอบเป็นศูนย์ ฉันจะทำอย่างไร?

ตอบ

ถ้าฉันเข้าใจถูกต้อง รายการของคุณมีพารามิเตอร์ประเภทเดียวกับคลาสคอนเทนเนอร์ หากเป็นกรณีนี้แล้ว:

Type typeParameterType = typeof(T);

ถ้าคุณอยู่ในสถานการณ์ที่โชคดีของการมีobjectเป็นพารามิเตอร์ชนิดให้ดูคำตอบของมาร์ค

(หมายเหตุ: ผมคิดว่าสิ่งที่คุณทราบว่าเป็นobjectหรือIListหรือคล้ายกันและรายการอาจจะเป็นประเภทใดที่รันไทม์)

หากคุณรู้ว่ามันคือ a 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];

ด้วยวิธีการต่อขยายต่อไปนี้ คุณจะหนีไปได้โดยไม่ไตร่ตรอง:

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

ลอง

list.GetType().GetGenericArguments()

งานต่อไปนี้สำหรับฉัน โดยที่ myList เป็นรายการที่ไม่รู้จัก

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

หากคุณไม่ต้องการตัวแปร Type ทั้งหมด และเพียงต้องการตรวจสอบประเภท คุณสามารถสร้างตัวแปรชั่วคราวและใช้ตัวisดำเนินการได้อย่างง่ายดาย

T checkType = default(T);

if (checkType is MyClass)
{}

คุณสามารถใช้สิ่งนี้สำหรับประเภทส่งคืนของรายการทั่วไป:

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

พิจารณาสิ่งนี้:

ฉันใช้เพื่อส่งออก 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);
    }
}

ฉันใช้วิธีการขยายนี้เพื่อทำสิ่งที่คล้ายกันให้สำเร็จ:

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>>");
}

นี่ไม่ใช่สิ่งที่คุณกำลังมองหา แต่ก็มีประโยชน์ในการสาธิตเทคนิคที่เกี่ยวข้อง

คุณสามารถรับประเภทของ "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> { ... }

ฉันคิดว่าคุณสามารถส่งคืนอาร์เรย์ประเภทต่างๆได้หากต้องการสนับสนุนสิ่งนี้ ...

นอกจากนี้ คุณอาจต้องการแคชผลลัพธ์ตามประเภทคอลเลกชันหากคุณทำสิ่งนี้เป็นจำนวนมาก (เช่น วนซ้ำ)

GetGenericArgument()วิธีการจะต้องมีการตั้งอยู่บนฐานประเภทอินสแตนซ์ (ที่มีระดับเป็นระดับทั่วไปmyClass<T>) มิฉะนั้น จะส่งคืน type[0]

ตัวอย่าง:

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

ใช้โซลูชันของ 3dGrabber :

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

//and now

var list = new Dictionary<string, int>();
var stronglyTypedVar = list.GetEnumeratedType();
public bool IsCollection<T>(T value){
  var valueType = value.GetType();
  return valueType.IsArray() || typeof(IEnumerable<object>).IsAssignableFrom(valueType) || typeof(IEnumerable<T>).IsAssignableFrom(valuetype);
}

หากคุณต้องการทราบประเภทพื้นฐานของพร็อพเพอร์ตี้ ให้ลองทำดังนี้:

propInfo.PropertyType.UnderlyingSystemType.GenericTypeArguments[0]

นี่คือวิธีที่ฉันทำ:

internal static Type GetElementType(this Type type)
{
    // Use type.GenericTypeArguments if it exists
    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());

พิมพ์:

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