c# 委托

来源:互联网 发布:数据结构kmp算法 编辑:程序博客网 时间:2024/06/05 06:44

委托是一个类,它定义了方法的类型,使得可以将方法当作另一个方法的参数来进行传递。事件是一种特殊的委托。

  1.委托的声明

  (1). delegate

        delegate我们常用到的一种声明

    Delegate至少0个参数,至多32个参数,可以无返回值,也可以指定返回值类型。

    例:publicdelegate int MethodtDelegate(int x,int y);表示有两个参数,并返回int型。

  (2). Action

       Action是无返回值的泛型委托。

   Action 表示无参,无返回值的委托

   Action<int,string> 表示有传入参数int,string无返回值的委托

   Action<int,string,bool> 表示有传入参数int,string,bool无返回值的委托

       Action<int,int,int,int> 表示有传入4个int型参数,无返回值的委托

   Action至少0个参数,至多16个参数,无返回值。

   例:

       public void Test<T>(Action<T> action,T p)

       {

           action(p);

       }

    (3). Func

   Func是有返回值的泛型委托

   Func<int> 表示无参,返回值为int的委托

   Func<object,string,int> 表示传入参数为object, string 返回值为int的委托

   Func<object,string,int> 表示传入参数为object, string 返回值为int的委托

   Func<T1,T2,,T3,int> 表示传入参数为T1,T2,,T3(泛型)返回值为int的委托

   Func至少0个参数,至多16个参数,根据返回值泛型返回。必须有返回值,不可void

      例:   

       public int Test<T1,T2>(Func<T1,T2,int>func,T1 a,T2 b)

       {

           return func(a, b);

       }

    (4) .predicate

   predicate 是返回bool型的泛型委托

   predicate<int> 表示传入参数为int 返回bool的委托

   Predicate有且只有一个参数,返回值固定为bool

   例:publicdelegate bool Predicate<T> (T obj)

  

  2.委托的使用

  (1).Delegate的使用  


       public delegateint MethodDelegate(int x,int y);

       private static MethodDelegate method;

       static void Main(string[] args)

       {

           method = new MethodDelegate(Add);

           Console.WriteLine(method(10,20));

           Console.ReadKey();

       }


       private staticint Add(int x,int y)

       {

           return x + y;

       }


  (2).Action的使用   


 staticvoid Main(string[] args)

       {

           Test<string>(Action,"Hello World!");

           Test<int>(Action, 1000);

           Test<string>(p => { Console.WriteLine("{0}", p); },"Hello World");//使用Lambda表达式定义委托

           Console.ReadKey();

       }

       public staticvoid Test<T>(Action<T> action, T p)

       {

           action(p);

       }

       private staticvoid Action(string s)

       {

           Console.WriteLine(s);

       }

       private staticvoid Action(int s)

       {

           Console.WriteLine(s);

       }


  可以使用 Action<T1, T2, T3, T4> 委托以参数形式传递方法,而不用显式声明自定义的委托。 封装的方法必须与此委托定义的方法签名相对应。 也就是说,封装的方法必须具有四个均通过值传递给它的参数,并且不能返回值。 (在 C# 中,该方法必须返回 void)通常,这种方法用于执行某个操作。

  (3).Func的使用


       static void Main(string[] args)

       {

           Console.WriteLine(Test<int,int>(Fun,100,200));

           Console.ReadKey();

       }

       public staticint Test<T1, T2>(Func<T1, T2, int> func, T1 a, T2 b)

       {

           return func(a, b);

       }

       private staticint Fun(int a,int b)

       {

           return a + b;

       }


  (4). predicate的使用

  泛型委托:表示定义一组条件并确定指定对象是否符合这些条件的方法。此委托由 Array 和 List 类的几种方法使用,用于在集合中搜索元素。


       static void Main(string[] args)

       {

           Point[] points = { new Point(100,200), 

           new Point(150,250), new Point(250,375), 

           new Point(275,395), new Point(295,450) };

           Point first = Array.Find(points, ProductGT10);

           Console.WriteLine("Found: X = {0}, Y = {1}", first.X, first.Y);

           Console.ReadKey();

       }

       private staticbool ProductGT10(Point p)

       {

           if (p.X * p.Y > 100000)

           {

               return true;

           }

           else

           {

               return false;

           }

       }


  使用带有 Array.Find 方法的 Predicate 委托搜索 Point 结构的数组。如果 X 和 Y 字段的乘积大于 100,000,此委托表示的方法 ProductGT10 将返回 true。Find 方法为数组的每个元素调用此委托,在符合测试条件的第一个点处停止。

  3.委托的清空

  (1).在类中申明清空委托方法,依次循环去除委托引用。

         方法如下:


     public MethodDelegate OnDelegate;               

       public void ClearDelegate()       

       {             

           while (this.OnDelegate !=null) 

           {                 

               this.OnDelegate -= this.OnDelegate; 

           }        

       } 


  (2).如果在类中没有申明清空委托的方法,我们可以利用GetInvocationList查询出委托引用,然后进行去除。  

  方法如下:


       public MethodDelegate OnDelegate; 

     staticvoid Main(string[] args)

       {

           Program test = new Program();


           if (test.OnDelegate != null) 

           { 

               System.Delegate[] dels = test.OnDelegate.GetInvocationList(); 

               for (int i =0; i < dels.Length; i++) 

               {

                   test.OnDelegate -= dels[i] as MethodDelegate;

               }

           }

       }


  4.委托的特点

  委托类似于 C++ 函数指针,但它们是类型安全的。

  委托允许将方法作为参数进行传递。

  委托可用于定义回调方法。

  委托可以链接在一起;例如,可以对一个事件调用多个方法。

  方法不必与委托签名完全匹配。

  5.总结:

   Delegate至少0个参数,至多32个参数,可以无返回值,也可以指定返回值类型

  Func可以接受0个至16个传入参数,必须具有返回值

  Action可以接受0个至16个传入参数,无返回值

  Predicate只能接受一个传入参数,返回值为bool类型


其实他们两个都是委托【代理】的简写形式。

一、【action<>】指定那些只有输入参数,没有返回值的委托

Delegate的代码:

[csharp] view plaincopy


  1. public delegate void myDelegate(string str);  
  2. public static void HellowChinese(string strChinese)  
  3. {  
  4.     Console.WriteLine("Good morning," + strChinese);  
  5.     Console.ReadLine();  
  6. }  
  7.   
  8. myDelegate d = new myDelegate(HellowChinese);  
  9. d("Mr wang");  


用了Action之后呢:

 

[csharp] view plaincopy


  1. public static void HellowChinese(string strChinese)  
  2. {  
  3.     Console.WriteLine("Good morning," + strChinese);  
  4.     Console.ReadLine();  
  5. }  
  6.   
  7. Action<string> action = HellowChinese;  
  8. action("Spring.");  

就是相当于省去了定义委托的步骤了。

 

二、func<> 这个和上面的那个是一样的,区别是这个有返回值!

[csharp] view plaincopy


  1. public static string HelloEnglish(string strEnglish)  
  2. {  
  3.     return "Hello." + strEnglish;  
  4. }  
  5.   
  6. Func<stringstring> f = HelloEnglish;  
  7. Console.WriteLine(f("Srping ji"));  
  8. Console.ReadLine();  

 

  详细参考:http://www.fengfly.com/plus/view-209140-1.html

       http://www.cnblogs.com/foolishfox/archive/2010/09/16/1827964.html

原创粉丝点击