【.NET基础】--委托、事件、线程(1)
1,委托 是存放方法的指针的清单,也就是装方法的容器
A, 新建winform项目【01委托】,项目中添加dg_SayHi.cs 委托类 用于存储方法
namespace _01委托
{
//定义委托 【类】
delegate void dg_SayHi();
}
B, Winfrom中添加按钮 "委托",按钮事件“btnDelegate_Click”中新建委托对象,并添加、移除 方法:
/// <summary>
/// 点击按钮事件
/// </summary>
private void btnDelegate_Click(object sender, EventArgs e)
{
//委托是方法的容器,可以在委托对象中添加、移除方法 ///创建委托对象,并通过构造函数添加SayHiCN方法
dg_SayHi objSayHi = new dg_SayHi(SayHiCN);
//向委托对象中 "添加" 一个方法
objSayHi += SayHiEN;
//从委托对象中 "移除" 一个方法
objSayHi -= SayHiCN; //调用委托,委托是方法的容器 ,调用委托的时候委托里面的方法都会被调用
objSayHi(); } void SayHiCN()
{
MessageBox.Show("你好上海");
}
void SayHiEN()
{
MessageBox.Show("hi shanghai");
}
}
C,点击“委托”,查看效果:

D, 一张图片解释上面的运行过程:

