您的位置:首页 > 移动开发 > Objective-C

ExpandoObject,DynamicObject,DynamicMetaObject

2014-04-04 19:43 417 查看

ExpandoObject,DynamicObject,DynamicMetaObject

接上文:浅谈Dynamic关键字系列之三(上)

为什么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方法。

4:….

当前标签:dynamic

浅谈Dynamic关键字系列之四:dynamic为什么比反射快
LoveJenny2011-07-0719:58阅读:2523评论:4

浅谈Dynamic关键字系列之三(下):ExpandoObject,DynamicObject,DynamicMetaObject
LoveJenny2011-07-0622:19阅读:2432评论:4

浅谈Dynamic关键字系列之三(上):ExpandoObject,DynamicObject,DynamicMetaObject
LoveJenny2011-07-0519:22阅读:3010评论:2

浅谈Dynamic关键字系列之二:调用属性,方法,字段
LoveJenny2011-07-0419:47阅读:2168评论:4

浅谈Dynamic关键字系列之一:dynamic就是Object
LoveJenny2011-07-0406:27阅读:2668评论:17
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: