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;
}
يمكنك استخدام هذا واحد لإعادة تشغيل قائمة عامة.