Наследование событий с реализацией интерфейса по умолчанию в C#8 / чертами

В настоящее время существует мало документации, связанной с ограничениями событий с новыми реализациями интерфейса C#8 по умолчанию (трейтами). Меня особенно смущает предложение спецификации.В примере не только указан недопустимый C# (для события "переопределение" отсутствует идентификатор), но и реализация любого из них в C#8 (VS2019, .NET Core 3.0) возвращает множество исключений компилятора. Кроме того, в примечаниях к выпуску C#8 не упоминаются события для свойств интерфейса. Продолжая искать ответ, я также не смог собрать ничего полезного из списка открытых проблем.

Итак, вопросы: реализована ли и пригодна ли эта функция? Если да, то каков правильный синтаксис?

1 ответ

Решение

Члены интерфейса по умолчанию используются для признаков, а не только для управления версиями, и признак INPC имеет смысл.

К сожалению, прямо сейчас невозможно использовать DIM для создания событий, и реализация этого кажется проблемой - это потребует капитального ремонта механизма событий и поломки тонны кода, особенно кода библиотеки. Мы можем использовать модули DIM для добавления или удаления обработчиков, но это не так полезно.

Было бы неплохо иметь что-то вроде:

interface InpcTrait : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;

    private T Set(T value,String propertyName = "")
    {
        if (PropertyChanged != null)
        {
            PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
        }
        return value;
    }
}

class Customer
{
    private string _name;
    public string Name 
    {
        get=>_name;
        set=>_name=Set(value,"Name");
    }
}


К сожалению, это невозможно. Это потому чтоeventключевое слово в классе создает резервное поле, которое содержит обработчик событий и добавляет / удаляет средства доступа. Когда мы вызываем событие, мы вызываем этот обработчик события.

Интерфейсы не могут иметь состояния, а это значит, что мы не можем получить доступ к этому событию, чтобы вызвать его.

Когда мы указываем событие в интерфейсе, мы создаем виртуальное событие, и компилятор позволяет только добавлять / удалять к нему обработчики событий. Для поднятия интерфейса по-прежнему требуется доступ к полю поддержки.

Этот пример Sharplab.io показывает, что:

public class DemoCustomer : INotifyPropertyChanged
{
    // These fields hold the values for the public properties.
    private Guid idValue = Guid.NewGuid();
    private string customerNameValue = String.Empty;
    private string phoneNumberValue = String.Empty;

    public event PropertyChangedEventHandler PropertyChanged;

    private void NotifyPropertyChanged(String propertyName = "")
    {
        if (PropertyChanged != null)
        {
            PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
        }
    }
}

Генерирует

    [CompilerGenerated]
    private PropertyChangedEventHandler m_PropertyChanged;

    public event PropertyChangedEventHandler PropertyChanged
    {
        [CompilerGenerated]
        add
        {
            //some code
        }
        [CompilerGenerated]
        remove
        {
            //some code
        }
    }

    private void NotifyPropertyChanged(string propertyName = "")
    {
        if (this.m_PropertyChanged != null)
        {
            this.m_PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
        }
    }

Что мы можем сделать, так это добавить или удалить обработчики событий, но мы даже не можем проверить, есть ли у события уже другие обработчики. Мы рискуем добавить один и тот же обработчик событий несколько раз.

Это действительно:

interface INPCtrait:System.ComponentModel.INotifyPropertyChanged
{            
    private  void AddSomeDefaultHandler()
    {
       PropertyChanged+=Something;
    }

    private  void RemoveDefaultHandler()
    {
       PropertyChanged-=Something;
    }

    public void Something(Object sender,System.ComponentModel.PropertyChangedEventArgs args)
    {
    }    
}

Но у нас нет возможности узнать, нужно ли нам добавлять этот обработчик по умолчанию или нет.

Другие вопросы по тегам