C# Dynamic之:ExpandoObject,DynamicObject,DynamicMetaOb的應(yīng)用(下)
接上文:C# Dynamic關(guān)鍵字之:ExpandoObject,DynamicObject,DynamicMetaOb的應(yīng)用(上)
為什么TryXXX方法沒有被調(diào)用??
將DynamicProduct 中的name修飾符改為private:
private string name;
可以在TrySetMember方法中設(shè)置斷點,再次運行:
為什么訪問修飾符是Public不調(diào)用TrySetMember,是Private 就調(diào)用了呢??
難道是因為private拋出了異常嗎??
再次看看Msdn對此的TrySetMember方法的解釋:
Msdn備注
…………….動態(tài)語言運行庫 (DLR) 將首先使用語言聯(lián)編程序在類中查找屬性的靜態(tài)定義。 如果沒有此類屬性,DLR 調(diào)用 TrySetMember 方法。
問題的原因是這樣的:首先DLR 使用語言聯(lián)編程序在類中查找name的靜態(tài)定義,
因為name是public,所以查找到了,然后返回,不會去調(diào)用TrySetMember方法了,
但是如果name是private,那么聯(lián)編程序在類中沒找到name的靜態(tài)定義,于是DLR嘗試調(diào)用TrySetMember方法。
修改TrySetMember方法如下:
public override bool TrySetMember(SetMemberBinder binder, object value)
{
Console.WriteLine("TrySetMember被調(diào)用了,Name:{0}", binder.Name);
bool result = base.TrySetMember(binder, value);
return true;
}
運行,可以發(fā)現(xiàn)不會拋出異常了:
總結(jié):首先DLR會嘗試查找屬性的靜態(tài)定義,如果沒有找到則會調(diào)用相應(yīng)的TryXXX 方法,如果TryXXX方法返回false,代表TryXXX方法運行失敗,DLR隨后會拋出異常。
為了驗證是不是這樣,將DynamicProduct中屬性的靜態(tài)定義全部注釋掉,并且TryXXX方法全部返回True。完整的代碼如下:
class DynamicProduct : DynamicObject
{
#region dynamicProduct 的一些屬性的靜態(tài)定義
//private string name;
//public int Id { get; set; }
//public void ShowProduct()
//{
// Console.WriteLine("Id={0} ,Name={1}", Id, name);
//}
#endregion
#region Override DynamicObject 的方法
public override bool TryGetMember(GetMemberBinder binder, out object result)
{
Console.WriteLine("TryGetMember被調(diào)用了,Name:{0}", binder.Name);
bool tryResult = base.TryGetMember(binder, out result);
return true;
}
public override bool TrySetMember(SetMemberBinder binder, object value)
{
Console.WriteLine("TrySetMember被調(diào)用了,Name:{0}", binder.Name);
bool tryResult = base.TrySetMember(binder, value);
return true;
}
public override bool TryInvoke(InvokeBinder binder, object[] args, out object result)
{
Console.WriteLine("TryInvoke被調(diào)用了");
bool tryResult = base.TryInvoke(binder, args, out result);
return true;
}
public override bool TryInvokeMember(InvokeMemberBinder binder, object[] args, out object result)
{
Console.WriteLine("TryInvokeMember被調(diào)用了,Name:{0}", binder.Name);
bool tryResult = base.TryInvokeMember(binder, args, out result);
return true;
}
#endregion
}
Main方法不變:
static void Main(string[] args)
{
dynamic dynProduct = new DynamicProduct();
dynProduct.name = "n1"; //調(diào)用TrySetMember方法
dynProduct.Id = 1;
dynProduct.Id = dynProduct.Id + 3;
dynProduct.ShowProduct();
Console.ReadLine();
}
運行,結(jié)果如下:
d.P3 = d.M1(d.P1, d.M2(d.P2));
按照從左到右,從里到外的原則。
1:先調(diào)用d.P1,DLR會嘗試調(diào)用d 的GetMetaObject 方法,此方法返回一個MyMetaObject對象。
接著DLR知道你調(diào)用的是一個屬性,于是它調(diào)用返回的MyMetaObject對象的BindGetMember 方法,
輸出為GetMember of property P1
2:調(diào)用d.P2,和調(diào)用d.P1 一樣.
3:調(diào)用d.M2,同樣DLR調(diào)用d的GetMetaObject方法,返回一個MyMetaObject對象,接著調(diào)用返回對象的BindInvokeMember 方法。
- C#動態(tài)對象(dynamic)詳解(實現(xiàn)方法和屬性的動態(tài))
- C#中Dynamic和Dictionary性能比較
- C#中dynamic關(guān)鍵字的正確用法(推薦)
- 深入C# 4.0 新特性dynamic、可選參數(shù)、命名參數(shù)的詳細介紹
- C# Dynamic關(guān)鍵字之:dynamic為什么比反射快的詳解
- C# Dynamic之:ExpandoObject,DynamicObject,DynamicMetaOb的應(yīng)用(上)
- C# Dynamic關(guān)鍵字之:調(diào)用屬性、方法、字段的實現(xiàn)方法
- C# Dynamic關(guān)鍵字之:解析dynamic就是Object
- C# dynamic關(guān)鍵字的使用方法
- C#使用dynamic類型訪問JObject對象
相關(guān)文章
詳解C# WebApi 接口測試工具:WebApiTestClient
這篇文章主要介紹了詳解C# WebApi 接口測試工具:WebApiTestClient,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-07-07SQL+C#實現(xiàn)獲得當(dāng)前月的第一天與最后一天
本文分享了SQL+C#獲得當(dāng)前月的第一天與最后一天的代碼實例,代碼簡潔,適合初學(xué)者參考。需要的朋友可以看下2016-12-12