C#知识体系(二)用案例来理解委托与事件
上一篇博客讲到了LinQ和lambda的常用方法 还有很多我们未知但c#设计团队已经为我们封装好的类和方法。随着我们不断的熟悉C#语言,渐渐的就会接触到其他的知识点,委托、事件、反射、线程、同步,异步、IO、套接字。。。这些东西我们平常用到的不多,都是些概念性的东西,也许是因为不熟悉而可以回避了使用这些东西,不可否认的是 就算不用这些我们依然能想到问题的解决办法。但是几乎所有语言都会有这些概念,因为在某些场景它们能发挥不可思议的能力。
其实我到现在还是没有掌握委托和事件,在工作或者设计中也尽量回避使用,但如果想走技术这条路,这块硬骨头必须趁早啃了它。我们一直在疑惑:为什么要用委托 很想问别人 什么场景下用到委托 反正我是没有得到我想要的答案 所以也不能深刻的体会它的妙用。本文将讲诉普通的委托,事件的委托,有参 无参 有返回值的 以上都是同步的委托,异步的到后面的异步和套接字再讲。
基本语法
//无返回值 public delegate void delegateName() ;//无参 无返回值 public delegate void delegateName(int a,int b) ;//2个参数 无返回值 public delegate void delegateName(object o) ;//任意类型 无返回值 public delegate void delegateName(object o,EventArgs args) ;//任意类型,第2个参数为一个事件对象或者基础了该事件对象的子类 无返回值
//有返回值 public delegate bool delegateName() ;//无参 有返回值 public delegate bool delegateName(object) ;//与参 有返回值 这个是不是有点熟悉呢 它就是我们常用的userlist.where()方法 要求传参的委托 返回布尔 带一个object参数或多个原型如下 Func<TSource, bool> predicate 最后一个参数为返回值 public delegate object delegateName(int a,int b) ;//2个参数 有返回值 public delegate List<object> delegateName(object o) ;//任意类型 有返回值 public delegate bool delegateName(object o,EventArgs args) ;//任意类型,第2个参数为一个事件对象或者基础了该事件对象的子类 有返回值
基本使用大家可能都略知一二,那就来点案例加深理解,来个大家熟悉的场景 QQ空间。
案例分析
场景如下:
QQ空间发布说说大家都常用,每当我们更新了说说或者日志,好友进空间的时候总能知道是谁更新了说说。但他是怎么知道你更新了的呢?
分析:
实现1:在每个用户所属的表里面存有每个好友的上一次更新的说说编号,当进空间初始化时根据每个编号去和好友空间最新的编号对比,获取大于当前编号的日志,如果好友上百估计很慢
实现2:用户每次发布说说时,遍历他所有好友,主动插入一条说说编号到他好友说说更新表里面或者缓存表中,当好友进空间时 根据未读说说状态的编号去查询加载出来
...
实现方式多种多样,具体用的什么方法我也猜不出来 如果有熟悉的或者来自鹅厂的大神请给我们解惑!
接下来的案例代码是以第二种方式来实现观察者模式的(发布者与订阅者)
功能分析
在这个场景中,有当前QQ用户发布说说(发布者),他所有的好友(订阅者),还有说说(日志)实体
这里面会出现3个对象
User QQ Log
public class User
{
public User() { }
public event PubLogHanler Handler;
public User(int id,QQ qq)
{
ID = id;
QQ = qq;
}
public int ID { get; set; }
//用户的QQ,里面包含好友列表等信息
public QQ QQ { get; set; }
//发说说(用户方法)
public void PubLog(Log log)
{
Console.WriteLine("发布说说中...");
StringBuilder sb = new StringBuilder();
sb.AppendLine("====================================");
sb.AppendLine(log.Title);
sb.AppendLine(" "+log.Content);
sb.Append("\t"+log.Author.ToString()+"\t");
sb.AppendLine(log.PubTime);
sb.AppendLine("====================================");
Console.WriteLine(sb.ToString());
//触发事件
OnPubLog(this, new PubLogHanlerArgs(log));
}
public void OnPubLog(object o, PubLogHanlerArgs args)
{
if (Handler != null)
{
Handler(o, args);
}
}
}
然后是QQ实体对象,它包含好友列表等信息 这里只列举了好友列表
public class QQ
{
public QQ(){}
public QQ(int id,List<User>list)
{
ID = id;
FriendList = list;
}
//QQ号
public int ID { get; set; }
//好友列表
public List<User> FriendList { get; set; }
}
接下来是说说实体对象
/// <summary>
/// 说说类
/// </summary>
public class Log
{
public Log() { }
public Log(string title, string content, int author, string time)
{
Title = title;
Content = content;
Author = author;
PubTime = time;
}
public string Title { get; set; }
public string Content { get; set; }
//(QQID)作者
public int Author { get; set; }
public string PubTime { get; set; }
}
调用类里面弄点初始化数据
User user1 = , , new List<User>() { }));
User user2 = , , new List<User>() { }));
User user3 = , , new List<User>() { }));
//new 一个用户 拥有一个QQ号和3个好友
User currUser = , , new List<User>() { user1, user2, user3 }));
如果是普通的 肯定就是直接发说说了
//发说说
currUser.PubLog(new Log("我的第一个说说","说说开通了可以分享咯",currUser.QQ.ID,DateTime.Now.ToString("M.d HH:mm:ss")));
//循环遍历所有好友
foreach (var item in currUser.QQ.FriendList)
{
////推送 存储日志编号到好友的待加载列表里面
}

