- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
假设我不能使用 ExpandoObject 并且必须自己滚动:-
class MyObject : DynamicObject {
dictionary<string, object> _properties = dictionary<string, object>();
public override bool TryGetMember(GetMemberBinder binder, out object result) {
string name = binder.Name.ToLower();
return _properties.TryGetValue(name, out result);
}
public override bool TrySetMember(SetMemberBinder binder, object value) {
_properties[binder.Name.ToLower()] = value;
return true;
}
}
在类层次结构的下方
class MyNewObject : MyObject {
public string Name {
get {
// do some funky stuff
}
set {
// ditto
}
}
}
这很好,因为现在我可以执行以下操作:-
dynamic o = MyNewObject();
o.Age = 87; // dynamic property, handled by TrySetMember in MyObject
o.Name = "Sam"; // non dynamic property, handled by the setter defined in MyNewObject
但上面假设我在编译时知道属性(例如年龄、姓名)。
假设直到运行时我才知道它们会是什么。
如何更改以上内容以支持我只会在运行时知道的属性?
基本上我想我问的是如何调用直接调用 TrySetMember 的代码,以便它创建一个新属性或使用 getter/setter(如果已定义)。
最终解决方案如下:-
using System.Dynamic;
using Microsoft.CSharp.RuntimeBinder;
using System.Runtime.CompilerServices;
class MyObject : DynamicObject {
Dictionary<string, object> _properties = new Dictionary<string, object>();
public object GetMember(string propName) {
var binder = Binder.GetMember(CSharpBinderFlags.None,
propName, this.GetType(),
new List<CSharpArgumentInfo>{
CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null)});
var callsite = CallSite<Func<CallSite, object, object>>.Create(binder);
return callsite.Target(callsite, this);
}
public void SetMember(string propName, object val) {
var binder = Binder.SetMember(CSharpBinderFlags.None,
propName, this.GetType(),
new List<CSharpArgumentInfo>{
CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null),
CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null)});
var callsite = CallSite<Func<CallSite, object, object, object>>.Create(binder);
callsite.Target(callsite, this, val);
}
public override bool TryGetMember(GetMemberBinder binder, out object result) {
string name = binder.Name.ToLower();
return _properties.TryGetValue(name, out result);
}
public override bool TrySetMember(SetMemberBinder binder, object value) {
_properties[binder.Name.ToLower()] = value;
return true;
}
}
最佳答案
虽然 c# 编译器将动态关键字用法转换为使用字符串名称的 dlr 调用,但如果没有编译器的帮助,这些 Api 很难直接使用。开源框架Dynamitey (通过 nuget 作为 PCL 库提供)封装了 dlr API,以便于您只需调用 Impromptu.InvokeSet(target, name, value) .
using Dynamitey;
...
dynamic o = MyNewObject();
Dynamic.InvokeSet(o,"Age" ,87);
Dynamic.InvokeSet(o,"Names" ,"Sam");
Getter 和 Setter 是直接使用实际 Microsoft API 最简单的方法,因此如果您不想使用第 3 方框架,请转至 source也是一种选择。
using Microsoft.CSharp.RuntimeBinder;
using System.Runtime.CompilerServices;
...
dynamic o = MyNewObject();
var binder = Binder.SetMember(CSharpBinderFlags.None,
"Age",
typeof(object),
new List<CSharpArgumentInfo>{
CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null),
CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null)
});
var callsite = CallSite<Func<CallSite, object, object, object>>.Create(binder);
callsite.Target(callsite,o,87);
var binder2 =Binder.SetMember(CSharpBinderFlags.None,
"Name",
typeof(object),
new List<CSharpArgumentInfo>{
CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null),
CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null)
});
var callsite2 = CallSite<Func<CallSite, object, object, object>>.Create(binder2);
callsite2.Target(callsite2,o,"Sam");
没有StackOverflow,没有什么可以改进的
关于C# DynamicObject 动态属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12057516/
我希望能够存储一个 Action 字典,并能够根据字典的键调用那些 Action 。如果未找到 key ,则不会调用任何操作。我有一个想法,即使用扩展 DynamicObject 的对象执行调用并在构
我有一个 DynamicObject 的子类,我想像 DO 的显式转换方法 TryConvert 一样实现原始类型的隐式转换;也就是说,无需编写多个隐式运算符 [type] 函数。 用法: dynam
假设我不能使用 ExpandoObject 并且必须自己滚动:- class MyObject : DynamicObject { dictionary _properties = dicti
这是从 DynamicObject 派生的 DynamicDataObject 类 public class DynamicDataObject : DynamicObject {
我有列表,其中 MyType:DynamicObject。 MyType 继承自 DynamicObject 的原因是我需要一个可以包含未知数量属性的类型。 在我需要过滤列表之前,一切正常。有没有办法
我知道 viewbag 是现成的 DynamicObject 之一,但是如果您不在 View 或 Controller 中,是否还有其他现成的 DynamicObject 除了自己写? (顺便说一句,
我正在实现一个通用函数来从任意提供的动态对象中提取一个值,但不知道如何调用 TryGetMember因为它需要 GetMemberBinder这是抽象的,因此我无法创建它。 样本... public
我正在试验 C# 4.0 的动态对象模型。 我创建了一个名为“Block”的抽象类,它继承自 DynamicObject。它覆盖 TryGetMember 和 TrySetMember。 此外,我创建
假设我有一个继承自 DynamicObject 的类: public class DynamicBase : DynamicObject { public override bool TryGe
在我们的应用程序中,Resources.designer.cs 文件中有 1000 次静态属性的使用。例如: Resources.string_to_identify_the_resource 我们也
我正在编写一个 Javascript C# 桥并遇到了以下问题: 有一个 JSObject 类: public class JSObject : DynamicObject { public
我有一个派生自 DynamicObject 类的类。在调用 JsonConvert.SertializeObject 时,没有任何动态属性被序列化。 类定义为, public clas
现在我有一个扩展 DynamicObject 并覆盖 TryGetMember 的类。 public class FieldCollection : DynamicObject, ICollectio
在 DynamicObject 的文档中,有一个 DynamicDictionary 的示例,它允许您像使用具有属性的类一样使用字典。 这是类(为简洁起见略作修改): public class Dyn
我正在阅读 this关于C# 4.0 中的动态对象的文章。 在该示例中,函数 TryGetMember(GetMemberBinder binder, out object result) 的第二个参
我已成功将事件处理程序添加到我的动态对象中。但是,我无法删除它们。 dynamic d = new MyDynamicObject(); d.MyEvent += new EventHandler(t
我正在尝试设计一个执行提取转换加载操作的过程。我想在我的管道中使用 ExpandoObject 以允许我轻松地将列添加到我的数据流中。基本上,我从某种数据源读取数据,将其转换为动态数据,然后将其返回到
通常,当您覆盖/实现 bool TryDoSomething(...) 方法时,如果它不起作用,您将返回 false。 在 DynamicObject 的情况下,我们返回 false 并且它抛出一个仅
是否可以确定动态成员访问期望的类型?我试过了 dynamic foo = new MyDynamicObject(); int x = foo.IntValue; int y = (int)foo.I
我正在尝试构建一个能够处理通用方法调用的 DynamicObject,但似乎需要的 API - 尽管存在于 RC versions of 4.0 Framework 中- 已在 RTM 中标记为内部(
我是一名优秀的程序员,十分优秀!