События равны нулю, когда на них не подписаны клиенты, поэтому попытка вызвать событие, у которого нет подписчиков, завершится с NullReferenceException.
Некоторые распространенные приемы, позволяющие избежать этого:
1) Проверка на нулевое значение в поточно-ориентированном режиме (с точки зрения сборщика событий; на клиенте все еще есть состояние гонки, однако ответственность за это лежит на них)
var handler = this.ModemCommEvent;
if( handler != null ) {
handler(this, new ModemCommEventArgs( ModemCommEventArgs.eModemCommEvent.Connected ));
}
Приведенный выше код является более сложной версией этого:
if( this.ModemCommEvent != null ) {
this.ModemCommEvent(this, new ModemCommEventArgs(ModemCommEventArgs.eModemCommEvent.Connected));
}
Первый (который создает локальную переменную) безопаснее с точки зрения сборщика событий, потому что локальная переменная будет либо нулевой, либо нет, и ничто не изменит это. Во втором случае, однако, клиент, работающий в отдельном потоке, может отписаться от события в промежутке между временем, когда выполняется проверка на нулевое значение, и тем, когда вы инициируете событие. В этом случае вы снова получите исключение NullReferenceException. Если ни вы, ни клиенты вашего кода не выполняются в нескольких потоках (без BackgroundWorker, объекта Thread, асинхронного вызова и т. Д.), То более безопасная проверка избыточна. Однако, если вы не уверены, это хорошая практика. Это, или сделайте # 2.
2) По умолчанию ваше событие пустое значение
public event CommanderEventHandler ModemCommEvent = delegate { };
Это полностью устраняет проблему, всегда имея хотя бы одного подписчика. Синтаксис делегат {} создает анонимный метод, который не делает ничего, что является «подписчиком по умолчанию» для события. Независимо от того, сколько клиентов подписались или отписались от вашего события, этот анонимный метод всегда будет там, чтобы ваше событие не было нулевым.
-
Это обсуждалось до тошноты по всему Интернету. Вот один из таких примеров:
C # События и безопасность потоков