c# - شرح - كيفية الحصول على نوع T من عضو في فئة أو أسلوب عام؟




generics in c# شرح (11)

لنفترض أن لديّ عضوًا عامًا في الفصل أو المنهج ، لذلك:

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() ، لأن القائمة قد تحتوي على عناصر الصفر. كيف يمكنني فعل ذلك؟


(ملاحظة: أفترض أن كل ما تعرفه هو 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];

أستخدم طريقة الإضافة هذه لإنجاز شيء مشابه:

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

أنت تستخدمها على هذا النحو:

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

هذا ليس تماما ما تبحث عنه ، ولكن من المفيد في إظهار التقنيات المعنية.


إذا كنت تريد معرفة النوع الأساسي للعقار ، فجرّب ذلك:

propInfo.PropertyType.UnderlyingSystemType.GenericTypeArguments[0]

إذا كنت لا تحتاج إلى متغير نوع بالكامل وترغب فقط في التحقق من النوع يمكنك بسهولة إنشاء متغير مؤقت والاستخدام هو عامل التشغيل.

T checkType = default(T);

if (checkType is MyClass)
{}

باستخدام حل 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 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;

يجب GetGenericArgument() أسلوب GetGenericArgument() على النوع الأساسي من المثيل الخاص بك (فئة له فئة عامة myClass<T> ). خلاف ذلك ، تقوم بإرجاع نوع [0]

مثال:

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

يمكنك الحصول على نوع "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> { ... }

أفترض أنه يمكنك إرجاع مجموعة من الأنواع إذا كنت بحاجة إلى دعم هذا ...

أيضا ، قد ترغب أيضًا في تخزين النتيجة لكل نوع تجميع إذا كنت تفعل هذا كثيرًا (على سبيل المثال في حلقة).


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

يمكنك استخدام هذا واحد لإعادة تشغيل قائمة عامة.







generics