2, 委托返回类型和添加到委托的方法的返回类型保持一致
委托可以作为参数使用,传递方法。把委托对象作为参数,委托对象是方法的容器(上图中的椭圆),这样实际上就传递了方法;上面示例我们只是在委托中添加了方法,还没有把委托作为参数传递,我们看下面的示例介绍
A, 我们有两个方法,分别是获取int数组中最大数的方法 int GetMaxNum() 和 获取int数组中最小数的方法 int GetMinNum()
/// <summary>
/// 2.1 返回数组中的最大数
/// </summary>
/// <returns></returns>
int GetMaxNum(int[] arr)
{
int numMax = arr[];
for (int i = ; i < arr.Length; i++)
{
if (arr[i] > numMax)
{
numMax = arr[i];
}
}
return numMax;
}
/// <summary>
/// 2.2 返回数组中的最小数
/// </summary>
/// <returns></returns>
int GetMinNum(int[] arr)
{ int numMin = arr[];
for (int i = ; i < arr.Length; i++)
{
if (arr[i] < numMin)
{
numMin = arr[i];
}
}
return numMin;
}
B,如果把这两个方法添加到委托类中,所以我们需要添加int类型的委托类 int dg_GetMaxAndMin()
//定义委托 【类】
delegate void dg_SayHi(); //再定义一个委托类,接收返回类型为int的方法
delegate int dg_GetMaxAndMin(int[] arr);
C, 添加按钮事件,事件中读取委托中方法的返回值
private void BtnDelegateAsPara_Click(object sender, EventArgs e)
{
int[] arr = { , , , , , };
//创建委托对象,并通过构造函数添加 GetMaxNum 方法
dg_GetMaxAndMin objGetMax = new dg_GetMaxAndMin(GetMaxNum);
int numMax = objGetMax(arr); //传递参数数组
//委托对象添加 GetMinNum 方法
objGetMax += GetMinNum;
int numMin = GetMinNum(arr); //传递参数数组 MessageBox.Show("最大数" + numMax.ToString() + " 最小数" + numMin.ToString()); }
D,点击按钮查看效果:

AA, 我们还有一个方法返回年龄最大的Person
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace _01委托
{
public class Person
{
#region 姓名
string name; public string Name
{
get { return name; }
set { name = value; }
}
#endregion #region 年龄
int age; public int Age
{
get { return age; }
set { age = value; }
}
#endregion
}
}
/// <summary>
/// 2.3 返回年龄最大的Person
/// </summary>
/// <returns></returns>
Person GetMaxAgePer(Person[] perArr)
{
Person MaxAgePer = perArr[];
for (int i = ; i < perArr.Length; i++)
{
if (perArr[i].Age > MaxAgePer.Age)
{
MaxAgePer = perArr[i];
}
}
return MaxAgePer;
}
BB,如果我们把Person类添加到委托中,那么我们就还需要添加 Person类型的委托类 Person dg_GetMax()。
//定义委托 【类】
public delegate void dg_SayHi(); //再定义一个委托类,接收返回类型为int的方法
delegate int dg_GetMaxAndMin(int[] arr); //再定义一个委托类,接收返回类型为Person的方法
delegate Person dg_GetMaxAgePer(Person[] perArr);
CC,更新按钮事件,新建Person类型的委托对象,并添加 Person GetMaxAgePer()方法
private void BtnDelegateAsPara_Click(object sender, EventArgs e)
{
int[] arr = { , , , , , }; Person[] perArr = {
new Person {Name="甜馨",Age=},
new Person {Name="奥莉",Age=},
new Person {Name="嗯哼",Age=}
}; //创建委托对象,并通过构造函数添加 GetMaxNum 方法
dg_GetMaxAndMin objGetMax = new dg_GetMaxAndMin(GetMaxNum);
int numMax = objGetMax(arr); //传递参数数组
//委托对象添加 GetMinNum 方法
objGetMax += GetMinNum;
int numMin = GetMinNum(arr); //传递参数数组 dg_GetMaxAgePer objGetMaxAge = new dg_GetMaxAgePer(GetMaxAgePer);
Person per = objGetMaxAge(perArr); MessageBox.Show("最大数" + numMax.ToString() + " 最小数" + numMin.ToString());
MessageBox.Show(string.Format("最大年龄是:{0},年龄{1}岁", per.Name, per.Age)); }
DD, 运行查看运行结果:

3,委托改写为泛型委托,作为参数进行传递
按照上面的示例,我们每个不同返回类型的方法,都定义与之返回类型相同同类型的委托,有些麻烦,现在我们用泛型进行定义。
A, 由于不知道要比较的对象,我们先试着写一个泛型比较方法,如下,方法中传递过来了比较的对象T[] arr,但是对象我们不确定是int[]数组还是person[]数组,这样for循环里面我们就还需要添加一个具体的比较方法。这个方法就需要通过委托传递比较方法过来了!
public T GetMax<T>(T[] arr)
{
T max = arr[];
for (int i = ; i < arr.Length; i++)
{
//比较大小、具体该怎么判断? 判断的对象无法确定,有可能是int类型也有可能是Person类
}
return max;
}
B, 上面的For循环中应该是把下面的两个比较方法传递进去的
#region 2.4 两个不同对象的比较方法
//两个整形数比价大小
int Compare(int a, int b)
{
int num = ;
if (a > b)
{
num = ;
}
else if (a == b)
{
num = ;
}
else if (a < b)
{
num = -;
}
return num;
}
//两个Person类型比较大小
int Compare(Person AA, Person BB)
{
int num = ;
if (AA.Age > BB.Age)
{
num = ;
}
else if (AA.Age == BB.Age)
{
num = ;
}
else if (AA.Age < BB.Age)
{
num = -;
}
return num;
}
#endregion
C,现在我们定义可以接收上面两个不同对象比较方法的委托:int dg_GetMax<T>(T t1, T t2)
//定义委托 【类】
delegate void dg_SayHi(); //再定义一个委托类,接收返回类型为int的方法
delegate int dg_GetMaxAndMin(int[] arr); //再定义一个委托类,接收返回类型为Person的方法
delegate Person dg_GetMaxAgePer(Person[] perArr); //定义泛型委托, 这个委托接收比较不同对象大小的方法Compare, 委托方法的参数两个: t1,t2 ,比较这两个大小后返回 int数值
public delegate int dg_GetMax<T>(T t1, T t2);
D, 这样,我们就可以在A步骤中添加委托参数,实现比较
/// <summary>
/// 2.3 泛型方法,获得数组中最大的元素
/// </summary>
/// <typeparam name="T">泛型参数,如果比较的是int则返回int,如果比较的是person则返回person</typeparam>
/// <param name="arr">泛型数组</param>
/// <param name="dgGetMax">委托参数,传递过来可以装比较方法的委托</param>
/// <returns></returns>
public T GetMax<T>(T[] arr, dg_GetMax<T> dg)
{ T max = arr[];
for (int i = ; i < arr.Length; i++)
{
//比较大小、具体该怎么判断? 判断的对象无法确定,有可能是int类型也有可能是Person类
//所以,我们方法中 添加了第二个泛型委托变量, 委托中的方法实现了具体的比较并返回值。返回1则表明第一个数比第二个数大
if (dg(arr[i], max) == )
{
max = arr[i];
}
}
return max; //这个max有可能是int,也有可能是max
}
E, 更新按钮事件,使用我们刚写好的泛型委托:
private void BtnDelegateAsPara_Click(object sender, EventArgs e)
{
int[] arr = { , , , , , }; Person[] perArr = {
new Person {Name="甜馨",Age=},
new Person {Name="奥莉",Age=},
new Person {Name="嗯哼",Age=}
}; //我们用写好的委托方法再次实现 比较大小
dg_GetMax<int> dg_GetMaxInt = new dg_GetMax<int>(Compare);
int numMax = GetMax(arr, dg_GetMaxInt);
dg_GetMax<Person> dg_GetMaxPer = new dg_GetMax<Person>(Compare);
Person per = GetMax(perArr, dg_GetMaxPer); MessageBox.Show("最大数" + numMax.ToString());
MessageBox.Show(string.Format("最大年龄是:{0},年龄:{1}岁", per.Name, per.Age)); ////创建委托对象,并通过构造函数添加 GetMaxNum 方法
//dg_GetMaxAndMin objGetMax = new dg_GetMaxAndMin(GetMaxNum);
//int numMax = objGetMax(arr); //传递参数数组
////委托对象添加 GetMinNum 方法
//objGetMax += GetMinNum;
//int numMin = GetMinNum(arr); //传递参数数组 //MessageBox.Show("最大数" + numMax.ToString() + " 最小数" + numMin.ToString());
//MessageBox.Show(string.Format("最大年龄是:{0},年龄{1}岁", per.Name, per.Age)); }
F,运行查看效果:

--------【以上Demo下载】-----------
4,匿名方法介绍:
使用Delegate的时候,很多时候没有必要单独去定义好一个普通的方法(上面例子中的Compare方法),因为这个方法也就只有Delegate会用,而且只用一次,这时候就适合用匿名方法。
//三,Compare方法改写为委托方法 ,可以把Compare改写为匿名方法
//int numMax = GetMax<int>(arr, Compare);
int numMax = GetMax<int>(arr, delegate(int a, int b)
{
int num = ;
if (a > b)
{
num = ;
}
else if (a == b)
{
num = ;
}
else if (a < b)
{
num = -;
}
return num;
});
//Person per = GetMax<Person>(perArr, Compare);
Person per = GetMax<Person>(perArr, delegate(Person AA, Person BB)
{
int num = ;
if (AA.Age > BB.Age)
{
num = ;
}
else if (AA.Age == BB.Age)
{
num = ;
}
else if (AA.Age < BB.Age)
{
num = -;
}
return num;
});
5,委托Delegate编译以后就是一个类Class,这个类继承于 MulticastClass (多播委托)
我们用.NET Relfector反编译,查看委托实质:

--------- ------------
A, 委托的实质是类,继承于多播委托MulticastDelegate,MulticastDelegate继承于System.Delegate类

B, objSayHi += SayHiEN; 这段代码的实质是: 把两个集合中的方法都存放到一个集合中,然后返回
///创建委托对象,并通过构造函数添加SayHiCN方法
dg_SayHi objSayHi = new dg_SayHi(SayHiCN);
//向委托对象中 "添加" 一个方法
objSayHi += SayHiEN;
上面把方法添加到委托中,编译后就是下面的这样:

总结: 本文了解委托,并使用委托作为方法进行传递,又接触到了泛型和匿名方法,借助反编译器查看委托的实质。
参考:
返回委托中方法的值:http://m.blog.csdn.net/blog/lrz8745/7325056
http://www.cnblogs.com/linlf03/archive/2011/05/09/2041657.html
实用插件:Indent Guides 插件 代码中显示虚竖线

【.NET基础】--委托、事件、线程(1)的更多相关文章
- c#基础学习笔记-----------委托事件
这里有一个比较简单的委托实例应用(跨窗体操作控件) http://blog.csdn.net/bdstjk/article/details/7004035 还有一个比较详细的介绍并深入理解委托事件的讲 ...
- C# 基础 - 委托、事件
1. 委托 sequenceDiagram 方法->>委托: 返回值和入参一样 委托->>方法: 调用委托就是调用绑定的方法 delegate int NumTest(int ...
- [.net 面向对象编程基础] (22) 事件
[.net 面向对象编程基础] (22) 事件 事件(Event)是学习.net面向对象编程很重要的一部分,在学习事件之前,我们实际上已经在很多地方使用了事件,比如控件的click事件等,这些都是. ...
- C# ~ 从 委托事件 到 观察者模式 - Observer
委托和事件的部分基础知识可参见 C#/.NET 基础学习 之 [委托-事件] 部分: 参考 [1]. 初识事件 到 自定义事件: [2]. 从类型不安全的委托 到 类型安全的事件: [3]. 函数指针 ...
- 谈.Net委托与线程——解决窗体假死
转自:http://www.cnblogs.com/smartls/archive/2011/04/08/2008981.html#2457370 引言 在之前的<创建无阻塞的异步调用> ...
- C#基础知识回顾--线程传参
C#基础知识回顾--线程传参 在不传递参数情况下,一般大家都使用ThreadStart代理来连接执行函数,ThreadStart委托接收的函数不能有参数, 也不能有返回值.如果希望传递参数给执行函数, ...
- SQL Server 扩展事件(Extented Events)从入门到进阶(2)——在GUI中创建基础扩展事件
本文属于 SQL Server 扩展事件(Extented Events)从入门到进阶 系列 第一篇文章中提到了如何在Profiler中创建跟踪(trace),并以服务器端(server-side)跟 ...
- C#基础委托回顾
C#基础委托回顾 前言 快忘记了. 委托的特点 委托类似于 C++ 函数指针,但它们是类型安全的. 委托允许将方法作为参数进行传递. 委托可用于定义回调方法. 委托可以链接在一起:例如,可以对一个事件 ...
- C#基础之事件(2)
在“C#基础之事件(1)”中已对事件有了一个大概,这里对事件进行更深入的学习. 本节按以下内容展开: 1.事件拥有者与事件响应者的关系: 2.事件订阅的多种写法: 3.事件的订阅和取消订阅: 4.多事 ...
- C# 利用委托事件进行窗体间的传值(新手必看)
引言: 窗体间传值是每个学习WinForm新手的常见问题,最初级的方法就是 在窗体中先获取到要接受值窗体.然后通过.得到某个空间或者属性,直接赋值,这个需要接收放的窗体属性或者空间必须是public ...
随机推荐
- 使用 EasyMock 更轻松地进行测试
from:http://www.ibm.com/developerworks/cn/java/j-easymock.html 测试驱动开发是软件开发的重要部分.如果代码不进行测试,就是不可靠的.所有代 ...
- Tun/Tap interface tutorial
Foreword: please note that the code available here is only for demonstration purposes. If you want t ...
- 如何判断js中的数据类型(转)
如何判断js中的数据类型:typeof.instanceof. constructor. prototype方法比较 如何判断js中的类型呢,先举几个例子: var a = "iamstri ...
- GitHub 上排名前 100 的 Android 开源库进行简单的介绍
若有任何疑问可通过邮件或微博联系我 项目名称 项目简介 1. react-native 这个是 Facebook 在 React.js Conf 2015 大会上推出的基于 JavaScript 的开 ...
- Python学习入门基础教程(learning Python)--5.1 Python下文件处理基本过程
Python下的文件读写操作过程和其他高级语言如C语言的操作过程基本一致,都要经历以下几个基本过程. 1. 打开文件 首先是要打开文件,打开文件的主要目的是为了建立程序和文件之间的联系.按程序访问文件 ...
- 浅析WINFORM工具条的重用实现
一直以来,我都想看看别人家的工具栏重用(图1)到底是如何实现的,但在网上搜索了很久都没有找到过,即使找到一些程序,要么就是把这个工具栏写在具体的画面(图2),要么就是没有源代码的, 我在想,是否别人也 ...
- 用iframe设置代理解决ajax跨域请求问题
面对ajax跨域请求的问题,想用代理的方式来解决这个跨域问题.在服务器端创建一个静态的代理页面,在客户端用iframe调用这个代理 今天在项目中需要做远程数据加载并渲染页面,直到开发阶段才意识到aja ...
- CSS Sprites图片处理
简介: CSS Sprites是一个网页图片处理方式,在国内都叫CSS精灵,css Sprites允许你将一个页面涉及到的所有零星图片都包含到一张大图中去,这样一来,当访问该页面时,载入的图片就不会像 ...
- TP复习13
## ThinkPHP 3.1.2 模板的使用技巧#讲师:赵桐正微博:http://weibo.com/zhaotongzheng 本节课大纲:一.模板包含 <include file=&quo ...
- [AngularJS] Using Services in Angular Directives
Directives have dependencies too, and you can use dependency injection to provide services for your ...