Delegate trong C# là gì? Nếu chưa biết câu trả lời, mời bạn tham khảo những điều cần biết về Delegate trong C#.
Delegate trong C# là tương tự như con trỏ tới các hàm, trong C hoặc trong C++. Delegate là một biến kiểu tham chiếu chứa tham chiếu tới một phương thức. Tham chiếu đó có thể được thay đổi tại runtime.
Đặc biệt, các delegate được sử dụng để triển khai các sự kiện và các phương thức call-back. Tất cả delegate được dẫn xuất ngầm từ lớp System.Delegate trong C#.
Khai báo Delegate trong C#
Khai báo Delegate trong C# xác định các phương thức có thể được Delegate tham chiếu. Một Delegate có thể tham chiếu tới một phương thức, mà có cùng dấu hiệu như của Delegate đó.
Ví dụ, xét delegate sau:
public delegate int MyDelegate (string s);
Delegate trên có thể được sử dụng để tham chiếu bất kỳ phương thức nào có một tham số string đơn và trả về một biến kiểu int.
Cú pháp để khai báo delegate trong C# là:
delegate <kiểu_trả_về> <tên_delegate> <danh_sách_tham_số>
Khởi tạo Delegate trong C#
Khi kiểu delegate được khai báo, đối tượng delegate phải được tạo với từ khóa new và được liên kết với một phương thức cụ thể. Khi tạo một delegate, tham số được truyền tới biểu thức new được viết tương tự như một lời gọi phương thức, nhưng không có tham số tới phương thức đó. Ví dụ:
public delegate void printString(string s);
...
printString ps1 = new printString(WriteToScreen);
printString ps2 = new printString(WriteToFile);
Ví dụ sau minh họa cách khai báo, khởi tạo và sử dụng delegate để tham chiếu các phương thức, lấy tham số nguyên (integer) và trả về một giá trị integer.
using System;
delegate int NumberChanger(int n);
namespace DelegateAppl {
class TestDelegate {
static int num = 10;
public static int AddNum(int p) {
num += p;
return num;
}
public static int MultNum(int q) {
num *= q;
return num;
}
public static int getNum() {
return num;
}
static void Main(string[] args) {
//tạo thể hiện delegate
NumberChanger nc1 = new NumberChanger(AddNum);
NumberChanger nc2 = new NumberChanger(MultNum);
//gọi phương thức sử dụng đối tượng delegate
nc1(25);
Console.WriteLine("Giá trị của số: {0}", getNum());
nc2(5);
Console.WriteLine("Giá trị của số: {0}", getNum());
Console.ReadKey();
}
}
}
Khi chạy code trên bạn sẽ nhận được kết quả như sau:
Giá trị của số: 35
Giá trị của số: 175
Multicast (đa hướng) một Delegate trong C#
Các đối tượng Delegate có thể được hợp thành từ các delegate khác nhờ toán tử "+". Một delegate được hợp thành gọi hai Delegate mà nó được hợp thành từ đó. Chỉ có các delegate cùng kiểu mới có thể được hợp thành. Toán tử "-" có thể được sử dụng để gỡ bỏ một delegate thành phần khỏi một delegate được hợp thành.
Sử dụng đặc tính này của các delegate, bạn có thể tạo một danh sách triệu hồi của các phương thức mà sẽ được gọi khi delegate đó được triệu hồi. Điều này được gọi là Multicasting của một Delegate. Chương trình ví dụ sau minh họa Multicasting của một Delegate trong C#:
using System;
delegate int NumberChanger(int n);
namespace QTMCSharp {
class Tester {
static int num = 10;
public static int AddNum(int p) {
num += p;
return num;
}
public static int MultNum(int q) {
num *= q;
return num;
}
public static int getNum() {
return num;
}
static void Main(string[] args) {
//tạo các thể hiện delegate
NumberChanger nc;
NumberChanger nc1 = new NumberChanger(AddNum);
NumberChanger nc2 = new NumberChanger(MultNum);
nc = nc1;
nc += nc2;
//gọi multicast
nc(5);
Console.WriteLine("Giá trị của số: {0}", getNum());
Console.ReadKey();
}
}
}
Kết quả khi chạy chương trình trên sẽ như sau:
Giá trị của số: 75
Cách sử dụng Delegate trong C#
Ví dụ dưới đây sẽ minh họa cách sử dụng của delegate trong C#. Delegate với tên printString có thể được sử dụng để tham chiếu phương thức nhận input là một string và không trả về cái gì.
Chúng ta sử dụng delegate này để gọi hai phương thức: phương thức đầu tiên in chuỗi tới Console, và phương thức thứ hai in nó tới một File.
using System;
using System.IO;
namespace QTMCsharp {
class TestCsharp {
static FileStream fs;
static StreamWriter sw;
// khai báo delegate
public delegate void printString(string s);
// phương thức thứ nhất để in trên console
public static void WriteToScreen(string str) {
Console.WriteLine("Chuỗi la: {0}", str);
}
//phương thức thứ hai để ghi dữ liệu vào file
public static void WriteToFile(string s)
{
fs = new FileStream("c:\\message.txt",
FileMode.Append, FileAccess.Write);
sw = new StreamWriter(fs);
sw.WriteLine(s);
sw.Flush();
sw.Close();
fs.Close();
}
// phương thức này nhận delegate làm tham số và
// sử dụng nó để gọi các phương thức nếu cần.
public static void sendString(printString ps)
{
ps("Quantrimang.com");
}
static void Main(string[] args)
{
Console.WriteLine("Ví dụ về Delegate C#:");
Console.WriteLine("--------------------------");
printString ps1 = new printString(WriteToScreen);
printString ps2 = new printString(WriteToFile);
sendString(ps1);
sendString(ps2);
Console.ReadKey();
}
}
}
Biên dịch và chạy chương trình C# trên để xem kết quả.
Ví dụ về Delegate C#:
--------------------------
Chuỗi là: Quantrimang.com
Ưu và nhược điểm khi dùng delegate trong C#
Ưu điểm:
- Giúp gọi các phương thức tĩnh và động.
- Gọi một hoặc nhiều phương thức có cùng đặc điểm
- Xác định phương thức callback và gọi trình xử lý sự kiện.
- Có thể được kết hợp thành delegate đa hướng để chạy một chuỗi delegate theo thứ tự.
Nhược điểm:
- Một delegate sẽ chạy chậm hơn một phương thức thông thường bởi thời gian chạy cần giải quyết tham chiếu phương thức trước khi gọi delegate được xử lý và gọi thành công phương thức.
- Ngoại lệ trong phương thức được tham chiếu bởi một delgate không hiện nguyên nhân gây lỗi. Điều này khiến việc gỡ lỗi trở nên khó khăn hơn.
- Càng sử dụng nhiều delegate, code càng khó đọc.
- Khi dùng delegate, trình biên dịch JIT và runtime có thể tối ưu hóa kém hơn so với các chức năng thông thường.
Nhìn chung, Delegate trong C# là lựa chọn phù hợp để lập trình hướng đối tượng. Chỉ cần nắm được những kiến thức cơ bản kể trên, bạn có thể sử dụng delegate C# dễ dàng.
Bài trước: Indexer trong C#
Bài tiếp: Sự kiện (Event) trong C#