这样其实也算是实现功能了,如果要求不高那就提交代码可以领盒饭了。
好吧 到现在为止也没委托啥事呀!!!
委托的定义和快递的职责是一样一样的:发件人委托给快递,快递再投送给接收者。你说其实他也可以直接给接收者,但他为什么要委托快递呢 懒!就是懒 哈哈 这样理解也是奇葩了 !
你让我说说为啥委托非用不可 那我还真答不上来,就像我们也可以不用快递一样,只不过代价就大了,可以想象那场景,网上卖件衣服都要送出国再自己回来,那酸爽。。。
既然机智的选择用快递,那我们就定义一个送快递的吧
public delegate void delegateEMS(User o, Log log);
//再定义一个快递送快递的方法
public static void PubEms(User user,Log log)
{
foreach (var item in user.QQ.FriendList)
{
Console.WriteLine("已推送说说标题为: " + log.Title + " 的说说给:" + item.QQ.ID);
//投送 发往指定人
}
}
然后就是调用
User user1 = , , new List<User>() { }));
User user2 = , , new List<User>() { }));
User user3 = , , new List<User>() { }));
//new 一个用户 拥有一个QQ号和3个好友
User currUser = , , new List<User>() { user1, user2, user3 }));
Log temp = new Log("我的第一个说说", "说说开通了可以分享咯", currUser.QQ.ID, DateTime.Now.ToString("M.d HH:mm:ss"));
//发说说
currUser.PubLog(temp);
delegateEMS ems = new delegateEMS(PubEms);
ems(currUser, temp);
然后就没有然后了 等着收货的给差评吧!

