浅谈Dynamic 关键字系列之三:ExpandoObject, DynamicObject, DynamicMetaObject
2012-12-24 01:44
495 查看
ExpandoObject:表示一个对象,该对象包含可在运行时动态添加和移除的成员。
dynEO.number中number是动态添加属性。
dynEO.Increment中Increment是动态添加的Action委托。
枚举ExpandoObject的所有成员:
结果如下:
接收属性更改的通知:
结果:
System.Dynamic.DynamicObject:提供用于指定运行时的动态行为的基类
新建类DynamicProduct,基本和Product类似:
可以看到继承了DynamicObject后,可以override一大堆TryXXX的方法了。
重点需要了解的是:
完整的代码如下:
Main函数代码如下:
结果如下:
理论上来说,应该输出:
TrySetMember:设置name字段
TrySetMember:设置Id属性
TryGetMember:获取Id属性
TrySetMember:设置Id属性
TryInvokeMember:调用ShowProduct方法
Id=4,Name=n1
为什么TryXXX方法没有被调用??
将DynamicProduct中的name修饰符改为private:
privatestringname;
可以在TrySetMember方法中设置断点,再次运行:
为什么访问修饰符是Public不调用TrySetMember,是Private就调用了呢??
难道是因为private抛出了异常吗??
再次看看Msdn对此的TrySetMember方法的解释:
Msdn备注
…………….动态语言运行库(DLR)将首先使用语言联编程序在类中查找属性的静态定义。如果没有此类属性,DLR调用TrySetMember方法。
问题的原因是这样的:首先DLR使用语言联编程序在类中查找name的静态定义,
因为name是public,所以查找到了,然后返回,不会去调用TrySetMember方法了,
但是如果name是private,那么联编程序在类中没找到name的静态定义,于是DLR尝试调用TrySetMember方法。
修改TrySetMember方法如下:
运行,可以发现不会抛出异常了:
总结:首先DLR会尝试查找属性的静态定义,如果没有找到则会调用相应的TryXXX方法,如果TryXXX方法返回false,代表TryXXX方法运行失败,DLR随后会抛出异常。
为了验证是不是这样,将DynamicProduct中属性的静态定义全部注释掉,并且TryXXX方法全部返回True。完整的代码如下:
Main方法不变:
运行结果如下:
DynamicMetaObject:表示动态绑定和参与动态绑定的对象的绑定逻辑。
新建类MyDynamicObject:
Main方法如下:
运行,结果如下:
d.P3=d.M1(d.P1,d.M2(d.P2));
按照从左到右,从里到外的原则。
1:先调用d.P1,DLR会尝试调用d的GetMetaObject方法,此方法返回一个MyMetaObject对象。
接着DLR知道你调用的是一个属性,于是它调用返回的MyMetaObject对象的BindGetMember方法,
输出为GetMemberofpropertyP1
2:调用d.P2,和调用d.P1一样.
3:调用d.M2,同样DLR调用d的GetMetaObject方法,返回一个MyMetaObject对象,接着调用返回对象的BindInvokeMember方法。
dynamicdynEO=newExpandoObject(); dynEO.number=10; dynEO.Increment=newAction(()=>{dynEO.number++;}); Console.WriteLine(dynEO.number); dynEO.Increment(); Console.WriteLine(dynEO.number);
dynEO.number中number是动态添加属性。
dynEO.Increment中Increment是动态添加的Action委托。
枚举ExpandoObject的所有成员:
foreach(varpropertyin(IDictionary<String,Object>)dynEO)
{
Console.WriteLine(property.Key+":"+property.Value);
}
结果如下:
接收属性更改的通知:
staticvoidMain(string[]args)
{
………
((INotifyPropertyChanged)dynEO).PropertyChanged+=newPropertyChangedEventHandler(Program_PropertyChanged);
dynEO.Name="changed";
dynEO.Name="another";
Console.ReadLine();
}
staticvoidProgram_PropertyChanged(objectsender,PropertyChangedEventArgse)
{
Console.WriteLine("属性{0}已更改",e.PropertyName);
}
结果:
System.Dynamic.DynamicObject:提供用于指定运行时的动态行为的基类
新建类DynamicProduct,基本和Product类似:
可以看到继承了DynamicObject后,可以override一大堆TryXXX的方法了。
重点需要了解的是:
假设sampleObject就是dynamicObject | |
TryGetMember | 在调用intnumber=sampleObject.Number.时使用 |
TrySetMember | 在调用sampleObject.Number=number时使用 |
TryInvoke | 在调用sampleObject(100)时使用 |
TryInvokeMember | 在调用sampleObject.someMethod(100)时使用 |
classDynamicProduct:DynamicObject
{
publicstringname;
publicintId{get;set;}
publicvoidShowProduct()
{
Console.WriteLine("Id={0},Name={1}",Id,name);
}
#regionOverrideDynamicObject的方法
publicoverrideIEnumerable<string>GetDynamicMemberNames()
{
returnbase.GetDynamicMemberNames();
}
publicoverrideboolTryGetMember(GetMemberBinderbinder,outobjectresult)
{
Console.WriteLine("TryGetMember被调用了,Name:{0}",binder.Name);
returnbase.TryGetMember(binder,outresult);
}
publicoverrideboolTrySetMember(SetMemberBinderbinder,objectvalue)
{
Console.WriteLine("TrySetMember被调用了,Name:{0}",binder.Name);
returnbase.TrySetMember(binder,value);
}
publicoverrideboolTryInvoke(InvokeBinderbinder,object[]args,outobjectresult)
{
Console.WriteLine("TryInvoke被调用了");
returnbase.TryInvoke(binder,args,outresult);
}
publicoverrideboolTryInvokeMember(InvokeMemberBinderbinder,object[]args,outobjectresult)
{
Console.WriteLine("TryInvokeMember被调用了,Name:{0}",binder.Name);
returnbase.TryInvokeMember(binder,args,outresult);
}
#endregion
}
Main函数代码如下:
staticvoidMain(string[]args)
{
dynamicdynProduct=newDynamicProduct();
dynProduct.name="n1";//调用TrySetMember方法
dynProduct.Id=1;
dynProduct.Id=dynProduct.Id+3;
dynProduct.ShowProduct();
Console.ReadLine();
}
结果如下:
理论上来说,应该输出:
TrySetMember:设置name字段
TrySetMember:设置Id属性
TryGetMember:获取Id属性
TrySetMember:设置Id属性
TryInvokeMember:调用ShowProduct方法
Id=4,Name=n1
为什么TryXXX方法没有被调用??
将DynamicProduct中的name修饰符改为private:
privatestringname;
可以在TrySetMember方法中设置断点,再次运行:
为什么访问修饰符是Public不调用TrySetMember,是Private就调用了呢??
难道是因为private抛出了异常吗??
再次看看Msdn对此的TrySetMember方法的解释:
Msdn备注
…………….动态语言运行库(DLR)将首先使用语言联编程序在类中查找属性的静态定义。如果没有此类属性,DLR调用TrySetMember方法。
问题的原因是这样的:首先DLR使用语言联编程序在类中查找name的静态定义,
因为name是public,所以查找到了,然后返回,不会去调用TrySetMember方法了,
但是如果name是private,那么联编程序在类中没找到name的静态定义,于是DLR尝试调用TrySetMember方法。
修改TrySetMember方法如下:
publicoverrideboolTrySetMember(SetMemberBinderbinder,objectvalue)
{
Console.WriteLine("TrySetMember被调用了,Name:{0}",binder.Name);
boolresult=base.TrySetMember(binder,value);
returntrue;
}
运行,可以发现不会抛出异常了:
总结:首先DLR会尝试查找属性的静态定义,如果没有找到则会调用相应的TryXXX方法,如果TryXXX方法返回false,代表TryXXX方法运行失败,DLR随后会抛出异常。
为了验证是不是这样,将DynamicProduct中属性的静态定义全部注释掉,并且TryXXX方法全部返回True。完整的代码如下:
classDynamicProduct:DynamicObject
{
#regiondynamicProduct的一些属性的静态定义
//privatestringname;
//publicintId{get;set;}
//publicvoidShowProduct()
//{
//Console.WriteLine("Id={0},Name={1}",Id,name);
//}
#endregion
#regionOverrideDynamicObject的方法
publicoverrideboolTryGetMember(GetMemberBinderbinder,outobjectresult)
{
Console.WriteLine("TryGetMember被调用了,Name:{0}",binder.Name);
booltryResult=base.TryGetMember(binder,outresult);
returntrue;
}
publicoverrideboolTrySetMember(SetMemberBinderbinder,objectvalue)
{
Console.WriteLine("TrySetMember被调用了,Name:{0}",binder.Name);
booltryResult=base.TrySetMember(binder,value);
returntrue;
}
publicoverrideboolTryInvoke(InvokeBinderbinder,object[]args,outobjectresult)
{
Console.WriteLine("TryInvoke被调用了");
booltryResult=base.TryInvoke(binder,args,outresult);
returntrue;
}
publicoverrideboolTryInvokeMember(InvokeMemberBinderbinder,object[]args,outobjectresult)
{
Console.WriteLine("TryInvokeMember被调用了,Name:{0}",binder.Name);
booltryResult=base.TryInvokeMember(binder,args,outresult);
returntrue;
}
#endregion
}
Main方法不变:
staticvoidMain(string[]args)
{
dynamicdynProduct=newDynamicProduct();
dynProduct.name="n1";//调用TrySetMember方法
dynProduct.Id=1;
dynProduct.Id=dynProduct.Id+3;
dynProduct.ShowProduct();
Console.ReadLine();
}
运行结果如下:
DynamicMetaObject:表示动态绑定和参与动态绑定的对象的绑定逻辑。
新建类MyDynamicObject:
publicclassMyMetaObject:DynamicMetaObject
{
publicMyMetaObject(Expressionparameter,objectvalue)
:base(parameter,BindingRestrictions.Empty,value)
{
}
publicoverrideDynamicMetaObjectBindInvokeMember(InvokeMemberBinderbinder,DynamicMetaObject[]args)
{
returnthis.PrintAndReturnIdentity("InvokeMemberofmethod{0}",binder.Name);
}
publicoverrideDynamicMetaObjectBindSetMember(SetMemberBinderbinder,DynamicMetaObjectvalue)
{
returnthis.PrintAndReturnIdentity("SetMemberofproperty{0}",binder.Name);
}
publicoverrideDynamicMetaObjectBindGetMember(GetMemberBinderbinder)
{
returnthis.PrintAndReturnIdentity("GetMemberofproperty{0}",binder.Name);
}
privateDynamicMetaObjectPrintAndReturnIdentity(stringmessage,stringname)
{
Console.WriteLine(String.Format(message,name));
returnnewDynamicMetaObject(
Expression,
BindingRestrictions.GetTypeRestriction(
Expression,
typeof(MyDynamicObject)));
}
}
Main方法如下:
staticvoidMain(string[]args)
{
dynamicd=newMyDynamicObject();
d.P3=d.M1(d.P1,d.M2(d.P2));
Console.ReadLine();
}
运行,结果如下:
d.P3=d.M1(d.P1,d.M2(d.P2));
按照从左到右,从里到外的原则。
1:先调用d.P1,DLR会尝试调用d的GetMetaObject方法,此方法返回一个MyMetaObject对象。
接着DLR知道你调用的是一个属性,于是它调用返回的MyMetaObject对象的BindGetMember方法,
输出为GetMemberofpropertyP1
2:调用d.P2,和调用d.P1一样.
3:调用d.M2,同样DLR调用d的GetMetaObject方法,返回一个MyMetaObject对象,接着调用返回对象的BindInvokeMember方法。
相关文章推荐
- 浅谈Dynamic 关键字系列之三(下):ExpandoObject,DynamicObject,DynamicMetaObject
- 浅谈Dynamic 关键字系列之三(上):ExpandoObject, DynamicObject, DynamicMetaObject
- 浅谈Dynamic 关键字系列之三(上):ExpandoObject, DynamicObject, DynamicMetaObject
- 浅谈Dynamic 关键字系列之四:dynamic为什么比反射快
- 浅谈Dynamic 关键字系列之一:dynamic 就是Object
- 浅谈Dynamic 关键字系列之一:dynamic 就是Object
- ExpandoObject,DynamicObject,DynamicMetaObject
- 浅谈Dynamic 关键字系列之一:dynamic 就是Object(转)
- C#4.0的dynamic和var及object关键字辨析
- C#4.0的dynamic和var及object关键字辨析
- 浅谈Dynamic 关键字系列之二:调用属性,方法,字段
- [翻译]关键字“dynamic”和“object”(及“var”)有什么不同?
- C# Dynamic之:ExpandoObject,DynamicObject,DynamicMetaOb的应用(上)
- C#4.0中的dynamic关键字和ExpandoObject对象
- 浅谈Dynamic 关键字系列之二:调用属性,方法,字段
- C# Dynamic关键字之:解析dynamic就是Object
- 浅谈Dynamic 关键字系列之四:dynamic为什么比反射快
- C# Dynamic之:ExpandoObject,DynamicObject,DynamicMetaOb的应用(下)
- Effective C# 学习笔记(四十一)使利用DynamicObject或IDynamicMetaObjectProvider接口实现数据驱动的动态类型
- C#4.0中的dynamic关键字和ExpandoObject对象