.net - INotifyPropertyChanged 属性名称 - 硬编码与反射?

标签 .net wpf reflection

使用 INotifyPropertyChanged 时指定属性名称的最佳方式是什么?

大多数示例将属性名称硬编码为 PropertyChanged 事件的参数。我正在考虑使用 MethodBase.GetCurrentMethod.Name.Substring(4) 但对反射开销有点不安。

最佳答案

不要忘记一件事:PropertyChanged 事件主要由将使用反射来获取指定属性的值的组件消耗。

最明显的例子是数据绑定(bind)。

当您触发 PropertyChanged 事件并将属性名称作为参数传递时,您应该知道此事件的订阅者可能会通过调用来使用反射,例如,GetProperty(如果使用 PropertyInfo 缓存,至少是第一次),然后是 GetValue。最后一个调用是属性 getter 方法的动态调用 (MethodInfo.Invoke),其成本高于仅查询元数据的 GetProperty。 (请注意,数据绑定(bind)依赖于整个 TypeDescriptor 事物——但默认实现使用反射。)

因此,当然,在触发 PropertyChanged 时使用硬代码属性名称比使用反射动态获取属性名称更有效,但恕我直言,平衡您的想法很重要。在某些情况下,性能开销并不那么重要,您可以从某种强类型事件触发机制中受益。

以下是我有时在 C# 3.0 中使用的方法,此时性能不是问题:

public class Person : INotifyPropertyChanged
{
    private string name;

    public string Name
    {
        get { return this.name; }
        set 
        { 
            this.name = value;
            FirePropertyChanged(p => p.Name);
        }
    }

    private void FirePropertyChanged<TValue>(Expression<Func<Person, TValue>> propertySelector)
    {
        if (PropertyChanged == null)
            return;

        var memberExpression = propertySelector.Body as MemberExpression;
        if (memberExpression == null)
            return;

        PropertyChanged(this, new PropertyChangedEventArgs(memberExpression.Member.Name));
    }

    public event PropertyChangedEventHandler PropertyChanged;
}

请注意使用表达式树来获取属性名称,以及使用 lambda 表达式作为表达式:

FirePropertyChanged(p => p.Name);

关于.net - INotifyPropertyChanged 属性名称 - 硬编码与反射?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/141370/

相关文章:

C# 如何在参数数量相等时使用 System.Reflection 调用私有(private)重载方法

.net - 如何通过反射引用字段

c# - 当 TextBox 插入符号更改时,是否有机会引发事件?

c# - 如何在德里访问类的成员(用 C# 创建)

c# - 按住右键单击与 Manipulation 事件冲突

c# - 找不到类型或命名空间名称 'PrintDocument'(是否缺少 using 指令或程序集引用?)

c# - 获取所有引用程序集的路径

java - 将 Object[] 转换为另一种类型

c# - 在 ListView 中选择一行

.net - Datagridview 的行自动调整大小