【发布时间】:2010-10-13 08:58:44
【问题描述】:
有没有办法在运行时将Field(或FieldInfo,也许是一样的)添加到类中?
【问题讨论】:
标签: c# reflection
有没有办法在运行时将Field(或FieldInfo,也许是一样的)添加到类中?
【问题讨论】:
标签: c# reflection
不,C# 不允许monkey-patching。
您可以使用 CodeDOM 或 Reflection.Emit 生成新的类,但不能修改现有的。
【讨论】:
您不能在运行时更改类定义。但是,您可以创建一个继承自原始类的新类(如果它不是sealed)并声明该字段。您可以通过使用System.Reflection.Emit 发出适当的IL 代码来做到这一点。
【讨论】:
不完全是。
但是,您可以实现 ICustomTypeDescriptor 来近似它,然后只需使用哈希表来存储字段名/值对。许多使用反射的框架首先要求 ICustomTypeDescriptor。
【讨论】:
直到 C# 4.0 添加了动态查找并基于包含 DLR 的 CLR 4.0,然后它不会严格添加到类中,因为类不会出现在图片中。
【讨论】:
C# 不允许这样做,因为它的所有类都基于元数据。 CLR(不是 C#)不允许在运行时向元数据添加字段 (1)。这是 C# 能够在 runitme 添加字段的唯一方法。
这与 IronPython 等动态语言不同,后者本质上没有具体的元数据类。它们有更多的动态结构,可以在运行时改变。我相信 IronPython 只是将它的成员(字段和方法)保存在相当于一个可以在运行时轻松更改的哈希表中。
在 C# 3.0 中,最好的资源是使用 Reflection.Emit。但这将生成一个全新的类,而不是改变现有的类。
(1) 某些 API(例如分析 API 或 ENC)允许这样做,但我不确定它们的功能是否扩展到添加字段。
【讨论】:
正如其他人已经说过的,这是不可能的。你的问题的原因是什么?如果您需要在类中动态存储一些额外的数据,那么您可能只使用字典:
class My {
Dictionary<string, object> data;
public My() { data = new Dictionary<string, object>(); }
}
.. 但这真的取决于你真正想要达到的目标?
【讨论】:
请参阅this question 了解在 C# 4.0 中执行此操作的简单方法,将ExpandoObject 强制转换为IDictionary,这会使添加到字典中的值成为对象的属性。我在another answer of mine 中演示了这一点,表明它们实际上变成了属性。
但是,这只能通过 ExpandoObject 的实例或通过子类化 DynamicObject 来实现。
如果其他类实现了IDynamicMetaObjectProvider,也可以与其他类一起使用,但我认为要使其正常工作涉及到太多工作。
【讨论】:
也许你可以使用装饰器模式。
在面向对象的编程中,装饰器模式是一种设计模式,它允许将行为静态或动态添加到单个对象,而不会影响同一类中其他对象的行为。
【讨论】: