112

有很好的文章提出了不同的实现方式INotifyPropertyChanged

考虑以下基本实现:

class BasicClass : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;

    private void FirePropertyChanged(string propertyName)
    {
        var handler = PropertyChanged;
        if (handler != null)
            handler(this, new PropertyChangedEventArgs(propertyName));
    }

    private int sampleIntField;

    public int SampleIntProperty
    {
        get { return sampleIntField; }
        set
        {
            if (value != sampleIntField)
            {
                sampleIntField = value;
                FirePropertyChanged("SampleIntProperty"); // ouch ! magic string here
            }
        }
    }
}

我想用这个替换它:

using System.Runtime.CompilerServices;

class BetterClass : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;
    // Check the attribute in the following line :
    private void FirePropertyChanged([CallerMemberName] string propertyName = null)
    {
        var handler = PropertyChanged;
        if (handler != null)
            handler(this, new PropertyChangedEventArgs(propertyName));
    }

    private int sampleIntField;

    public int SampleIntProperty
    {
        get { return sampleIntField; }
        set
        {
            if (value != sampleIntField)
            {
                sampleIntField = value;
                // no "magic string" in the following line :
                FirePropertyChanged();
            }
        }
    }
}

但有时我读到该[CallerMemberName]属性与替代品相比性能较差。这是真的吗?为什么?它使用反射吗?

4

1 回答 1

224

不,使用[CallerMemberName]不比上层基本实现慢。

这是因为,根据这个 MSDN 页面

调用者信息值在编译时作为文字发送到中间语言 (IL)

我们可以使用任何 IL 反汇编程序(如ILSpy)检查这一点:属性的“SET”操作的代码以完全相同的方式编译: 带有 CallerMemberName 的反编译属性

所以这里没有使用反射。

(用VS2013编译的示例)

于 2014-03-22T16:55:44.403 回答