gpt4 book ai didi

c# - 为什么我不能对从 DynamicObject 继承的对象使用 GetType().GetProperty()?

转载 作者:行者123 更新时间:2023-11-30 12:48:31 25 4
gpt4 key购买 nike

我正在尝试使用反射从动态类中检索字段或属性,但是当我使用 Getfield 或 GetProperty 调用动态对象时,它永远找不到该字段,并且没有输入任何动态对象。Try* 方法被输入。

不确定为什么这在 .net4 上不起作用。

请参阅下面的 testdyn 动态测试类。

我是这样调用它的:

        dynamic td = new testdyn();
td.SendDebugEvent += new DebugDelegate(debug);
td.test();

得到这些结果:

one = -1 
two = -1
fiddle = -1
test = -1
set: fiddle = 241827974
fiddle = -1

期待看到

one = 1
two = 2
fiddle = 3
test = -1
set: fiddle = 241827974
fiddle = 241827974

我做错了什么?

注意:如果我调用“td.fiddle”它确实有效......但是你不知道创建类的名称但你知道它来访问它似乎很奇怪?

从这篇文章看来,动态对象可能不支持反射,因为它实现了 idynamicmetaobjectprovider

How do I reflect over the members of dynamic object?

问题是此代码正被使用反射的外部应用程序使用。

如果你有想法,请告诉我。

代码如下。

   public delegate void DebugDelegate(string msg);
public class testdyn : System.Dynamic.DynamicObject
{
List<string> items = new List<string>(new string[] { "one", "two", "fiddle", "my", "lou" });
List<int> vals = new List<int>( new int[] { 1,2,3,5,8 });

public event DebugDelegate SendDebugEvent;
void debug(string msg)
{
if (SendDebugEvent!=null)
SendDebugEvent(msg);
}

public void set(string name, int v)
{
var idx = items.IndexOf(name);
if (idx < 0)
return;
vals[idx] = v;
debug("set: " + name + " = " + v);
}

int get(string name)
{
object o = null;
var t = GetType();
try {
o = t.GetProperty(name).GetValue(this, null);
int v = (int)o;
return v;
} catch
{
try
{
var f = t.GetField(name);
o = f.GetValue(this);
return (int)o;
}
catch
{

}
}
return -1;
}

string g(string name) { return name+" = "+get(name).ToString(); }
Random r = new Random();
public void test() { test(string.Empty); }
public void test(string mytmp)
{
var t = GetType();
// do some reads
debug(g("one"));
debug(g("two"));
debug(g("fiddle"));
debug(g("test"));
// do some sets
set("fiddle", r.Next());
// they should change
debug(g("fiddle"));
}

public override bool TryInvokeMember(System.Dynamic.InvokeMemberBinder binder, object[] args, out object result)
{
debug("got invoke member");
return base.TryInvokeMember(binder, args, out result);
}

public override bool TrySetMember(System.Dynamic.SetMemberBinder binder, object value)
{
debug("got setmember");
return base.TrySetMember(binder, value);
}

public override bool TryGetIndex(System.Dynamic.GetIndexBinder binder, object[] indexes, out object result)
{
debug("got getindex");
return base.TryGetIndex(binder, indexes, out result);
}

public override bool TryGetMember(System.Dynamic.GetMemberBinder binder, out object result)
{
// get index of column value trying to be retrieved
var idx = items.IndexOf(binder.Name);
// default to empty
result = string.Empty;
// return error if we can't find
if (idx < 0)
{
return base.TryGetMember(binder, out result);
}
// get result
result = vals[idx];
return true;
}

public override bool TryInvoke(System.Dynamic.InvokeBinder binder, object[] args, out object result)
{
debug("got invoke");
return base.TryInvoke(binder, args, out result);
}



public override bool TryCreateInstance(System.Dynamic.CreateInstanceBinder binder, object[] args, out object result)
{
debug("got create instance");
return base.TryCreateInstance(binder, args, out result);
}

public override IEnumerable<string> GetDynamicMemberNames()
{
debug("got member names");
return items.ToArray();
}
}

最佳答案

反射不适用于动态属性,(尽管如果您实现 ICustomTypeProvider 可能会有一个 .net4.5 异常(exception))

我写了一个名为 ImpromptuInterface 的开源 DLR 瑞士军刀框架(在 nuget 中可用)。出于外部代码使用反射动态访问属性的确切原因,我在其中设计了一个静态方法来桥接动态属性。 Impromptu.ActLikeProperties(this object originalDynamic, IDictionary<string, Type>propertySpec) ActLikeProperties .

问题是您需要向我的方法提供属性名称和返回类型的字典,然后将结果传递给外部 api,我的方法通过包装您的 DynamicObject 来工作。带有一个发出的代理,该代理使用 dlr 将您在字典中描述的属性调用从它的静态定义转发到您的动态类型。

td.ActLikeProperties(new Dictionary<string,type>{{"one":typeof(int)},{"two":typeof(int) },{"fiddle":typeof(int) },{"test":typeof(int) }});

关于c# - 为什么我不能对从 DynamicObject 继承的对象使用 GetType().GetProperty()?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13964332/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com