1. 委托概述

“委托”相当于C++中的“函数指针”,委托必须与所要“指向”的函数在“参数”和“返回类型”上保持一致;

// 定义Person类
public class Person {
public string Name = "Rain Man";
public string Speak(string words) {
Console.WriteLine(this.Name + " said: " + words);
return words;
}
} // 定义委托
public delegate string Dele_Speak(string str); class Program {
static void Main(string[] args) {
Person p = new Person(); // 实例化Person类
Dele_Speak dp = new Dele_Speak(p.Speak); // 实例化委托:变量dp实际上就是指向p.Speak函数的指针
dp("Welcome to my blog!"); // 输出:Rain Man said: Welcome to my blog!
Console.ReadLine();
}
}
  • 代理“Dele_Speak”与“Speak”方法在参数和返回类型保持一致;
  • “Dele_Speak dp = new Dele_Speak(p.Speak)”,实际上就是创建了一个“dp”指针,指向“p.Speak”方法
  • “dp("Welcome to my blog!")”,实际上就是“p.Speak("Welcome to my blog!")”

2. 多路广播

// 定义Person类
public class Person {
public string Speak(string words) {
Console.WriteLine("Speak: " + words);
return "111";
}
public string Say(string words) {
Console.WriteLine("Say: " + words);
return "222";
}
public string Translate(string words) {
Console.WriteLine("Translate: " + words);
return "333";
}
}
// 声明代理
public delegate string Dele_Str(string str); class Program {
static void Main(string[] args) {
Person p = new Person(); // 实例化Person类
Dele_Str dp_Speak = new Dele_Str(p.Speak); // 实例化委托指向 p.Speak
Dele_Str dp_Say = new Dele_Str(p.Say); // 实例化委托指向 p.Say
Dele_Str dp_Translate = new Dele_Str(p.Translate); // 实例化委托指向 p.Transpate // 多路广播
dp_Speak = dp_Speak + dp_Say;
dp_Speak = dp_Speak + dp_Translate; string str = dp_Speak("Rain Man");
Console.WriteLine(str); // 输出:333 Console.ReadLine();
}
}

在Person类中创建了三个函数:Speak、Say、Translate,这三个函数在参数和返回类型上相同,因此可是使用同一个委托(Dele_Str)。

多路委托:使用同一个委托“指向”不同的函数,使这几个函数可以“计算”,其执行逻辑如下:

执行:
string str = dp_Speak("Rain Man");
输出:
Speak: Rain Man
Say: Rain Man
Translate: Rain Man 实际上就是执行下述代码:
p.Speak("Rain Man");
p.Say("Rain Man");
p.Translate("Rain Man"); 返回值:即最后一个函数的返回值

3. 事件代理

有两个窗体:

  • FrmMain:该窗体中有一个按钮“btnAdd”,当点击此按钮时通过ShowDialog()方法打开“FrmUserAdd”窗体
  • FrmUserAdd: 该窗体中有一个按钮“btnOK”,当点击此按钮时“对外”(对FrmMain窗体)发送一个“UserAddEvent”事件,通过该事件将“FrmUserAdd”中填写的“用户信息”传至“FrmMain”窗体中。

3.1 FrmUserAdd窗体:

public partial class FrmUserAdd : Form 
{
// 1. 定义事件参数类
public class UserAddEventArgs : EventArgs {
public User AddedUser;
public UserAddEventArgs(User user) {
this.AddedUser = user;
}
} // 2. 定义委托,并指定参数类型
public delegate void UserAddEventHandler(object sender, UserAddEventArgs e); // 3. 定义事件,并指定该事件的委托类型
public event UserAddEventHandler UserAddEvent; private void btnOK_Click(object sender, EventArgs e) {
User user = new User(1, "Rain Man", "");
UserAddEventArgs args = new UserAddEventArgs(user);
if (UserAddEvent != null) {
this.UserAddEvent(this, args);
}
}
}

3.1.1. 自定义事件参数类:UserAddEventArgs

自定义的事件参数类“UserAddEventArgs”必须继承自“EventArgs”类,在此基础上添加了public成员“AddedUser”

