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. (十九)WebGIS中I查询的原理及设计(包含AGS、GeoServer、Supermap)

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/. 1.背景 我们在使用arcmap时,经常会用到被称为I查询的工具.具体 ...

  2. HTML基本元素(一)

    HTML基本元素(一) 1.换行符 <br /> Ps:br 是换行(Break)的缩写,文本会在这个标签的地方换行. 实例: 第一行<br />第二行 2.段落 <p& ...

  3. 给 C# 开发者的代码审查清单

    这是为C#开发者准备的通用性代码审查清单,可以当做开发过程中的参考.这是为了确保在编码过程中,大部分通用编码指导原则都能注意到.对于新手和缺乏经验(0到3年工作经验)的开发者,参考这份清单编码会很帮助 ...

  4. React Native at first sight

    what is React Native? 跟据官方的描述, React Native是一套使用 React 构建 Native app 的编程框架. 推出不久便引发了广泛关注, 这也得益于 Java ...

  5. authorization与URL授权

    利用Web.config中的authorization标签设置授权属于URL授权. 使用 URL 授权 通过 URL 授权,您可以显式允许或拒绝某个用户名或角色对特定目录的访问权限.为此,请在该目录的 ...

  6. 浅谈web网站架构演变过程

    前言 我们以javaweb为例,来搭建一个简单的电商系统,看看这个系统可以如何一步步演变.   该系统具备的功能:   用户模块:用户注册和管理 商品模块:商品展示和管理 交易模块:创建交易和管理 阶 ...

  7. 在Windows Server 2012 R2的Hyper-V中设置虚拟机启用增强会话模式

    在Windows Server 2012 R2的Hyper-V中,可以为虚拟机提供一种全新的连接方式,就是“增强会话模式”,它将让您更加方便的对虚拟机进行操作,比如分辨率的调整.设备的加载,最为方便的 ...

  8. IOS 动画播放案例

    #import "ViewController.h" @interface ViewController () @property (weak,nonatomic) IBOutle ...

  9. MVC 自定义Htmlhelper扩展

    在MVC中,我们不仅可以使用它原来的方法,我们还可以自定义,这不不仅加大了我们开发的效率,同时使界面更简洁. 具体什么是扩展方法,你可以这样理解,必须是静态且在形参中第一个参数是以this开头,大概先 ...

  10. JDK,JRE,JVM分别是什么?

    JDK(Java Development Kit) 是 Java 语言的软件开发工具包(SDK). JDK包含的基本组件包括: javac – 编译器,将源程序转成字节码 jar – 打包工具,将相关 ...