1.在使用反射時,反射可以繞過安全訪問級別(private、protected)修飾的類或屬性,來獲取需要的信息。 2.泛型的反射:可以使用Type.ContainsGenericParameters這個屬性來判斷一個類或方法是否包含尚未設置的泛型實參,Type.IsGenericType屬性表示是 ...
1.在使用反射時,反射可以繞過安全訪問級別(private、protected)修飾的類或屬性,來獲取需要的信息。
2.泛型的反射:可以使用Type.ContainsGenericParameters這個屬性來判斷一個類或方法是否包含尚未設置的泛型實參,Type.IsGenericType屬性表示是否為泛型類型。
3.特性(Attribute):可以使用特性修飾類、介面。結構、枚舉、委托、事件、屬性、欄位、方法、構造器、索引器、參數、類型參數、返回值、程式集、模塊,使用特性的語法有2種,可為多個“[特性類型]”或“[特性類型,特性類型]”。對於其中列出多數構造來說,都可以使用上面的語法標記,但是這個語法不適合“返回值、程式集、模塊”。
(1)程式集,[assembly:特性名]。
(2)模塊,[module:特性名]。
(3)返回值,[return:特性名]。
4.大多數特性只針對特定的構造進行修飾,為了避免特性不恰當的使用,可以使用[AttributeUsageAttribute(AttributeTargets.xxx)]特性類進行標記特性限制。
5.具名參數:如[AttributeUsage(AttributeTargets.Class, AllowMultiple = true)]語法有別於構造器初始化語法,因為AttributeUsageAttribute類並不包含兩個參數的構造器,雖然C#4.0支持命名參數,但也是指定方法本身所需的參數。具名參數是用於在特性構造器調用中設置特定的公共屬性和欄位,即使構造器不包含對應的參數,具名參數雖然是可選的,但它允許對特性的額外實例數據進行設置,同時無需提供一個對應的構造器參數。
6.序列化(Serializable):本質上System.Runtime,Serialization.SerializationInfo對象是由“名稱/值”對構成的集合。在實現自定義的序列化時(需要實現ISerializable介面),內中會使用到SerializationInfo對象。
7.動態編程(Dynamic):反射的關鍵功能之一就是動態查找和調用特定類型的一個成員,這需要在執行時識別成員名或其他特征。在C#4.0中新增的動態編程-Dynamic,提供一個更簡單的辦法來通過反射調用成員,但這個技術的限制在於,編譯時需要知道成員名和簽名。若執行時發現事實上沒有這個成員,調用就會引發一個RuntimeBinderException異常。可查看UseDynamic.Test()代碼。
8.Dynamic:究其根本,Dynamic是一個Object,存在任何對象都能隱式轉換成Dynamic,Dynamic可以顯式轉換成其他對象,所以Dynamic在行為上就像Object,類似於Object,它甚至為它的預設值返回null(default(dynamic))。dynamic特殊動態行為只在調用時才會出現,這個行為是它與Object區分開來的關鍵。任何dynamic的成員調用都會返回為dynamic類型,但若對Dynamic執行GetType(),會返回是編譯好的類型(即最後賦值給動態變數的類型)。
9.實現自定義動態對象:定義自定義動態類型的關鍵是實現System.Dynamic.IDynamicMetaObjectProvider介面,但是不必從頭實現,相反首先的方案是從System.Dynamic.DynamicObject類繼承,並重寫相應的方法。DynamicObject類已經實現了IDynamicMetaObjectProvider介面,提供了預設處理。可查看代碼DynamicXml類,UseDynamic類中2個方法NormalMethod()與DynamicMethod()兩者的比較。NormalMethod()方法使用一般的xml數據讀取,DynamicMethod()方法使用了自定義動態對象解析xml內容。
[Serializable]
[AttributeUsage(AttributeTargets.Class, AllowMultiple = true)]
public class MyDescriptionAttribute : Attribute
{
}
public class UseDynamic
{
public static void Test()
{
dynamic data = "Hello! My name is Inigo Montoya";
Console.WriteLine(data);
data = (double)data.Length;
data = data * 3.5 + 28.6;
if (data == 2.4 + 112 + 26.2)
{
Console.WriteLine("data for length : {0}", data);
}
else
{
data.NonExistentMethodCallStillCompiles();
}
}
public static void NormalMethod()
{
XElement person = XElement.Parse("<Person><Name>主神</Name><Age>26</Age></Person>");
Console.WriteLine("{0},{1}", person.Descendants("Name").FirstOrDefault().Value, person.Descendants("Age").FirstOrDefault().Value);
}
public static void DynamicMethod()
{
dynamic person = DynamicXmL.Parse("<Person><Name>主神</Name><Age>26</Age></Person>");
Console.WriteLine("{0},{1}", person.Name, person.Age);
}
}
/// <summary>
/// 實現自定義的動態對象
/// </summary>
public class DynamicXmL : DynamicObject
{
private XElement element;
public DynamicXmL(XElement xElement)
{
element = xElement;
}
public static dynamic Parse(string text)
{
return new DynamicXmL(XElement.Parse(text));
}
public override bool TryGetMember(GetMemberBinder binder, out object result)
{
bool success = false;
result = null;
XElement firstDescendant = element.Descendants(binder.Name).FirstOrDefault();
if (firstDescendant != null)
{
if (firstDescendant.Descendants().Count() > 0)
{
result = new DynamicXmL(firstDescendant);
}
else
{
result = firstDescendant.Value;
}
success = true;
}
return success;
}
public override bool TrySetMember(SetMemberBinder binder, object value)
{
bool success = false;
XElement firstDescendant = element.Descendants(binder.Name).FirstOrDefault();
if (firstDescendant != null)
{
if (value.GetType() == typeof(XElement))
{
firstDescendant.ReplaceWith(value);
}
else
{
firstDescendant.Value = value.ToString();
}
success = true;
}
return success;
}
}
View Code
-----------------以上內容根據《C#本質論 第三版》進行整理