3.1.2 定义委托:UserAddEventHandler

  • 注意该委托的参数类型,第二个参数为“自定义的事件参数”。
  • 该委托用于在“FrmMain”窗体中实例化,实例化后绑定事件处理函数“OnUserAdd”。

3.1.3 定义事件变量:UserAddEvent

“UserAddEvent”变量可以理解为“UserAddEventHandler”委托的一个实例化对象,即

public UserAddEventHandler UserAddEvent;    // 在该示例中把"event"修饰符去掉也是可以的

3.2 FrmMain窗体

public partial class FrmMain : Form {
// UserAddEvent事件绑定的处理函数
private void OnUserAdd(object sender, FrmUserAdd.UserAddEventArgs e) {
MessageBox.Show(e.AddedUser.username);
} private void btnAdd_Click(object sender, EventArgs e) {
FrmUserAdd frm = new FrmUserAdd();
FrmUserAdd.UserAddEventHandler dele_fn = new FrmUserAdd.UserAddEventHandler(OnUserAdd); frm.UserAddEvent += dele_fn;
frm.ShowDialog();
}
}

3.2.1 FrmUserAdd.UserAddEventHandler dele_fn = new FrmUserAdd.UserAddEventHandler(OnUserAdd);

dele_fn为“UserAddEventHandler”的一个实例(指针),它指向事件处理函数“OnUserAdd”

3.2.2 frm.UserAddEvent += dele_fn;

可以看出此处实际就是“多路广播”,同时也可以看出“UserAddEvent”事件变量实际就是“UserAddEventHandler”委托的一个实例。

3.3 执行逻辑

该示例看似复杂,其实质是将本在“一个窗体”中的实现,拆成了“两个窗体”。下面将两个窗体的代码合成“一个窗体”

public partial class FrmUserAdd : Form {
// 定义事件参数
public class UserAddEventArgs : EventArgs {
public User AddedUser;
public UserAddEventArgs(User user) {
this.AddedUser = user;
}
} // 定义委托,并指定参数类型
public delegate void UserAddEventHandler(object sender, UserAddEventArgs e); // 定义事件,并指定该事件的“委托”
public UserAddEventHandler UserAddEvent;
public event UserAddEventHandler UserAddEvent; // UserAddEvent事件绑定的处理函数
private void OnUserAdd(object sender, FrmUserAdd.UserAddEventArgs e) {
MessageBox.Show(e.AddedUser.username);
} private void btnOK_Click(object sender, EventArgs e) {
User user = new User(1, "Rain Man", "");
UserAddEventArgs args = new UserAddEventArgs(user); FrmUserAdd.UserAddEventHandler dele_fn = new FrmUserAdd.UserAddEventHandler(OnUserAdd);
this.UserAddEvent += dele_fn; if (UserAddEvent != null)
{
this.UserAddEvent(this, args);
}
}
}