好吧说跑题了,继续回来说说QQ空间的事,刚才讲快递是为了加深大家对于委托的理解,我们用到了普通的委托有参无返回 但是发现每发一篇博客就要多写一次 大量重复的代码 我们接下来改为事件+委托
委托的进阶使用 事件+委托
事件可以说是无处不在 鼠标点击事件 移动事件就不说了 之所以用到事件就是为了实现触发器的效果 删除一条记录就邮件通知相关负责人 出现多少个警告也会通知 在我们项目中比较常用
基本语法
public event PubLogHanler Handler; //这里事件类型是自定义的
//定义推送委托
public delegate void PubLogHanler(object o, PubLogHanlerArgs args);
//发布日志触发
public class PubLogHanlerArgs : EventArgs
{
public PubLogHanlerArgs(){}
public PubLogHanlerArgs(Log log)
{
LogInfo = log;
}
public Log LogInfo { get; set; }
}
这里自定义了一个说说发布的事件,它继承EventArgs类,是任何事件的基类,触发一个事件时会记录下说说的信息
public void PubLog(Log log)
{
Console.WriteLine("发布说说中...");
StringBuilder sb = new StringBuilder();
sb.AppendLine("====================================");
sb.AppendLine(log.Title);
sb.AppendLine(" "+log.Content);
sb.Append("\t"+log.Author.ToString()+"\t");
sb.AppendLine(log.PubTime);
sb.AppendLine("====================================");
Console.WriteLine(sb.ToString());
//触发事件
OnPubLog(this, new PubLogHanlerArgs(log));
}
public void OnPubLog(object o, PubLogHanlerArgs args)
{
if (Handler != null)
{
Handler(o, args);
}
}
从上面的代码可以看到 调用发布说说方法后会触发事件,然后事件会去执行推送方法达到触发器的效果
//推送事件
static void currUser_Handler(object o, PubLogHanlerArgs args)
{
User user = o as User;
if(user!=null)
{
foreach (var item in user.QQ.FriendList)
{
Console.WriteLine("已推送说说标题为: "+args.LogInfo.Title+" 的说说给:"+item.QQ.ID);
//推送 存储日志编号到好友的待加载列表里面
}
}
}
其实上面只是个方法,只是参数看起来是事件罢了,接下来就是绑定事件
//推送给好友
currUser.Handler += new PubLogHanler(currUser_Handler);
//发说说
currUser.PubLog(new Log("我的第一个说说", "说说开通了可以分享咯", currUser.QQ.ID, DateTime.Now.ToString("M.d HH:mm:ss")));
currUser.PubLog(new Log("终于上博客园首页啦", "博客园是个学习的好地方", currUser.QQ.ID, DateTime.Now.ToString("M.d HH:mm:ss")));

事件的妙处就在于可以绑定多个事件 ,叫多播委托。再定义一个事件
//已通知更新缓存
static void UpdateCache_Handler(object o, EventArgs args)
{
User user = o as User;
if (user != null)
{
Console.WriteLine("\n<*>已通知更新缓存<*>\n");
//已通知更新缓存
}
}
调用代码如下
User user1 = , , new List<User>() { }));
User user2 = , , new List<User>() { }));
User user3 = , , new List<User>() { }));
//new 一个用户 拥有一个QQ号和3个好友
User currUser = , , new List<User>() { user1, user2, user3 }));
//推送给好友
currUser.Handler += new PubLogHanler(currUser_Handler);
//通知更新缓存
currUser.Handler += new PubLogHanler(UpdateCache_Handler);
//发说说
currUser.PubLog(new Log("我的第一个说说", "说说开通了可以分享咯", currUser.QQ.ID, DateTime.Now.ToString("M.d HH:mm:ss")));
currUser.PubLog(new Log("终于上博客园首页啦", "博客园是个学习的好地方", currUser.QQ.ID, DateTime.Now.ToString("M.d HH:mm:ss")));
Console.ReadKey();

