YOMEDIA
ADSENSE
Phần 1: Delegate
98
lượt xem 18
download
lượt xem 18
download
Download
Vui lòng tải xuống để xem tài liệu đầy đủ
Một delegate là kiểu tham chiếu tới một phương thức( hàm, mình sẽ gọi là hàm trong suốt bài này luôn nghen ). Một khi delegate được gán tham chiếu tới một hàm, nó sẽ giống hệt hàm đó. Hàm delegate được dùng giống hệt các hàm bình thường khác, nghĩa là cũng có kiểu trả về, có tham số truyền vào.
AMBIENT/
Chủ đề:
Bình luận(0) Đăng nhập để gửi bình luận!
Nội dung Text: Phần 1: Delegate
- Phần 1: Delegate Một delegate là kiểu tham chiếu tới một phương thức( hàm, mình sẽ gọi là hàm trong suốt bài này luôn nghen ). Một khi delegate được gán tham chiếu tới một hàm, nó sẽ giống hệt hàm đó. Hàm delegate được dùng giống hệt các hàm bình thường khác, nghĩa là cũng có kiểu trả về, có tham số truyền vào. Ví dụ: public delegate int PerformCalculation(int x, int y); Ở ví dụ trên, tên của delegate sẽ là "PerformCalculation". Các bạn nên chú ý: khai báo delegate thường rất dễ bị nhầm lẫn với khai báo hàm. Tất cả các hàm có đặc tả hàm giống với đặc tả của delegate trên (các tham số truyền vào có kiểu giống nhau, kết quả trả về giống nhau) đều có thể dùng để delegate trên tham chiếu tới. Do đó chỉ cần biết đặc tả của delegate là chúng ta có thể tạo hàm của riêng mình để delegate tham chiếu tới. Ví dụ về các hàm mà delegate trên có thể tham chiếu tới: public int AddMethod(int x, int y); public int SubtractMethod(int x, int y); public int PowerMethod(int x, int y); Bây giờ việc cuối cùng là chúng ta tạo ra đối tượng delegate cụ thể (instance của đặc tả delegate ở trên) và sau đó cho nó tham chiếu tới một hàm bất kì (cùng đặc tả). Ví dụ: PerformCalculation addDelegate = new PerformCalculation(AddMethod); Bây giờ thì thử viết chương trình cộng hai số dùng delegate trên: public delegate int PerformCalculation(int x, int y); class TestDelegate { public int AddMethod(int m, int n) { System.Console.WriteLine("Thuc hien phep cong trong ham AddMethod"); System.Console.WriteLine("{0} + {1} = {2} ", m, n, m + n); return m + n; } [STAThread] static void Main() { TestDelegate phepToan = new TestDelegate(); int x, y; string strInput = string.Empty; do { try { Console.Write("Nhap vao so thu nhat: "); strInput = Console.ReadLine(); x = System.Convert.ToInt16(strInput); Console.Write("Nhap vao so thu hai: "); strInput = Console.ReadLine(); y = System.Convert.ToInt16(strInput); break; } catch(FormatException) { Console.WriteLine("Vui long nhap lai"); } }while(true); PerformCalculation add = new PerformCalculation(phepToan.AddMethod); int tong = add(x,y); System.Console.WriteLine("Gia tri cua tong: " + tong); } }
- Phần 2: Events và Delegates Event, dịch ra nghĩa tiếng Việt là sự kiện. Và đúng thiệt, mình thấy trong .Net nó dùng event theo đúng nghĩa sự kiện thiệt. Bây giờ hãy bỏ ngôn ngữ lập trình sang một bên và xem xét event trong cuộc sống đời thường của chúng ta. Ví dụ về một sự kiện: "xe quảng cáo show diễn rao: Tối nay ca sĩ Lươn Bàng Quan sẽ biểu diễn tại sân đình" chẳng hạn. Trong sự kiện trên chúng ta thấy xe quảng cáo chính là đối tượng phát sinh ra sự kiện rao show diễn, còn tên ca sĩ và địa điểm biểu diễn là thông tin đi kèm với sự kiện trên, ta cũng có thể thấy vì xe quảng cáo có loa phóng thanh rất to nên ko phải một mình ta nghe mà còn nhiều người khác nghe nữa, tức là cùng lúc có thể có nhiều đối tượng nhận sự kiện. Trong nhiều đối tượng nhận được sự kiện trên, mỗi đối tượng sẽ có những cách xử lý sự kiện trên khác nhau: + Một số người là fan hâm mộ thì sẽ sắp xếp đi xem. + Một số không quan tâm đến các show diễn thì sẽ không làm gì cả, + ...v...v. event trong .Net cũng giống y chang không khác gì sự kiện trên. Một event sẽ có các thông tin: + Đối tượng phát ra event (được khai báo kiểu Object) + Các thông tin đi kèm với event (được gọi là EventArgs) + Các hàm của các đối tượng đăng kí nhận event sẽ có cách xử lí event khác nhau. Với các đặc tính trên, events sẽ dùng delegate để kích họat (trigger) hàm xử lí của các hàm xử lí đã đăng kí nhận events. Bây giờ ta sẽ dùng ví dụ cũ để làm ví dụ: // Declare a delegate delegate void PerformCalculation(object sender, PerformCalculationEvenArgs e); class PerformCalculationEvenArgs : System.EventArgs // lớp chứa thông tin đi kèm với event, nhất thiết phải kế thừa từ lớp System.EventArgs { int x, y; public PerformCalculationEvenArgs(int _x, int _y) { x = _x; y = _y; } public int X { get {return this.x;} } public int Y { get {return this.y;} } } class myTestdelegate { public static void AddMethod(object sender, PerformCalculationEvenArgs e) { System.Console.WriteLine("Thuc hien phep cong trong ham AddMethod"); System.Console.WriteLine("{0} + {1} = {2} ", e.X, e.Y, e.X + e.Y); } public static void SubtractMethod(object sender, PerformCalculationEvenArgs e) { System.Console.WriteLine("Thuc hien phep tru trong ham SubtractMethod"); System.Console.WriteLine("{0} - {1} = {2} ", e.X, e.Y, e.X - e.Y); } public static void PowerMethod(object sender, PerformCalculationEvenArgs e) { System.Console.WriteLine("Thuc hien phep luy thua x^y trong ham PowerMethod"); System.Console.WriteLine("{0} ^ {1} = {2} ", e.X, e.Y, e.X ^ e.Y); } } class TestDelegate { public event PerformCalculation Calculations = null;// khai báo một event cụ thể [STAThread] static void Main() {
- TestDelegate phepToan = new TestDelegate(); phepToan.Calculations += new PerformCalculation(myTestdelegate.AddMethod); // đăng kí hàm xử lí event phepToan.Calculations += new PerformCalculation(myTestdelegate.SubtractMethod); // đăng kí hàm xử lí event phepToan.Calculations += new PerformCalculation(myTestdelegate.PowerMethod);// đăng kí hàm xử lí event int x, y; string strInput = string.Empty; do { try { Console.Write("Nhap vao so thu nhat: "); strInput = Console.ReadLine(); x = System.Convert.ToInt16(strInput); Console.Write("Nhap vao so thu hai: "); strInput = Console.ReadLine(); y = System.Convert.ToInt16(strInput); break; } catch(FormatException) { Console.WriteLine("Vui long nhap lai"); } }while(true); Console.WriteLine("Raise event"); if (phepToan.Calculations != null)// kiem tra xem event co ham nao dang ki xu li khong, nhất thiết phải kiểm tra trước khi raise event phepToan.Calculations(phepToan, new PerformCalculationEvenArgs(x,y));// raise event } } Delegate có thể được xem như là kiểu đối tượng mới trong C#, mà có môt số điểm quen thu ộc v ới l ớp. Chúng tồn tại trong tình huống mà ta muốn truyền phương thức xung quanh nh ững ph ương th ức khác, đ ể minh hoạ ta xem dòng mã sau: int i = int.Parse(“99″); - Chúng ta quen với việc truyền dữ liệu đến một phương thức như là thông số, vì vậy ý tường truyền phương thức như là thông s ố nghe có vẻ h ơi lạ đ ối với chúng ta. Tuy nhiên có trường hợp mà ta có 1 phương thức mà làm 1 điều gì đó, nhiều hơn là xử lí dữ liệu, phương thức đó có thể cần làm điều gì đó mà liên quan đến việc thực thi phương thức khác phức t ạp hơn, bạn không biết vào lúc nào thì phương thức thứ hai sẽ được biên dịch. Thông tin đó ch ỉ bi ết vào lúc ch ạy, và chính vì lí do đó mà phương thức 2 sẽ cần truyền vào như là thông s ố cho phương th ức đầu tiên. Điều này nghe có vẻ hơi khó hiểu, nhưng nó sẽ được làm rõ hơn trong 1 vài ví dụ sau: Luồng bắt đầu: C# có thể bảo máy tính bắt đầu một chuỗi thực thi m ới song song v ới vi ệc th ực thi đương thời. 1 chuỗi liên tiếp này gọi là luồng,và việc bắt đầu này đ ược làm b ằng cách dùng ph ương thức, Start() trên 1 thể hiện của lớp cơ sởSystem.Threading.Thread. Khi ch ương trình b ắt đ ầu ch ạy,n ơi nó b ắt đầu làmain(). Tương tự như vậy khi bạn muốn máy tính chạy một chuỗi th ực thi thì b ạn ph ải báo cho máy tính biết bắt đầu chạy là ở đâu. Bạn phải cung cấp cho nó chi tiết của ph ương th ức mà vi ệc thực thi có th ể b ắt đ ầu
- – nói cách khác, phương thức Thread.Start()phải lấy 1 thông số mà định nghĩa phương th ức đ ược thi hành b ởi luồng. Lớp thư viện chung . khi 1 nhiệm vụ chứa đựng nhiệm vụ con mà mã của các nhi ệm vụ con này được viết trong các thư viện chỉ có sử dụng thư viện mới biết nó làm gì. Ví d ụ, chúng ta mu ốn vi ết m ột l ớp chứa một mảng đối tuợng và sắp nó tăng dần. 1 phần công vi ệc đ ược l ặp l ại là l ấy 2 đ ối t ượng trong l ớp so sánh với nhau để xem đối tượng nào đứng truớc.nếu ta muốn lớp có khả năng s ắp xếp b ất kì đ ối t ượng nào, không có cách nào có thể làm được việc so sánh trên. Mã client dùng m ảng đ ối t ượng c ủa ta s ẽ b ảo cho ta biết cách so sánh cụ thể đối tượng mà nó muốn sắp xếp. Nói cách khác, mã client s ẽ ph ải truy ền cho l ớp c ủa ta phương thức thích hợp mà có thể được gọi, để làm vi ệc so sánh. Nguyên tắc chung là: mã của ta sẽ cần thông báo cho thời gian ch ạy .NET bi ết ph ương th ức nào x ử lí tình huống nào. - Vì thế chúng ta phải thiết lập những nguyên tắc mà đôi lúc, nh ững ph ương thức c ần l ấy chi ti ết c ủa ph ương thức khác như là thông số. Kế tiếp chúng ta sẽ minh họa cách làm đi ều đó, cách đ ơn gi ản nh ất là truy ền tên của phương thức như là thông số. Giả sử chúng ta muốn bắt đầu m ột luồng m ới, và chúng ta có ph ương th ức được gọi là entrypoint(), mà ta muốn luồng bắt đầu chạy từ đó: void EntryPoint() { // làm những gì luồng mới cần làm } - Có thể chúng ta bắt đầu luồng mới với một đoạn mã: Thread NewThread = new Thread(); Thread.Start(EntryPoint); // sai - Thật sự đây là cách đơn giản nhất. Trong một vài ngôn ngữ dùng cách này nh ư C và C++ (trong c và c++ thông số entrypoint là con trỏ hàm). Không may, cách thực thi trực tiếp này gây ra m ột s ố v ấn đ ề v ề an toàn kiểu. Nhớ rằng ta đang lập trình hướng đối tượng, phương thức hi ếm khi nào t ồn t ại đ ộc lập, mà th ường là phải kết hợp với phương thức khác trưóc khi được gọi. Vì vậy .NET không cho làm đi ều này, thay vào đó n ếu ta muốn truyền phương thức ta phải gói chi tiết của phương th ức trong m ột loại đ ối t ượng m ới là 1 delegate. delegate đơn giản là một kiểu đối tượng đặc biệt – đặc biệt ở chỗ, trong khi tất cả đ ối t ượng chúng ta đ ịnh nghĩa trước đây chứa đựng dữ liệu, thì delegate chứa đựng chi tiết của ph ương th ức. 1. Dùng delegate trong C# - Đầu tiên ta phải định nghĩa delegate mà ta muốn dùng ,nghĩa là b ảo cho trình biên d ịch bi ết lo ại ph ương th ức mà delegate sẽ trình bày.sau đó ta tạo ra các thể hiện của delegate. Cú pháp: delegate void VoidOperation(uint x); - Ta chỉ định mỗi thể hiện của delegate có thể giữ một tham chiếu đến 1 ph ương th ức mà ch ứa m ột thông s ố uint và trả vầ kiểu void. - Ví dụ khác: nếu bạn muốn định nghĩa 1 delegate gọi là twolongsOp mà trình bày 1 hàm có 2 thông số kiểu long và trả về kiểu double. ta có thể viết : delegate double TwoLongsOp(long first, long second); - Hay 1 delegate trình bày phương thức không nhận thông s ố và trả v ề kiểu string.
- delegate string GetAString(); - Cú pháp cũng giống như phương thức, ngoại trừ việc không có ph ần thân c ủa ph ương th ức, và b ắt đ ầu v ới delegate, ta cũng có thể áp dụng các cách thức truy nhập thông th ường trên m ột đ ịnh nghĩa delegate – public, private, protected … public delegate string GetAString(); - Mỗi lần ta định nghĩa một delegate chúng ta có thể t ạo ra m ột thể hi ện c ủa nó mà ta có th ể dùng đ ề l ưu tr ữ các chi tiết của 1 phưong thức cụ thể. - Lưu ý: với lớp ta có 2 thuật ngữ riêng biệt: lớp để chỉ định nghĩa chung, đ ối t ượng đ ể ch ỉ m ột th ể hi ện c ủa 1 lớp, tuy nhiên đối với delegate ta chỉ có một thuật ngữ là ’1 delegate’ khi t ạo ra m ột th ể hi ện c ủa delegate ta cũng gọi nó là delegate. Vì vậy cần xem xét ngữ cảnh để phân bi ệt. - Đoạn mã sau minh hoạ cho 1 delegate: static void Main(string[] args) { int x = 40; GetAString firstStringMethod = new GetAString(x.ToString); Console.WriteLine("String is" + firstStringMethod()); // With firstStringMethod initialized to x.ToString(), // the above statement is equivalent to saying // Console.WriteLine("String is" + x.ToString()); } - Trong mã này, ta tạo ra delegate GetAString, và khởi tạo nó để nó tham khảo đến phương thức ToString() của một biến nguyên x .chúng ta sẽ biên dịch lỗi nếu cố gắng khởi tạo FirstStringMethod với bất kì phương thức nào có thông số vào và kiểu trả về là chuỗi. - 1 đặc tính của delegate là an kiểu an toàn ( type-safe) để thấy rằng chúng ph ải đ ảm b ảo d ấu ấn ( signature) của phương thức được gọi là đúng. Tuy nhiên 1 điều thú vị là, chúng không quan tâm ki ểu c ủa đ ối t ượng phương thức là gì khi gọi hoặc thậm chí liệu rằng phương thức đó là static hay là một phương thức thể hiện. - Để thấy điều này ta mở rộng đoạn mã trên, dùng delegate FirstStringMethod để gọi các phương thức khác trên những đối tượng khác – 1 phương thức thể hiện và 1 phương thức tĩnh. Ta cũng dùng l ại c ấu trúc currency, và cấu trúc currency đã có overloadriêng của nó cho phương thức ToString().để xem xét delegate với phương thức tĩnh ta thêm 1 phương thức tĩnh với cùng d ấu ấn nh ư currency: struct Currency { public static string GetCurrencyUnit() { return "Dollar"; } } - Bây giờ ta sử dụng thể hiện GetAString như sau: private delegate string GetAString(); static void Main(string[] args) { int x = 40;
- GetAString firstStringMethod = new GetAString(x.ToString); Console.WriteLine("String is " + firstStringMethod()); Currency balance = new Currency(34, 50); firstStringMethod = new GetAString(balance.ToString); Console.WriteLine("String is " + firstStringMethod()); firstStringMethod = new GetAString(Currency.GetCurrencyUnit); Console.WriteLine("String is " + firstStringMethod()); } - Đoạn mã này chỉ cho ta biết làm thế nào để gọi 1 phương th ức qua trung gian là delegate, đăng kí l ại delegate để tham chiếu đến một phương thức khác trên 1 thể hiện khác của lớp. - Tuy nhiên ta vẫn chưa nắm rõ được quy trình truyền 1 delegate đ ến 1 ph ương th ức khác, cũng nh ư ch ưa thấy được lợi ích của delegate qua ví dụ trên. Như ta có th ể g ọi trực tiếpToString() từ int hay currency mà không cần delegate. Ta cần những ví dụ phức tạp hơn để hiểu rõ delegate. Ta s ẽ trình bày 2 ví d ụ: ví d ụ 1 đ ơn giản sử dụng delegate để gọi vào thao tác khác. Nó chỉ rõ làm thế nào đ ể truy ền delegate đ ến ph ương th ức và cách sử dụng mảng trong delegate. Ví dụ 2 phức t ạp hơn là l ớp BubbleSorter, thực thi 1 phương thức sắp xếp mảng đối tượng tăng dần. lớp này sẽ rất khó viết nếu không có delegate. a. Ví dụ SimpleDelegate - Trong ví dụ này ta sẽ tạo lớp MathOperations mà có vài phương thức static để thực thi 2 thao tác trên kiểu double, sau đó ta dùng delegate để gọi những phương thức này. L ớp nh ư sau: class MathsOperations { public static double MultiplyByTwo(double value) { return value * 2; } public static double Square(double value) { return value * value; } } - Sau đó ta gọi phương thức này như sau: using System; namespace SimpleDelegate { delegate double DoubleOp(double x); class MainEntryPoint { static void Main() { DoubleOp[] operations =
- { new DoubleOp(MathsOperations.MultiplyByTwo), new DoubleOp(MathsOperations.Square) }; for (int i = 0; i < operations.Length; i++) { Console.WriteLine("Using operations[{0}]:", i); ProcessAndDisplayNumber(operations[i], 2.0); ProcessAndDisplayNumber(operations[i], 7.94); ProcessAndDisplayNumber(operations[i], 1.414); Console.WriteLine(); } } static void ProcessAndDisplayNumber(DoubleOp action, double value) { double result = action(value); Console.WriteLine( "Value is {0}, result of operation is {1}", value, result); } } } - Trong đoạn mã này ta khởi tạo 1 mảng delegate doubleOp. Mỗi phần tử của mảng được khởi động để tham chiếu đến 1 thao tác khác được thực thi bởi lớp MathOperations. Sau đó, nó lặp xuyên suốt mảng,ứng dụng mỗi thao tác đến 3 kiểu giá trị khác nhau. Điều này minh h ọa cách s ử d ụng delegate – là có th ể nhóm nh ững phương thức lại với nhau thành mảng để sử dụng, để ta có thể gọi một vài ph ương thức trong vòng l ặp. - Chỗ quan trọng trong đoạn mã là chỗ ta truyền 1 delegate vào ph ương th ứcProcessAndDisplayNumber(), ví dụ : ProcessAndDisplayNumber(operations[i], 2.0); - Ở đây ta truyền tên của delegate, nhưng không có thông s ố nào.cho rằng operation[i]là 1 delegate : operation[i] nghĩa là ‘delegate’,nói cách khác là phương thức đại diện cho delegate. operation[i](2.0) nghĩa là ‘ gọi thực sự phương thức này, truyền giá trị vào trong ngo ặc’. Phương thức ProcessAndDisplayNumber() được định nghĩa để lấy 1 delegate như là thông số đầu tiên của nó : static void ProcessAndDisplayNumber(DoubleOp action, double value) Sau đó khi ở trong phương thức này , ta gọi: double result = action(value); Thể hiện delegate action được gọi và kết quả trả về được l ưu trữ trong result chạy ví dụ ta có: SimpleDelegate Using operations[0]:
- Value is 2, result of operation is 4 Value is 7.94, result of operation is 15.88 Value is 1.414, result of operation is 2.828 Using operations[1]: Value is 2, result of operation is 4 Value is 7.94, result of operation is 63.0436 Value is 1.414, result of operation is 1.999396 b. Ví dụ BubleSorter - Sau đây ta sẽ xem 1 ví dụ cho thấy sự hữu ích của delegate. Ta s ẽ t ạo l ớp bublesorter. Lớp này thực thi 1 phương thức tĩnh, Sort() lấy thông số đầu là 1 mảng đối tượng, và sắp xếp lại chúng tăng d ần. Ví d ụ đ ể s ắp xếp 1 mảng số nguyên bằng thuật toán Bubble sort : ///đây không phải là 1 phần của ví dụ for (int i = 0; i < sortArray.Length; i++) { for (int j = i + 1; j < sortArray.Length; j++) { if (sortArray[j] < sortArray[i]) // problem with this test { int temp = sortArray[i]; // swap ith and jth entries sortArray[i] = sortArray[j]; sortArray[j] = temp; } } } - Thuật toán này tốt cho số nguyên, nhưng ta muốn phương thức sort() sắp xếp cho mọi đối tượng,ta thấy vấn đề nằm ở dòng if(sortArray[j] < sortArray[i]) trong đoạn mã trên.bởi ta muốn so sánh 2 đối t ượng trên m ảng mà cái nào là lớn hơn. Chúng ta có thể sắp xếp kiểu int, nhưng làm thế nào để sắp xếp những lớp chưa biết hoặc không xác định cho đến lúc chạy.câu trả lời là mã client, mà biết về lớp muốn sắp xếp, phải truyền 1 delegate gói trong một phương thức sẽ làm công việc so sánh. - Định nghĩa delegate như sau: delegate bool CompareOp(object lhs, object rhs); - Và xây dựng phương thức sort() là : static public void Sort(object [] sortArray, CompareOp gtMethod) - Phần hướng dẫn cho phương thức này sẽ nói rõ rằng gtmethod phải tham chiếu đến 1 phương thức static có 2 đối số, và trả về true nếu giá trị của đối số thứ 2 là ‘lớn hơn’ (nghĩa là năm sau trong m ảng) đ ối s ố th ứ nh ất. Mặc dù ta có thể sử dụng delegate ở đây, nhưng cũng có thể gi ải quy ết v ấn đề b ằng cách s ử d ụng interface. .NET xây dựng 1 interfaceIComparer cho mục đích này. Tuy nhiên, ta sử dụng delegate vì loại vấn đề này thì thường có khuynh hướng dùng delegate. - Sau đây là lớp bublesorter : class BubbleSorter { static public void Sort(object[] sortArray, CompareOp gtMethod) {
- for (int i = 0; i < sortArray.Length; i++) { for (int j = i + 1; j < sortArray.Length; j++) { if (gtMethod(sortArray[j], sortArray[i])) { object temp = sortArray[i]; sortArray[i] = sortArray[j]; sortArray[j] = temp; } } } } } - Để dùng lớp này ta cần định nghĩa 1 số lớp khác mà có thể dùng thi ết l ập m ảng c ần s ắp x ếp. Ví d ụ công ty điện thoại có danh sách tên khách hàng, và muốn s ắp danh sách theo l ương.m ỗi nhân viên trình bày b ởi th ể hiện của một lớp , Employee: class Employee { private string name; private decimal salary; public Employee(string name, decimal salary) { this.name = name; this.salary = salary; } public override string ToString() { return string.Format(name + ", {0:C}", salary); } public static bool RhsIsGreater(object lhs, object rhs) { Employee empLhs = (Employee)lhs; Employee empRhs = (Employee)rhs; return (empRhs.salary > empLhs.salary) ? true : false; } } - Lưu ý để phù hợp với dấu ấn của delegate CompareOp, chúng ta phải định nghĩaRhsIsGreater trong lớp này lấy 2 đối tượng để tham khảo,hơn là tham khảo employeenhư là thông số.điều này có nghĩa là ta phải ép ki ểu những thông số vào trong tham khảoemployee để thực thi việc so sánh. - Bây giờ ta viết mã yêu cầu sắp xếp : namespace Wrox.ProCSharp.AdvancedCSharp
- { delegate bool CompareOp(object lhs, object rhs); class MainEntryPoint { static void Main() { Employee[] employees = { new Employee("Karli Watson", 20000), new Employee("Bill Gates", 10000), new Employee("Simon Robinson", 25000), new Employee("Mortimer", (decimal)1000000.38), new Employee("Arabel Jones", 23000), new Employee("Avon from 'Blake's 7'", 50000)}; CompareOp employeeCompareOp = new CompareOp(Employee.RhsIsGreater); BubbleSorter.Sort(employees, employeeCompareOp); for (int i = 0; i < employees.Length; i++) Console.WriteLine(employees[i].ToString()); } } } - Chạy mã này sẽ thấy employees được sắp xếp theo lương . BubbleSorter Bill Gates, £10,000.00 Karli Watson, £20,000.00 Arabel Jones, £23,000.00 Simon Robinson, £25,000.00 Avon from 'Blake's 7', £50,000.00 Mortimer, £1,000,000.38 2. Multicast delegate - Đến lúc này mỗi delegate mà chúng ta sử dụng chỉ gói ghém trong 1 ph ương th ức đ ơn g ọi. G ọi delegate nào thì dùng phương thức đó. Nếu ta muốn gọi nhiều hơn 1 phương th ức, ta cần t ạo m ột l ời gọi t ường minh xuyên suốt delegate nhiều hơn một lần. Tuy nhiên, 1 delegate có thể gói ghém nhi ều h ơn 1 ph ương th ức. 1 delegate như vậy gọi là multicast delegate. nếu 1 multicast delegate được gọi, nó sẽ gọi liên tiếp những phương thức theo thứ tự. Để làm điều này, delegate phải trả về là void. Nếu ta dùng một delegate có kiểu trả về làvoid, trình biên dịch sẽ coi như đây là một multicast delegate. Xem ví dụ sau, dù cú pháp giống như trước đây nh ưng nó thực sự là một multicast delegate, operations, class MainEntryPoint
- { static void Main() { DoubleOp operations = new DoubleOp(MathOperations.MultiplyByTwo); operations += new DoubleOp(MathOperations.Square); } } - Trong ví dụ trên muốn tham khảo đến 2 phương thức ta dùng mảng delegate. Ở đây, đ ơn gi ản ta ch ỉ thêm 2 thao tác này vào trong cùng một multicast delegate.multiccast delegate nhận toán tử + và +=. Nếu ta muốn, ta có thể mở rộng 2 dòng mã trên, có cùng cách tác động: DoubleOp operation1 = new DoubleOp(MathOperations.MultiplyByTwo); DoubleOp operation2 = new DoubleOp(MathOperations.Square); DoubleOp operations = operation1 + operation2; - multicast delegate cũng biết toán tử – và -= để bỏ đi phương thức được gọi từ delegate. - một muticast delegate là một lớp được dẫn xuất từSystem.MulticastDelegate mà lại được dẫn xuất từ System.Delegate. System.MulticastDelegate có thêm những thành phần để cho phép nối những phương thức gọi cùng với nhau vào một danh sách. - Minh hoạ cho sử dụng multicast delegate ta sử dụng lại ví d ụ simpleDelegate biến nó thành một ví dụ mới MulticastDelegate. bởi vì ta cần delegate trả về kiểu void, ta phải viết lại những phương thức trong lớp Mathoperations, chúng sẽ trình bày kết quả thay vì trả v ề: class MathOperations { public static void MultiplyByTwo(double value) { double result = value * 2; Console.WriteLine( "Multiplying by 2: {0} gives {1}", value, result); } public static void Square(double value) { double result = value * value; Console.WriteLine("Squaring: {0} gives {1}", value, result); } } - Để dàn xếp sự thay đổi này, ta viết lại ProcessAndDisplayNumber: static void ProcessAndDisplayNumber(DoubleOp action, double value) { Console.WriteLine("\nProcessAndDisplayNumber called with value = " + value); action(value); } - Bây giờ thử multicast delegate ta vừa tạo : static void Main() { DoubleOp operations = new DoubleOp(MathOperations.MultiplyByTwo);
- operations += new DoubleOp(MathOperations.Square); ProcessAndDisplayNumber(operations, 2.0); ProcessAndDisplayNumber(operations, 7.94); ProcessAndDisplayNumber(operations, 1.414); Console.WriteLine(); } - Bây giờ mỗi lần ProcessAndDisplayNumber được gọi, nó sẽ trình bày 1 thông điệp để báo rằng nó được gọi câu lệnh sau: action(value); - Sẽ làm cho mỗi phương thức gọi trong thể hiện delegate action được gọi liên tiếp nhau. - Kết quả : MulticastDelegate ProcessAndDisplayNumber called with value = 2 Multiplying by 2: 2 gives 4 Squaring: 2 gives 4 ProcessAndDisplayNumber called with value = 7.94 Multiplying by 2: 7.94 gives 15.88 Squaring: 7.94 gives 63.0436 ProcessAndDisplayNumber called with value = 1.414 Multiplying by 2: 1.414 gives 2.828 Squaring: 1.414 gives 1.999396 - Nếu dùng multicast delegate , ta nên nhận thức đến thứ tự phương thức được nối với nhau trong cùng m ột delegate sẽ được gọi là không xác định. Do đó ta nên tránh vi ết mã mà nh ững ph ương th ức đ ược g ọi liên h ệ với nhau theo một thứ tự cụ thể.
Thêm tài liệu vào bộ sưu tập có sẵn:
Báo xấu
LAVA
AANETWORK
TRỢ GIÚP
HỖ TRỢ KHÁCH HÀNG
Chịu trách nhiệm nội dung:
Nguyễn Công Hà - Giám đốc Công ty TNHH TÀI LIỆU TRỰC TUYẾN VI NA
LIÊN HỆ
Địa chỉ: P402, 54A Nơ Trang Long, Phường 14, Q.Bình Thạnh, TP.HCM
Hotline: 093 303 0098
Email: support@tailieu.vn