Как упоминалось выше, лямбда-выражениям теперь отдается большее предпочтение по сравнению с анонимными методами, поэтому начнем именно с них. Ниже приведен пример программы, в которой лямбда-выражение используется в качестве обработчика событий.
// Использовать лямбда-выражение в качестве обработчика событий, using System;
// Объявить тип делегата для события, delegate void MyEventHandler(int n);
// Объявить класс, содержащий событие, class MyEvent {
public event MyEventHandler SomeEvent;
// Этот метод вызывается для запуска события, public void OnSomeEvent(int n) { if(SomeEvent != null)
SomeEvent(n);
}
}
class LambdaEventDemo { static void Main() {
MyEvent evt = new MyEvent();
// Использовать лямбда-выражение в качестве обработчика событий, evt.SomeEvent += (n) =>
Console.WriteLine("Событие получено. Значение равно " + п);
// Запустить событие, evt.OnSomeEvent(1); evt.OnSomeEvent(2);
}
}
Вот к какому результату приводит выполнение этой программы.
Событие получено. Значение равно 1 Событие получено. Значение равно 2
Обратите особое внимание на то, как в этой программе лямбда-выражение используется в качестве обработчика событий.
evt.SomeEvent += (n) =>
Console.WriteLine("Событие получено. Значение равно " + п);
Синтаксис для использования лямбда-выражения в качестве обработчика событий остается таким же, как для его применения вместе с любым другим типом делегата.
Несмотря на то что при создании анонимной функции предпочтение следует теперь отдавать лямбда-выражениям, в качестве обработчика событий можно по-прежнему использовать анонимный метод. Ниже приведен вариант обработчика событий из предыдущего примера, измененный с целью продемонстрировать применение анонимного метода.
11 Использовать анонимный метод в качестве обработчика событий, evt.SomeEvent += delegate(int n) {
Console.WriteLine("Событие получено. Значение равно " + n);
};
Как видите, синтаксис использования анонимного метода в качестве обработчика событий остается таким же, как и для его применения вместе с любым другим типом делегата.
Рекомендации по обработке событий в среде .NET Framework
В C# разрешается формировать какие угодно разновидности событий. Но ради совместимости программных компонентов со средой .NET Framework следует придерживаться рекомендаций, установленных для этой цели корпорацией Microsoft. Эти рекомендации, по существу, сводятся к следующему требованию: у обработчиков событий должны быть два параметра. Первый из них — ссылка на объект, формирующий событие, второй — параметр типа EventArgs, содержащий любую дополнительную информацию о событии, которая требуется обработчику. Таким образом, .NET-совместимые обработчики событий должны иметь следующую общую форму.
void обработчик(object отправитель, EventArgs е) {
// ...
}
Как правило, отправитель — это параметр, передаваемый вызывающим кодом с помощью ключевого слова this. А параметр е типа EventArgs содержит дополнительную информацию о событии и может быть проигнорирован, если он не нужен.
Сам класс EventArgs не содержит поля, которые могут быть использованы для передачи дополнительных данных обработчику. Напротив, EventArgs служит в качестве базового класса, от которого получается производный класс, содержащий все необходимые поля. Тем не менее в классе EventArgs имеется одно поле Empty типа static, которое представляет собой объект типа EventArgs без данных.
Ниже приведен пример программы, в которой формируется .NET-совместимое событие.
// Пример формирования .NET-совместимого события, using System;
// Объявить класс, производный от класса EventArgs. class MyEventArgs : EventArgs { public int EventNum;
}
// Объявить тип делегата для события.
delegate void MyEventHandler(object source, MyEventArgs arg);
/’/ Объявить класс, содержащий событие, class MyEvent {
static int count = 0;
// Этот метод запускает событие SomeEvent. public void OnSomeEvent() {
MyEventArgs arg = new MyEventArgs();
if(SomeEvent != null) { arg.EventNum = count++;