Sự kiện (Event) trong C#
Event là một khái niệm khá phổ biến trong lập trình và được sư dụng với mô hình publisher. Trong C#, Event được sử dụng trong các mô hình lập trình đồ họa. Event và Delegate có quan hệ gần gũi và thường gây nhầm lẫn. Trong bài viết dưới đây, eLib sẽ tìm hiểu chi tiết về Event và phân biệt nó với Delegate.
Mục lục nội dung
1. Event trong C#
Sự kiện (Event) là các hành động của người dùng, ví dụ như nhấn phím, click, di chuyển chuột, … Các Application cần phản hồi các sự kiện này khi chúng xuất hiện. Ví dụ, các ngắt (interrupt). Các sự kiện (Event) được sử dụng để giao tiếp bên trong tiến trình.
2. Sử dụng Delegate với Event trong C#
Các Event được khai báo và được tạo trong một lớp và được liên kết với Event Handler bởi sử dụng các Delegate bên trong cùng lớp đó hoặc một số lớp khác. Lớp mà chứa Event được sử dụng để công bố event đó. Điều này được gọi là lớp Publisher. Một số lớp khác mà chấp nhận Event này được gọi là lớp Subscriber. Các Event trong C# sử dụng mô hình Publisher-Subscriber.
Một Publisher trong C# là một đối tượng mà chứa định nghĩa của event và delegate đó. Mối liên hệ event-delegate cũng được định nghĩa trong đối tượng này. Một đối tượng lớp Publisher triệu hồi Event và nó được thông báo tới các đối tượng khác.
Một Subscriber trong C# là một đối tượng mà chấp nhận event và cung cấp một Event Handler. Delegate trong lớp Publisher triệu hồi phương thức (Event Handler) của lớp Subscriber.
3. Khai báo Event trong C#
Để khai báo một Event bên trong một lớp, đầu tiên một kiểu delegate cho Event đó phải được khai báo. Ví dụ:
public delegate void BoilerLogHandler(string status);
Tiếp theo, chính Event đó được khai báo, bởi sử dụng từ khóa event trong C#:
//định nghĩa event dựa vào delegate ở trên
public event BoilerLogHandler BoilerEventLog;
Code trên định nghĩa một delegate với tên là BoilerLogHandler và một Event với tên là BoilerEventLog, mà triệu hồi delegate đó khi nó được tạo ra.
4. Ví dụ
Ví dụ 1: Event trong C#
Tạo hai lớp có tên lần lượt là EventTest, TestCsharp như sau:
Lớp EventTest:
using System;
namespace Csharp
{
class EventTest
{
private int value;
public delegate void NumManipulationHandler();
public event NumManipulationHandler ChangeNum;
protected virtual void OnNumChanged()
{
if (ChangeNum != null)
{
ChangeNum();
}
else
{
Console.WriteLine("Kich hoat su kien!");
}
} public EventTest(int n)
{
SetValue(n);
} public void SetValue(int n)
{
if (value != n)
{
value = n;
OnNumChanged();
}
}
}
}
Lớp TestCsharp:
using System;
namespace Csharp
{
class TestCsharp
{
static void Main(string[] args)
{
Console.WriteLine("Vi du minh hoa Su kien (Event) trong C#");
Console.WriteLine("----------------------------------");
//tao doi tuong EventTest
EventTest e = new EventTest(5);
e.SetValue(6);
e.SetValue(9);
Console.ReadKey();
}
}
}
Nếu bạn không sử dụng lệnh Console.ReadKey(); thì chương trình sẽ chạy và kết thúc luôn (nhanh quá đến nỗi bạn không kịp nhìn kết quả). Lệnh này cho phép chúng ta nhìn kết quả một cách rõ ràng hơn.
Biên dịch và chạy chương trình C# trên sẽ cho kết quả sau:
Ví dụ 2: Event trong C#
Ví dụ này cung cấp một ứng dụng đơn giản để xử lý sự cố cho một hệ thống nồi hơn đun nước nóng. Khi kỹ sư bảo dưỡng kiểm tra nồi hơi, nhiệt độ và áp suất nồi hơi được tự động ghi lại vào trong một log file cùng với các ghi chú của kỹ sư bảo dưỡng này.
Tạo 4 lớp có tên lần lượt là Boiler, DelegateBoilerEvent, BoilerInfoLogger, TestCsharp như sau:
Lớp Boiler:
using System;
namespace Csharp
{
class Boiler
{
private int temp;
private int pressure;
public Boiler(int t, int p)
{
temp = t;
pressure = p;
} public int getTemp()
{
return temp;
} public int getPressure()
{
return pressure;
}
}
}
using System;
namespace Csharp
{
class DelegateBoilerEvent
{
public delegate void BoilerLogHandler(string status);
//dinh nghia su kien dua vao delegate tren
public event BoilerLogHandler BoilerEventLog;
public void LogProcess()
{
string remarks = "OK!";
Boiler b = new Boiler(100, 12);
int t = b.getTemp();
int p = b.getPressure();
if(t > 150 || t < 80 || p < 12 || p > 15)
{
remarks = "Can duy tri";
}
OnBoilerEventLog("Thong tin log:\n");
OnBoilerEventLog("Nhiet do: " + t + "\nAp suat: " + p);
OnBoilerEventLog("\nThong bao: " + remarks);
} protected void OnBoilerEventLog(string message)
{
if (BoilerEventLog != null)
{
BoilerEventLog(message);
}
}
}
}
Lớp BoilerInfoLogger:
using System;
using System.IO;
namespace Csharp
{
class BoilerInfoLogger
{
FileStream fs;
StreamWriter sw;
public BoilerInfoLogger(string filename)
{
fs = new FileStream(filename, FileMode.Append, FileAccess.Write);
sw = new StreamWriter(fs);
} public void Logger(string info)
{
sw.WriteLine(info);
} public void Close()
{
sw.Close();
fs.Close();
}
}
}
using System;
namespace Csharp
{
class TestCsharp
{
static void Logger(string info)
{
Console.WriteLine(info);
} static void Main(string[] args)
{
Console.WriteLine("Vi du minh hoa su kien trong C#");
Console.WriteLine("---------------------------------");
BoilerInfoLogger filelog = new BoilerInfoLogger("D:\\boiler.txt");
DelegateBoilerEvent boilerEvent = new DelegateBoilerEvent();
boilerEvent.BoilerEventLog += new
DelegateBoilerEvent.BoilerLogHandler(Logger);
boilerEvent.BoilerEventLog += new
DelegateBoilerEvent.BoilerLogHandler(filelog.Logger);
boilerEvent.LogProcess();
Console.ReadLine();
Console.ReadKey();
filelog.Close();
}
}
}
Biên dịch và chạy chương trình C# trên sẽ cho kết quả sau:
Trên đây là bài viết của eLib.VN về sự kiện (Event) trong C#. Thực chất Event trong C# là một dạng điều chỉnh của degelate cho phù hợp với mô hình sub/pub. Trong đó Event che bớt đi một số tính năng của degelate trong client code.