C#:委托和自定义事件的更多相关文章

  1. C#委托五(自定义事件)

    事件: "在发生其他类或对象关注的事情时,类或对象可以通过事件通知他们.发送(或引发)事件的类称为"发行者",接受(或处理)事件的类称为"订户".&q ...

  2. C# 窗体间传值(使用委托与自定义事件)

    using System; using System.Drawing; using System.Windows.Forms; namespace 跨窗体调用控件 { public partial c ...

  3. 分享一个C#自定义事件的实际应用

    在C#.NET的开发中,事件是经常接触到的概念,比如为按钮添加点击事件,并写入点击按钮触发事件要运行的代码.不管是ASP.NET还是WinForm等各种形式的应用程序,最经常是为系统生成的事件写具体代 ...

  4. javaScript事件机制深入学习(事件冒泡,事件捕获,事件绑定方式,移除事件方式,阻止浏览器默认行为,事件委托,模拟浏览器事件,自定义事件)

    前言 JavaScript与HTML之间的交互是通过事件实现的.事件,就是文档或浏览器窗口中发生的一些特定的交互瞬间.可以使用侦听器(或处理程序)来预订事件,以便事件发生时执行相应的代码.这种在传统软 ...

  5. jQuery 学习笔记(5)(事件绑定与解绑、事件冒泡与事件默认行为、事件的自动触发、自定义事件、事件命名空间、事件委托、移入移出事件)

    1.事件绑定: .eventName(fn) //编码效率略高,但部分事件jQuery没有实现 .on(eventName, fn) //编码效率略低,所有事件均可以添加 注意点:可以同时添加多个相同 ...

  6. JS自定义事件之选项卡

    自定义事件是一种处理与DOM产生交互的代码逻辑片段之间耦合的很好的架构方法. 一个简单的jQuery插件——选项卡 让ul列表来响应点击事件.当用户点击一个列表项时,给这个列表项添加一个名为activ ...

  7. EventHandler委托与自定义委托

    http://blog.csdn.net/uuxyz/article/details/7175248 EventHandler委托与自定义委托 自定义委托: //1. public delegate ...

  8. C# 自定义事件(EventArgs)

    1,自定义事件: public class TextChangeEventArgs : EventArgs {     private string message;     public TextC ...

  9. Flex之自定义事件

    1.通过dispatchEvent委托事件模式完成自定义事件: <?xml version="1.0" encoding="utf-8"?> < ...

随机推荐

  1. SQLServer学习笔记系列11

    一.写在前面的话 身体是革命的本钱,这句放在嘴边常说的话,还是拿出来一起共勉,提醒一起奋斗的同僚们,保证睡眠,注意身体!偶尔加个班,也许不曾感觉到身体发出的讯号,长期晚睡真心扛不住!自己也制定计划,敦 ...

  2. SQL Server 存储过程中处理多个查询条件的几种常见写法分析,我们该用那种写法

    本文出处: http://www.cnblogs.com/wy123/p/5958047.html 最近发现还有不少做开发的小伙伴,在写存储过程的时候,在参考已有的不同的写法时,往往很迷茫,不知道各种 ...

  3. CSS 巧用 :before和:after

    前几天的晚上较全面的去看了下css的一些文档和资料,大部分的样式运用都没什么大问题了,只是有些许较陌生,但是也知道他们的存在和实现的是什么样式.今天主要想在这篇学习笔记中写的也不多,主要是针对:bef ...

  4. [Java 基础]基础语法

    Java代码基本规范 大小写敏感:Java是大小写敏感的,这就意味着标识符Hello与hello是不同的. 类名:对于所有的类来说,类名的首字母应该大写.如果类名由若干单词组成,那么每个单词的首字母应 ...

  5. 使用AutoMapper进行对象间映射

    在开发过程中,难免遇到下面这种情况:两个(或多个)对象所拥有的大多数属性是重复的,我们需要在对象间进行映射(即将一个对象的属性值赋给另一个对象.通常我们可以进行如下操作: A a=new A(); a ...

  6. 你不知道的HttpHandler相关知识

    一.关于IHttpHandler.IsReusable 很多人不明白,这哥们到底干嘛的,估计是微软最初的一个想法--让一个对象可以一直不断地被重复使用 ,但这个想法不成熟,会带来很多隐藏问题,一个对象 ...

  7. Debugging into .NET Core源代码的两种方式

    一.前言 .NET开源时间还不长,因为一直在做YOYOFx的关系,所似我常常有更深入的了解.NET Core和ASP.NET Core内容的需求,并且.NET Core平台与之前版本的变化太大,这也导 ...

  8. 我的runtime学习笔记

    0.简介: OC方法不同于C语言函数,属于动态调用过程,在编译的时候并不能决定真正调用哪个函数,只有在真正运行的时候才会根据函数的名称找到对应的函数来调用. 至于其他理论上的东西不必讲太多,编程讲的就 ...

  9. 修复DapperExtension做Insert对象主键为Guid时不能赋值的问题

    最新的dapperExtension有个bug,就是当做Insert操作的时候,实体的主键类型为GUID的时候,会自动生产一个新的GUID替换原来的GUID,使得使用者在Insert的时候不能在外部指 ...

  10. 数据结构(C语言第2版)-----数组,广义表,树,图

    任何一个算法的设计取决于选定的数据结构,而算法的实现依赖于采用的存储结构. 之前线性表的数据元素都是非结构的原子类型,元素的值是不可再分的.下面学习的这两个线性表是很特殊的,其中数据元素本身也可能是一 ...