型別資訊(Type Information)用來表示型別宣告的資訊,通過抽象基類System.Type的例項儲存這些資訊,當使用反射時,CLR獲取指定型別的Type物件,通過這個物件即可訪問該型別的任何資訊,是反射的核心用法;有以下幾種方式獲取指定型別的Type物件:
Type myType = typeof(MyType); //其中MyType是指定的型別 myType = myObj.GetType(); //其中myObj是指定型別的例項 myType = Type.GetType(myTypeName); //其中myTypeName是指定型別的全名,與Type的例項屬性FullName保持一致 myType = myAssembly.GetType(myTypeName); //其中myAssembly是指定型別所在的程式集例項
※如果指定的型別不在全域性名稱空間中,myTypeName需要加入該型別所在的名稱空間名稱,例如:
MyNameSpace.MyClass
※如果指定的型別是巢狀型別,那麼需要用+來分割其與宣告它的型別,例如:
MyNameSpace.MyClass+MyNestedClass
※使用隱式轉換、顯式轉換或as運算子將一個物件轉換成其它型別的變數時,該變數獲取到的型別資訊不會變,依然是其原型別;
※只要獲取的是同一種型別的型別資訊,不管使用哪一種方式,它們的引用都是同一個:
int myNum1 = 10; int myNum2 = 20; //以下獲取的所有型別例項都指向同一個物件,無論使用==還是Object.ReferenceEquals()都會返回true Type type1 = myNum1.GetType(); Type type2 = myNum2.GetType(); Type type3 = typeof(int); Type type4 = Type.GetType("System.Int32");
※對於泛型型別,只有其型別引數一致時,才會得到同樣的型別例項;
通過型別的Type物件,可以獲取該型別的各種成員資訊,包括:
※C#中的訪問修飾符protected和internal在IL中沒有任何意義,不會用於反射中;若要判斷欄位、方法、屬性和事件中的get方法和set方法是否被宣告為internal,使用欄位資訊類FieldInfo或方法資訊基類MethodBase中的屬性IsAssembly;若要判斷是否被宣告為protected,使用屬性IsFamily;若要判斷是否被宣告為protectedinternal,使用屬性IsFamilyOrAssembly;
※在各個獲取成員資訊的例項方法GetXXX()的引數列表中,有個BindingFlags型別的列舉組合用於指定搜尋的範圍組合,使用時必須要組合使用,例如:
BindingFlags.Public | BindingFlags.Instance //搜尋範圍為所有公共的例項成員
BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance //搜尋範圍為所有例項成員
※Type類的例項方法IsSubclassOf(Type c)用於確定當前類是否是從指定的類派生來的;
※通過基類MemberInfo中的例項方法IsDefined(Type attributeType, bool inherit)可以確定當前成員是否被指定了指定型別的特性;
※通過基類MemberInfo中的屬性MemberType可以獲取當前型別或成員的型別列舉組合,列舉型別的宣告為:
public enum MemberTypes { Constructor = 1, //建構函式 Event = 2, //事件 Field = 4, //欄位 Method = 8, //方法 Property = 16, //屬性 TypeInfo = 32, //型別 Custom = 64, //自定義成員型別 NestedType = 128, //巢狀型別
All = 191 }
型別資訊中涉及到的東西較多,本篇內容會不定期進行完善。
如果您覺得閱讀本文對您有幫助,請點一下“推薦”按鈕,您的認可是我寫作的最大動力!
作者:Minotauros
出處:https://www.cnblogs.com/minotauros/
本文版權歸作者和部落格園共有,歡迎轉載,但未經作者同意必須保留此段宣告,且在文章頁面明顯位置給出原文連線,否則保留追究法律責任的權利。