Delegate(委托与事件)

Delegate可以当它是一个占位符,比如你在写代码的时候并不知道你将要处理的是什么。你只需要知道你将要引入的参数类型和输出类型是什么并定义它即可。这就是书本上所传达的方法签名必须相同的意思。

系统自带的泛型委托

Action 无参数无返回值

Action<T> 没有返回值,最多有16个参数

Action<int> func = e => Console.WriteLine(e);

func(123)

Func 具有返回值,最多有16个参数

Func<int> f = () => 123;

Console.WriteLine(f() + 1);

List<string> list = new List<string>("赵晓虎,王小虎,乐虎,老虎,东北虎,华南虎".Split(','));

foreach (string item in list.Where(e => e.Contains("华"))) //Where,输入参数为字符串,返回参数为布尔
{
Console.WriteLine(item);
}

public class Person
    {
        public string Name { get; set; }
        public int Age { get; set; }
        public char Sex { get; set; }

        public override string ToString()
        {
            return string.Format("{0},{1}", Name, Age);
        }
    }

------------------------------------------------------------------
    public delegate int MyCompareHandler(Person p1, Person p2);

    public class Program
    {
        static void Main(string[] args)
        {
            MyCompareHandler MyCompare = null;
            Person[] ps = 
            {
                new Person() { Age=10, Name="赵晓虎", Sex='' },
                new Person() { Age=30, Name="波波", Sex='' },
                new Person() { Age=35, Name="杨茜", Sex='' },
                new Person() { Age=13, Name="马伦", Sex='' }
            };
            // 排序
            // 比较
            // int n = string.Compare("a", "b");
            // MyCompare = CompareByAge;
            //MyCompare = CompareByName;

            //if (MyCompare != null)
            //{
            //    int num = MyCompare(ps[0], ps[1]);
            //}

            // MyCompareHandler compare = CompareByAge;

            // Sort(ps, compare);

            // Sort(ps, CompareByName);

            Sort(ps, (a, b) => string.Compare(a.Name, b.Name));
            Sort(ps, (a, b) => a.Age - b.Age);
        }
//冒泡排序

        public static void Sort(Person[] ps, MyCompareHandler compare)
        {
            if (compare == null) return;
            for (int i = 0; i < ps.Length - 1; i++)
            {
                for (int j = 0; j < ps.Length - i - 1; j++)
                {
                    if (compare(ps[j], ps[j + 1]) > 0)
                    {
                        Person pTemp = ps[j];
                        ps[j] = ps[j + 1];
                        ps[j + 1] = pTemp;
                    }
                }
            }
        }

        //public static int CompareByAge(Person p1, Person p2)
        //{
        //    // n1 > n2 => n1 - n2 > 0
        //    return p1.Age - p2.Age;
        //}
        //public static int CompareByName(Person p1, Person p2)
        //{
        //    return string.Compare(p1.Name, p2.Name);
        //}
    }

 委托计算器

public class Calculator
    {
        public static int AdditionMethod(int num1, int num2)
        {
            return num1 + num2;
        }
        public static int SubtractionMethod(int num1, int num2)
        {
            return num1 - num2;
        }
        public static int MultiplicationMethod(int num1, int num2)
        {
            return num1 * num2;
        }
        public static int DivisionMethod(int num1, int num2)
        {
            return num1 / num2;
        }
    }


public delegate int MyCalculator(int n1, int n2);

 class Program
    {
        static void Main(string[] args)
        {
            // 需要什么?
            // 委托
            // 实现加减乘除的方法

            Console.WriteLine("请输入数字");
            int num1 = Convert.ToInt32(Console.ReadLine());
            Console.WriteLine("请输入运算符");
            string oper = Console.ReadLine();
            Console.WriteLine("请输入数字");
            int num2 = Convert.ToInt32(Console.ReadLine());

            MyCalculator cal = null;

            switch (oper)
            {
                case "+": cal = Calculator.AdditionMethod; break;
                case "-": cal = Calculator.SubtractionMethod; break;
                case "*": cal = Calculator.MultiplicationMethod; break;
                case "/": cal = Calculator.DivisionMethod; break;
            }

            if (cal != null)
            {
                Console.WriteLine("{0}{1}{2}={3}", num1, oper, num2, cal(num1, num2));
            }

            Console.ReadKey();
        }
    }
委托计算器

-------------------------------------------------------------

