鼎鼎知识库
Nelze vybrat více než 25 témat Téma musí začínat písmenem nebo číslem, může obsahovat pomlčky („-“) a může být dlouhé až 35 znaků.

24.委托.md 6.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. > 委托
  2. - 当要把方法传递给其他方法时,就需要使用委托。
  3. > 声明委托
  4. - 使用委托时
  5. - 首先必须定义要使用的委托,对于委托,定义它就是告诉编译器这种类型的委托表示哪种类型的方法。
  6. - 然后,必须创建该委托的一个或多个实例。编译器在后台将创建表示该委托的一个类。
  7. - 声明委托的语法如下:
  8. ```
  9. delegate void IntMethodInvoker(int x);
  10. 声明了一个委托IntMethodInvoker,并指定该委托的每个实例都可以包含一个方法的引用,该方法带有一个int参数,并返回void。
  11. 委托的类型安全性非常高。在定义委托时,必须给出它所表示的方法的签名和返回类型等全部细节。
  12. ```
  13. - 理解委托的一种好方式是把委托视为给方法的签名和返回类型指定名称。
  14. ```
  15. 定义一个委托TwoLongsOp,该委托表示的方法有两个long型参数,返回类型为double。
  16. delegate double TwoLongsOp(long first,long second);
  17. 或者定义一个委托,它表示的方法不带参数,返回一个string型的值
  18. delegate string GetAStringent();
  19. ```
  20. - 实际上,“定义一个委托”是指“定义一个新类”,委托实现为派生自基类System.MulticastDelegate的类,System.MulticastDelegate又派生自基类System.Delegate。
  21. > 使用委托
  22. - 在int值上调用ToString()方法的一种相当复杂的方式
  23. ```
  24. private delegate string GetAString();
  25. public static void Main()
  26. {
  27. int x=40;
  28. GetAString firstStringMethod=newGetAString(x.ToString);
  29. WriteLine($"String is (fistStringMethod{})");
  30. }
  31. ```
  32. > 简单的委托示例
  33. - 定义一个类MathOperations,它有两个静态方法,对double类型的值执行两种操作。然后使用该委托调用这些方法。
  34. ```
  35. class MathOperation
  36. {
  37. public static double MultiplyByTwo(double value)=>value*2;
  38. public static double Square(double value)=>value*value;
  39. }
  40. 调用这些方法
  41. using static System.Console;
  42. namespace Wrox.ProCSharp.Delegates
  43. {
  44. delegate double DoubleOp(double x);
  45. class Program
  46. {
  47. static void Main()
  48. {
  49. DoubleOp[] operations=
  50. {
  51. MathOperation.MultiplyByTwo,
  52. MathOperations.Square
  53. };
  54. for(int i=0;i<operations.Length;i++)
  55. {
  56. WriteLine($"Using operations[{i}]:");
  57. ProcessAndDisplayNumber(operations[i],2.0);
  58. ProcessAndDisplayNumber(operations[i],7.94);
  59. ProcessAndDisplayNumber(operations[i],1.414);
  60. WriteLine();
  61. }
  62. }
  63. static void ProcessAndDisplayNumber(DoubleOp action,double value)
  64. {
  65. double result =action(value);
  66. WriteLine($"Value is {value},result of operation is {result}");
  67. }
  68. }
  69. }
  70. ```
  71. > Action<T>和Func<T>委托
  72. - 泛型Action<T>委托表示引用一个void返回类型的方法。这个委托类存在不同的变体,可以传递至多16种不同的参数类型。没有泛型参数的Action类可调用没有参数的方法。Action<in T>调用带一个参数的方法,Action<in T1,in T2>调用带两个参数的方法,Action<in T1,in T2,in T3,in T4,in T5,in T6,in T7,in T8>调用8个参数的方法。
  73. - Func<T>允许调用带返回类型的方法。与Action<T>类似,Func<T>也定义了不同的变体,至多也可以传递16个参数类型和一个返回类型。Func<out TResult>委托类型可以调用带返回类型且无参数的方法,Func<in T,out TResult>调用带一个参数的方法,func<in T1,in T2,in T3,in T4,out TResult>调用4个参数的方法。
  74. - delegate double DoubleOp(double x);
  75. - 除了声明自定义委托DoubleOp之外,还可以使用Func<int T,out TResult>委托。可以声明一个该委托类型的变量,或者声明该委托类型的数组
  76. ```
  77. <double ,double>[] operations=
  78. {
  79. MathOperations.MultiplyByTwo,
  80. MathOperations.Square
  81. };
  82. 使用委托,并将ProcessAndDisplayNumber()方法作为参数:
  83. static void ProcessAndDisplayNumber(Func<double,double>action,double value)
  84. {
  85. double result=action(value);
  86. WriteLine{$"Value is{value},result of operation is {result}"}
  87. }
  88. ```
  89. **BubbleSorter示例**
  90. - 编写一个类BubbleSorter,它实现一个静态方法Sort(),这个方法的第一个参数是一个对象数组,把该数组按照升序重新排列。
  91. - 进行冒泡排序的方法如下:
  92. ```
  93. bool swapped=true;
  94. do
  95. {
  96. swapped=false;
  97. for(int i=0;i<sortArray.Length-1;i++)
  98. {
  99. if(sortArray[i]>sortArray[i+1])
  100. {
  101. int temp=sortArray[i];
  102. sortArray[i]=sortArray[i+1];
  103. sortArry[i+1]=temp;
  104. swapped=true;
  105. }
  106. }
  107. }while(swapped);
  108. ```
  109. - 定义BubbleSorter类
  110. ```
  111. class BubbleSorter
  112. {
  113. static public void Sort<T>(IList<T> sortArray,Func<T,T,bool>comparison)
  114. {
  115. bool swapped=true;
  116. do
  117. {
  118. swapped=false;
  119. for (int i=0;i<sortArray.Count-1;i++)
  120. {
  121. if(comparison(sortArray[i+1],sortArry[i]))
  122. {
  123. T temp=sortArray[i];
  124. sortArray[i]=sortArray[i+1];
  125. sortArray[i+1]=temp;
  126. swapped=true;
  127. }
  128. }
  129. }while(swapped);
  130. }
  131. }
  132. ```
  133. - 为了使用这个类,需要定义另一个类,从而建立要排序的数组。假定Mortimer Phones移动电话公司有一个员工列表,要根据他们的薪水进行排序。每个员工分别由类Employee的一个实例表示,
  134. ```
  135. class Employee
  136. {
  137. public Employee(string name,decimal salary)
  138. {
  139. Name=name;
  140. Salary=salary;
  141. }
  142. public string Name{get;}
  143. public decimal Salary{get;private set;}
  144. public override string ToString()=>$"{Name},{Salary:C}";
  145. public static bool CompareSalary(Employee e1,Employee e2)=>e1.Salary<e2.Salary;
  146. }
  147. ```
  148. - 为了匹配Func<T,T,bool>委托的签名,在这个类中必须定义CompareSalay,它的参数是两个Employee引用,并返回一个布尔值。
  149. ```
  150. using static System.Console;
  151. namespace Wrox.ProCSharp.Delegates
  152. {
  153. class Program
  154. {
  155. static void Main()
  156. {
  157. Employee[] employees=
  158. {
  159. new Employee("Bugs Bunny",20000),
  160. new Employee("Elmer Fudd",10000),
  161. new Employee("Daffy Duck",25000),
  162. new Employee("Wile Coyote",1000000.38m),
  163. new Employee("Foghorn Leghorn",23000),
  164. new Employee("RoadRunner",50000)
  165. };
  166. BubbleSorter.Sort(employee,Employee.CompareSalary);
  167. foreach(var employee in employees)
  168. {
  169. WriteLine(employee);
  170. }
  171. }
  172. }
  173. }
  174. ```