1

我需要能够监控大量对象的变化。几乎在任何情况下,我都可以使用INotifyPropertyChanged并收工。然而,我的情况并没有那么简单。我的项目的目标是创建一个可以插入、监控和访问任何对象的模拟。

插入是通过使用 Fluent API 的反射完成的,该 API 指定要包含在模拟环境中的属性和字段。它看起来像这样:

public void DeclareVariables( IVariableBuilder builder )
{
  builder.Include( x => x.PropertyA );
  builder.Include( x => x.FieldA );
}

然后将属性和字段包装在一个Variable类中并存储在一个Environment类中。

public class Variable<T> : IVariable
{
  private FieldInfo _field; // Properties are turned into FieldInfo via k__BackingField
  private object _obj;

  public string Id
  {
    get => $"{_obj.GetType}-{_field.Name}";
  }

  public T Value
  {
    get => _field.GetValue( _obj );
  }
}

public class Environment
{
  private Dictionary<string, IVariable> _variables;
}

这会正确存储对我尝试包含的所有字段和属性的引用。但是,我希望能够监视所有这些更改,并且大多数包含的字段都是闭源类或原语,它们都没有实现 INotifyPropertyChanged 或无法派生。

我来自 JavaScript 背景,您可以通过创建一个可以覆盖属性 getter 和 setter 的对象代理来完成此操作,并且由于 JavaScript 是鸭子类型的,您可以用代理替换实际的对象实例。

有没有办法在 C# 中做同样的事情?我想 C# 的类型安全规则不允许这样的做法,尤其是对于原语。然而,据我所知,监视这些类的唯一方法是以某种方式拦截正在设置值的操作。

4

1 回答 1

3

简而言之,对于任意对象是不可能的。

当您在 JavaScript 中执行此操作时,您可以:

  • 更改现有对象,例如替换一个特定的成员函数,因为没有修改限制
  • 创建一个新的代理对象

对于任意对象,选项一不能在 C# 中完成。

选择选项二,您可以使用具有自定义更改检测逻辑的代理对象包装任意实例。但是,有一些警告:

  1. 为此,所有交互都必须通过代理对象。如果直接修改底层对象,则无法检测到更改。
  2. 代理对象的类型可能与底层的具体类型不兼容。例如,假设您有一个YourCustomClass要包装的密封件。代理类,例如YourCustomClassProxy在 System.Object 之外的层次结构中不共享一个共同的祖先,因此您将无法用方法调用的YourCustomClass实例YourCustomClassProxy等替换 的实例。因此,虽然有成熟的库用于创建动态代理(参见Castle Project),它们支持接口和虚拟成员,因为它们可以在运行时被拦截而不会违反类型契约。

从理论上讲,可以重写 IL以添加您期望甚至封闭类的行为,但我认为它不实用。

总而言之,可用的策略是:

  • 组合新的(代理)对象INotifyPropertyChanged,从那些没有实现的对象中实现,并仅通过代理应用更改
  • 如果上述方法不可行,请查询对象以进行更改。假设您有一个 GUI 应用程序并且更改是由用户交互引起的,您有时可以预见哪些 GUI 会更改并强制刷新。
于 2019-03-09T19:51:03.460 回答