窗体传值

   public partial class Form1 : Form
    {
        Random r = new Random();
        public Form1()
        {
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {
            ChangeColor();
        }

        private void ChangeColor()
        {
            this.BackColor = Color.FromArgb(r.Next(256), r.Next(256), r.Next(256));
            
        }

        int i = 0;
        private void timer1_Tick(object sender, EventArgs e)
        {
            //i += 10;
            //    this.BackColor = Color.FromArgb(i, 256 - i, i);
           
        }

        private void button2_Click(object sender, EventArgs e)
        {
            // 创建子窗体
            Form2 f = new Form2();

            // 订阅方法
            f.MyFunc += ChangeColor;

            f.Show();
        }
    }

//////////////////////////////////////////////////////////////////////////
    public partial class Form2 : Form
    {
        public event Action MyFunc;


        public Form2()
        {
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {
            if (MyFunc != null)
            {
                MyFunc();
            }
        }
    }
View Code

-------------------------------------------------------------

有些时候委托绑定的方法比较简单,而且在代码中可能只是用一次
-> 如果不用定义,直接写一个表达式或者一个赋值操作就可以完成这个功能就好了
-> 匿名方法
-> 语法:
委托类型 委托名 = delegate(参数列表) { /* 方法体 */ }; 2、 SumHandler sum = (n1, n2) => n1 + n2;
Lambda表达式
参数 => 方法体
=> goes to
参数类型在不冲突的情况下可以省略
如果参数只有一个,可以使用
e => ...
没有参数的时候,可以使用
() => ...
方法体如果只有一句话,省略{}
方法体如果只有一句话,并且有返回值,可以省略return

---------------------------------------------------------------

与函数指针相比,delegate有许多函数指针不具备的优点。首先,函数指针只能指向静态函数,而delegate既可以引用静态函数,又可以引用非静态成员函数。

 

委托对象可使用 "+" 运算符进行合并。一个合并委托调用它所合并的两个委托。只有相同类型的委托可被合并。"-" 运算符可用于从合并的委托中移除组件委托。

使用委托的这个有用的特点,您可以创建一个委托被调用时要调用的方法的调用列表。这被称为委托的 多播(multicasting),也叫组播。

 

调用(invoke)委托,相当于调用委托所绑定的方法,一个委托可以绑定多个方法,使用"+="就可以向委托中添加新的方法,使用"-="可以从委托中删除方法

 利用鸿门宴解释委托与事件http://www.cnblogs.com/yinqixin/p/5056307.html

结合delegate的实现,我们可以将自定义事件的实现归结为以下几步:

1.定义delegate对象类型,它有两个参数,第一个参数是事件发送者对象,第二个参数是事件参数类对象。
2.定义事件参数类,此类应当从System.EventArgs类派生。如果事件不带参数,这一步可以省略。
3.定义事件处理方法,它应当与delegate对象具有相同的参数和返回值类型。
4. 用event关键字定义事件对象,它同时也是一个delegate对象。
5.用+=操作符添加事件到事件队列中(-=操作符能够将事件从队列中删除)。
6.在需要触发事件的地方用调用delegate的方式写事件触发方法。一般来说,此方法应为protected访问限制,既不能以public方式调用,但可以被子类继承。名字是OnEventName。
7. 在适当的地方调用事件触发方法触发事件。下面是一个简单的例子:

using System;
public class EventTest
{
// 步骤1,定义delegate对象
public delegate void MyEventHandler(object sender, System.EventArgs e);
// 步骤2省略
public class MyEventCls
{
// 步骤3,定义事件处理方法,它与delegate对象具有相同的参数和返回值类//
public void MyEventFunc(object sender, System.EventArgs e)
{
Console.WriteLine("My event is ok!");
}
}
// 步骤4,用event关键字定义事件对象
private event MyEventHandler myevent;
private MyEventCls myecls;
public EventTest()
{
myecls = new MyEventCls();
// 步骤5,用+=操作符将事件添加到队列中
this.myevent += new MyEventHandler(myecls.MyEventFunc);
}
// 步骤6,以调用delegate的方式写事件触发函数
protected void OnMyEvent(System.EventArgs e)
{
if(myevent != null)
myevent(this, e);
}
public void RaiseEvent()
{
EventArgs e = new EventArgs();
// 步骤7,触发事件
OnMyEvent(e);
}
public static void Main()
{
EventTest et = new EventTest();
Console.Write("Please input ''a'':");
string s = Console.ReadLine();
if(s == "a")
{
et.RaiseEvent();
}
else
{
Console.WriteLine("Error");
}
}
}

 

using System;
using System.Text;
using System.Data;

namespace Class1
{    
    //定义事件引发时,需要传的参数
    class NewMailEventArgs:EventArgs
    {
        private readonly string m_from;
        private readonly string m_to;
        private readonly string m_subject;
        public NewMailEventArgs(string from, string to, string subject)
        {
            m_from = from;
            m_to = to;
            m_subject = subject;
        }
        public string From
        {
            get{return m_from;}
        }
        public string To
        {
            get{return m_to;}
        }
        public string Subject
        {
            get{return m_subject;}
        }

    }

    //事件所用的委托(链表)
    delegate void NewMailEventHandler(object sender, NewMailEventArgs e);

    //提供事件的类
    class MailManager
    {
        public event NewMailEventHandler NewMail;
        //通知已订阅事件的对象
        protected virtual void OnNewMail(NewMailEventArgs e)
        {
            NewMailEventHandler temp = NewMail; //MulticastDelegate一个委托链表
            //通知所有已订阅事件的对象
            if(temp != null)
                temp(this,e); //通过事件NewMail(一种特殊的委托)逐一回调客户端的方法

        }
        //提供一个方法,引发事件
        public void SimulateNewMail(string from, string to, string subject)
        {
            NewMailEventArgs e = new NewMailEventArgs(from,to,subject);
            OnNewMail(e);
        }
    }


    //使用事件
    class Fax
    {
        public Fax(MailManager mm)
        {
            //Subscribe 
            mm.NewMail += new NewMailEventHandler(Fax_NewMail);
        }
        private void Fax_NewMail(object sender, NewMailEventArgs e)
        {
            Console.WriteLine("Message arrived at Fax...");
            Console.WriteLine("From={0}, To={1}, Subject='{2}'",e.From,e.To,e.Subject);
        }
        public void Unregister(MailManager mm)
        {
            mm.NewMail -= new NewMailEventHandler(Fax_NewMail);
        }
    }
    class Print
    {
        public Print(MailManager mm)
        {
            //Subscribe ,在mm.NewMail的委托链表中加入Print_NewMail方法
            mm.NewMail += new NewMailEventHandler(Print_NewMail);
        }
        private void Print_NewMail(object sender, NewMailEventArgs e)
        {
            Console.WriteLine("Message arrived at Print...");
            Console.WriteLine("From={0}, To={1}, Subject='{2}'",e.From,e.To,e.Subject);
        }
        public void Unregister(MailManager mm)
        {
            mm.NewMail -= new NewMailEventHandler(Print_NewMail);
        }
    }

    class ExcelProgram
    {
        [STAThread]
        static void Main(string[] args)
        {    
            MailManager mm = new MailManager();
            if(true)
            {
                Fax fax = new Fax(mm);
                Print prt = new Print(mm);
            }

            mm.SimulateNewMail("Anco","Jerry","Event test");
            Console.ReadLine();
        }
    }
}

 

 

ref和out的区别在C# 中,既可以通过值也可以通过引用传递参数。通过引用传递参数允许函数成员更改参数的值,并保持该更改。若要通过引用传递参数, 可使用ref或out关键字。ref和out这两个关键字都能够提供相似的功效,其作用也很像C中的指针变量。它们的区别是:

1、使用ref型参数时,传入的参数必须先被初始化。对out而言,必须在方法中对其完成初始化。

2、使用ref和out时,在方法的参数和执行方法时,都要加Ref或Out关键字。以满足匹配。

3、out适合用在需要retrun多个返回值的地方,而ref则用在需要被调用的方法修改调用者的引用的时候。

注:在C#中,方法的参数传递有四种类型:传值(by value),传址(by reference),输出参数(by output),数组参数(by array)。传值参数无需额外的修饰符,传址参数需要修饰符ref,输出参数需要修饰符out,数组参数需要修饰符params。传值参数在方法调用过程中如果改变了参数的值,那么传入方法的参数在方法调用完成以后并不因此而改变,而是保留原来传入时的值。传址参数恰恰相反,如果方法调用过程改变了参数的值,那么传入方法的参数在调用完成以后也随之改变。实际上从名称上我们可以清楚地看出两者的含义--传值参数传递的是调用参数的一份拷贝,而传址参数传递的是调用参数的内存地址,该参数在方法内外指向的是同一个存储位置。

方法参数上的 ref 方法参数关键字使方法引用传递到方法的同一个变量。当控制传递回调用方法时,在方法中对参数所做的任何更改都将反映在该变量中。

若要使用 ref 参数,必须将参数作为 ref 参数显式传递到方法。ref 参数的值被传递到 ref 参数。

传递到 ref 参数的参数必须最先初始化。将此方法与 out 参数相比,后者的参数在传递到 out 参数之前不必显式初始化。

属性不是变量,不能作为 ref 参数传递。

如果两种方法的声明仅在它们对 ref 的使用方面不同,则将出现重载。但是,无法定义仅在 ref 和 out 方面不同的重载。

out

方法参数上的 out 方法参数关键字使方法引用传递到方法的同一个变量。当控制传递回调用方法时,在方法中对参数所做的任何更改都将反映在该变量中。

当希望方法返回多个值时,声明 out 方法非常有用。使用 out 参数的方法仍然可以返回一个值。一个方法可以有一个以上的 out 参数。

若要使用 out 参数,必须将参数作为 out 参数显式传递到方法。out 参数的值不会传递到 out 参数。

不必初始化作为 out 参数传递的变量。然而,必须在方法返回之前为 out 参数赋值。

属性不是变量,不能作为 out 参数传递。

posted @ 2016-12-03 14:21  ecollab  阅读(3370)  评论(0编辑  收藏  举报