Наследование событий с реализацией интерфейса по умолчанию в 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)
{
}
}
Но у нас нет возможности узнать, нужно ли нам добавлять этот обработчик по умолчанию или нет.