วิธีรับประเภท 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
ฮ่า ๆ - ใช่จริงมาก ผมสันนิษฐานว่า OP มีเพียงobject, IListหรือที่คล้ายกัน - แต่นี้เป็นอย่างดีอาจจะเป็นคำตอบที่เหมาะสม
Marc Gravell

32
ฉันรักวิธีที่อ่านได้typeofมี หากคุณต้องการที่จะทราบชนิดของ T เพียงใช้typeof(T):)
demoncodemonkey

2
ฉันแค่ใช้ typeof (Type) และใช้งานได้ดีมาก
Anton

1
อย่างไรก็ตามคุณไม่สามารถใช้ typeof () กับพารามิเตอร์ทั่วไปได้
Reynevan

2
@ Reynevan แน่นอนคุณสามารถใช้typeof()กับพารามิเตอร์ทั่วไป คุณมีตัวอย่างที่ไม่สามารถใช้งานได้หรือไม่? หรือคุณกำลังสับสนพารามิเตอร์ประเภทและการอ้างอิง?
Luaan

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
พิมพ์ type = abc.GetType () .GetGenericArguments () [0]; ==> ดัชนีอาร์เรย์ไม่อยู่ในขอบเขต ...
Patrick Desjardins

28
@Daok: ไม่ใช่รายการ <T>
Marc Gravell

ต้องการบางอย่างสำหรับ BindingList หรือ List หรือวัตถุใด ๆ ที่มี <T> สิ่งที่ฉันกำลังทำใช้ BindingListView แบบกำหนดเอง
Patrick Desjardins

1
ลอง BindingList <T>, BindingListView <T> ของเราสืบทอดจาก BindingList <T> และทั้งคู่ฉันลองทั้งตัวเลือกแล้วมันใช้ไม่ได้ ฉันอาจทำสิ่งผิดปกติ ... แต่ฉันคิดว่าวิธีนี้ใช้ได้กับรายการประเภท <T> แต่ไม่ใช่รายการประเภทอื่น
Patrick Desjardins

พิมพ์ type = abc.GetType () .GetProperty ("Item") 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
toddmo

31

ลอง

list.GetType().GetGenericArguments()

7
รายการใหม่ <int> () .GetType (). GetGenericArguments () ส่งคืน System.Type [1] ที่นี่ด้วย System.Int32 เป็นรายการ
Rauhotz

@Rauhotz GetGenericArgumentsวิธีการส่งกลับวัตถุ Array Typeซึ่งคุณต้องแยกตำแหน่งของประเภททั่วไปที่คุณต้องการ เช่นType<TKey, TValue>: คุณจะต้องGetGenericArguments()[0]รับTKeyประเภทและGetGenericArguments()[1]รับTValueประเภท
GoldBishop

14

นั่นเป็นผลสำหรับฉัน โดยที่ myList เป็นรายการที่ไม่ทราบ

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

1
ฉันได้รับข้อผิดพลาดว่าต้องใช้อาร์กิวเมนต์ชนิด (เช่น<T>)
Joseph Humfrey

โจเซฟและคนอื่น ๆ เพื่อกำจัดข้อผิดพลาดมันอยู่ใน System.Collections
user2334883

1
ฉันต้องการแค่บรรทัดที่สอง A Listกำลังดำเนินการอยู่IEnumerableดังนั้นนักแสดงจึงไม่ได้เพิ่มอะไรเลย แต่ขอบคุณมันเป็นทางออกที่ดี
pipedreambomb

10

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

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" ด้วยหรือไม่
Cedric Arnould

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
สิ่งนี้ปรากฏขึ้นเพื่อตอบคำถามว่าประเภทนั้นเป็นรายการเรียงลำดับของสิ่งใดหรือไม่ แต่คำถามนั้นเกี่ยวกับวิธีกำหนดพารามิเตอร์ประเภททั่วไปที่เป็นที่รู้จักกันว่าเป็นรายการที่ได้รับการเตรียมใช้งานแล้ว
นาธาน Tuggy

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
โอลิเวอร์
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.