总结
委托和事件讲到这里也接近尾声了,我这也是临时抱佛脚,在网上看了写实例代码 自己实操了一遍就根据自己的理解发上来了。很多地方估计语句不是太通顺甚至概论都没理明白,如果有错误之处还望大家指正,免得误人子弟!
如果有需要源代码的话我稍后会发上来的。下一篇博客 我会讲一讲反射和线程,并且把我写的一个xml json datatable List 相互转换 的dll和代码分享给大家,请大家关注我的博客http://www.cnblogs.com/jingch/
源码
C#知识体系(二)用案例来理解委托与事件的更多相关文章
- 【详细】【转】C#中理解委托和事件 事件的本质其实就是委托 RabbitMQ英汉互翼(一),RabbitMQ, RabbitMQ教程, RabbitMQ入门
[详细][转]C#中理解委托和事件 文章是很基础,但很实用,看了这篇文章,让我一下回到了2016年刚刚学委托的时候,故转之! 1.委托 委托类似于C++中的函数指针(一个指向内存位置的指针).委托 ...
- C#学习之初步理解委托、事件、匿名方法和Lambda
最经在学习LinqtoSql,然后扯到Lambda表达式,然后扯到匿名方法,然后扯到委托,最后扯到事件处理...后来发现对委托这个概念和事件处理这个过程理解得不是很清晰,遂得一下学习笔记.那里说得不对 ...
- 【前端知识体系-JS相关】深入理解JavaScript异步和单线程
1. 为什么JavaScript是单线程? JavaScript语言的一大特点就是单线程,也就是说,同一个时间只能做一件事.那么,为什么JavaScript不能有多个线程呢?这样能提高效率啊. Jav ...
- 【前端知识体系-JS相关】深入理解MVVM和VUE
1. v-bind和v-model的区别? v-bind用来绑定数据和属性以及表达式,缩写为':' v-model使用在表单中,实现双向数据绑定的,在表单元素外使用不起作用 2. Vue 中三要素的是 ...
- 【前端知识体系-JS相关】深入理解JavaScript原型(继承)和原型链
1. Javascript继承 1.1 原型链继承 function Parent() { this.name = 'zhangsan'; this.children = ['A', 'B', 'C' ...
- C#基础篇 - 理解委托和事件
1.委托 委托类似于C++中的函数指针(一个指向内存位置的指针).委托是C#中类型安全的,可以订阅一个或多个具有相同签名方法的函数指针.简单理解,委托是一种可以把函数当做参数传递的类型.很多情况下,某 ...
- 【详细】【转】C#中理解委托和事件
文章是很基础,但很实用,看了这篇文章,让我一下回到了2016年刚刚学委托的时候,故转之! 1.委托 委托类似于C++中的函数指针(一个指向内存位置的指针).委托是C#中类型安全的,可以订阅一个或多个具 ...
- linux 运维知识体系
这里将会介绍一下,LINUX运维工程师的知识体系. 只能说是个人理解吧.并不是必要或者充分的,仅供网友参考. 大部分本博客都有涉及,并不完整. 1.LINUX运维基础 1.1.LINUX系统的简介,分 ...
- c#关于委托和事件(二)(介绍的很详细)
using System;using System.Collections.Generic;using System.Text; namespace Delegate { // 热水器 p ...
随机推荐
- sq楼盘信息
Sq楼盘信息 1.1按关注度来分 前10个 序号 楼盘 网址 价格 关注人数 地址 附件社区 1 上海公馆 http://shangqiu.jiwu.com/loupan/236459.html 38 ...
- innoDB 存储引擎
innodb 是在mysql 5.5.8 及之后的版本中成为mysql的默认存储引擎.之前都使用myisam. innodb 是事务型的存储引擎 支持ACID事务,适用于小事务. 1.表空间类 ...
- sql server 脚本创建数据库和表
USE [master] GO IF EXISTS(SELECT 1 FROM sysdatabases WHERE NAME=N'HkTemp') BEGIN DROP DATABASE HkTem ...
- react参考项目001
https://github.com/chen2009277025/webpack-ant-design-demo https://github.com/cobish/cobish.github.io ...
- JavaWeb AJAX
1. Asynchronized JavaScript And XML 异步JavaScript和XML,它并不是一门新的语言或技术,实际是几项技术按一定的方式组合在一起共同的协作中发挥各自的作用, ...
- Javascript中String()与new String()的差异
这里主要关注的是值类型和引用类型. 我们知道在javascript中的变量在内存中的存储有两种形式,值类型存储和引用类型存储. 通常可以进行值存储的包括 字符串类型,布尔值类型,数字类型,他们都包含 ...
- 《Pro Express.js》学习笔记——概述
要学Node.js,先学Express.js. Express.js是Node.js官方推荐的基础框架. Express.js框架经过一系列的发展,已经到了4.x版本.新的版本解决了3.x之前版本的依 ...
- Also unsere eigene Christian Louboutin Webshop bietet die überragende Christian Louboutin Schuhe uk schiebt zusammen mit kostengünstigen Wert
www.heelschuhe.de, Es ist wirklich eine der Frauen erfordern immer interessant und auch Louboutin Pu ...
- 一个有趣的回答(摘自http://www.51testing.com/html/03/n-860703.html)
假设这有一个各种字母组成的字符串,假设这还有另外一个字符串,而且这个字符串里的字母数相对少一些.从算法上讲,什么方法能最快的查出所有小字符串里的字母在大字符串里都有? 比如,如果是下面两个字符串: S ...
- GDB中汇编调试
GDB中汇编调试 1.输入代码 2.使用gcc - g example.c -o example -m32指令在64位的机器上产生32位汇编,时遇到问题使用-m32指令报错,参考卢肖明同学博客知道这是 ...