一、初步了解事件

  事件是委托的一个子集,为了满足“广播/订阅”模式的需求而生。

  事件就是限制委托字段的包装器。限制外界对委托字段内部的访问。相当于封装。

事件就是能够发生的什么事情,主要有以下5个主体。

1、事件的拥有者(event source,对象)

2、事件的成员(event,成员)

3、事件的响应者(event subscriber,对象)

4、事件处理器(event hander,成员)——本质上是一个回调方法

5、事件订阅——把事件处理器与事件关联在一起,本质上是一种以委托类型为基础的约定。

事件不会主动发生,一定是被拥有者内部逻辑触发,才会发生。

  1.1事件订阅的写法

A、比较常用的订阅写法

this.myButton.Click += new System.EventHandler(this.myButton_Click);

也可以像下面这么写

this.myButton.Click +=this.myButton_Click

eventhander就是事件处理器,只不过可以省略,直接写方法名。

+=为事件订阅操作符

+=后面的就是事件的处理器,格式就是 事件 +=(订阅)xxxx(事件处理器)

B、另一种订阅方式的写法

这种写法直接使用了匿名方法,也就是方法只针对本次事件使用,并不复用。

拉姆达写法

简写

事件只能在+=或者-=的左边,其他的时候无法调用事件。

  1.2事件处理器

下面的代码是不是处理器?

protected virtual void OnPriceChanged(PriceChangedEventArgs e) {
if (PriceChanged != null) PriceChanged(this, e);
}

结论:是的,OnPriceChanged就是事件处理器。另外,ElapsedEventArgs和EventArgs 都是事件处理器,一个是timer elapse的处理器另一个是click处理器,你可以自定义事件处理器,比如我上面举的例子,自定义的事件处理器生命要加Protected,以限制访问级别。

闪电就是事件,扳手是属性,方块是方法。

事件处理器(event hander,成员)——本质上是一个回调方法

注意:

A事件处理器是方法成员

B挂接事件处理器的时候,可以使用委托实例,也可以直接使用方法名,这是个语法糖

C事件处理器对事件的订阅不是随意的,匹配与否由声明事件时所使用的委托类型来检测

D事件可以同步调用也可以异步调用

  1.3事件本身的声明

简略的声明格式:

字段式声明,field-like

public event OrderEventHandler Order;

完整的声明格式

private OrderEventHandler orderEventHandler;
public event OrderEventHandler Order
{
add
{
this.orderEventHandler += value;
}
remove
{
this.orderEventHandler -= value;
}
}

  1.4事件声明的语法糖

事件委托的声明有语法糖,就是system类下面的eventhandler这个方法,他是厂商给我们定义好的一个语法糖。

如果使用这个语法糖,就可以不用再去声明委托事件了,如下就可以不写:

public delegate void OrderEventHandler(Customer customer,OrderEventArgs e);

二、事件的应用

  2.1事件基本声明模式

  这种方式跟委托的写法比较相像,只不过在声明一个委托对象时加上event关键字。然后委托方法的后缀要加上eventhandler或者handler,至于为什么加?

A加eventhandler为了让所有人明确是为了给事件服务的

B加eventhandler为了表明这个委托是用来约束事件处理器的

C 委托创建的实例是用来存储事件的

//声明(事件)委托类型有点像方法,但是不要写错地方,委托(事件委托)是一种类,小心写成嵌套类型。
public delegate void PriceChangedHandler(decimal oldPrice, decimal newPrice);
//事件拥有者 event source
public class IPhone6
{
decimal price;
//事件的成员
public event PriceChangedHandler PriceChanged;
public decimal Price
{
get { return price; }
set
{
if (price == value) return;
decimal oldPrice = price;
price = value;
// 如果调用列表不为空,则触发。
if (PriceChanged != null)
PriceChanged(oldPrice, price);
}
}
}
class BasicStyle
{
static void Main(string[] args)
{
IPhone6 iphone6 = new IPhone6() { Price = };
//订阅事件 PriceChanged为事件,iphone6_PriceChanged是事件处理器
iphone6.PriceChanged += iphone6_PriceChanged;
// 调整价格(事件发生)
iphone6.Price = ;
Console.ReadKey();
}
//事件处理器 console是事件响应者
static void iphone6_PriceChanged(decimal oldPrice, decimal price)
{
Console.WriteLine("年终大促销,iPhone 6 只卖 " + price + " 元, 原价 " + oldPrice + " 元,快来抢!");
}
}

运行结果:

  有人可能会问,如果把上面的event关键字拿掉,结果不是一样的吗,到底有何不同?

  没错可以用事件的地方就一定可以用委托代替。

  但事件有一系列规则和约束用以保证程序的安全可控,事件只有 += 和 -= 操作,这样订阅者只能有订阅或取消订阅操作,没有权限执行其它操作。如果是委托,那么订阅者就可以使用 = 来对委托对象重新赋值(其它订阅者全部被取消订阅),甚至将其设置为null,甚至订阅者还可以直接调用委托,这些都是很危险的操作,广播者就失去了独享控制权。

  2.2 事件标准声明模式

  .NET 框架为事件编程定义了一个标准模式。设定这个标准是为了让.NET框架和用户代码保持一致。System.EventArgs是标准模式的核心,它是一个没有任何成员,用于传递事件参数的基类。

  利用System.EventArgs加上Eventhandler这个语法糖,形成了标准模式,按照标准模式,我们对于上面的iPhone6示例进行重写:

//传入的事件参数,派生自EventArgs类。EventArgs本身是一个委托。
public class PriceChangedEventArgs : System.EventArgs
{
public readonly decimal OldPrice;
public readonly decimal NewPrice;
public PriceChangedEventArgs(decimal oldPrice, decimal newPrice)
{
OldPrice = oldPrice;
NewPrice = newPrice;
}
}
//事件拥有者 event source
public class IPhone6
{
decimal price;
//使用了EventHandler,PriceChanged是事件本身,PriceChangedEventArgs是事件委托
public event EventHandler<PriceChangedEventArgs> PriceChanged;
//注意使用protected,事件处理器(其实事件委托和ONxxx都可以理解为事件处理器,都是在处理这个事件)
protected virtual void OnPriceChanged(PriceChangedEventArgs e)
{
if (PriceChanged != null) PriceChanged(this, e);
}
public decimal Price
{
get { return price; }
set
{
if (price == value) return;
decimal oldPrice = price;
price = value;
// 如果调用列表不为空,则触发。
if (PriceChanged != null)
OnPriceChanged(new PriceChangedEventArgs(oldPrice, price));
}
}
}
class StandaryStyle
{
static void Main(string[] args)
{
IPhone6 iphone6 = new IPhone6() { Price = 5288M };
//订阅事件 PriceChanged为事件,iphone6_PriceChanged是事件处理器
iphone6.PriceChanged += iphone6_PriceChanged;
// 调整价格(事件发生)
iphone6.Price = ;
Console.ReadKey();
}
//事件处理器 console是事件响应者
static void iphone6_PriceChanged(object sender, PriceChangedEventArgs e)
{
Console.WriteLine("年终大促销,iPhone 6 只卖 " + e.NewPrice + " 元, 原价 " + e.OldPrice + " 元,快来抢!");
}
}

运行结果:

案例中

static void iphone6_PriceChanged(object sender, PriceChangedEventArgs e)

sender参数只是传递了指向引发事件的那个类的实例的一个引用(也就是event source)

PriceChangedEventArgs e 这个很好理解,就是我们上面定义的那个PriceChangedEventArgs 类。

  为事件定义委托,它的名称通用约定以EventHandler结尾。

  由于考虑到每个事件都要定义自己的委托很麻烦,.NET 框架为我们预定义好一个通用委托System.EventHandler<TEventArgs>:(也就是前文说的语法糖)

public event EventHandler<PriceChangedEventArgs> PriceChanged;

如果不需要参数,可以直接使用EventHandler(不需要<TEventArgs>)。

最后,事件标准模式还需要写一个受保护的虚方法来触发事件,这个方法必须以On为前缀,加上事件名(PriceChanged),还要接受一个EventArgs参数,如下:

public class IPhone6 {

...

public event EventHandler<PriceChangedEventArgs> PriceChanged;

protected virtual void OnPriceChanged(PriceChangedEventArgs e) {

if (PriceChanged != null) PriceChanged(this, e);

}

...

}

三、总结

直接用一个看到的教程的图片来总结啦。本博就是这么懒,包括上面的案例也是参照另外一个教程上面的,哈哈。

最新文章

  1. Linux学习笔记(9)-守护进程
  2. hdu 1002
  3. iOS 被拒问题及原因 - IDFA问题
  4. 拓扑编号 vijos1790
  5. ARM compiler No such file or directory
  6. 流程引擎Activiti系列:如何将kft-activiti-demo-no-maven改用mysql数据库
  7. apache开源项目--Synapse
  8. poj 2503 快排+二分
  9. linux日志审计项目案例实战(生产环境日志审计项目解决方案)
  10. bzoj3667: Rabin-Miller算法
  11. Android PackageManager基础知识
  12. PendingIntent.getBroadcast第四个参数flags
  13. TypeError: Cannot read property &amp;#39;style&amp;#39; of null 错误解决
  14. USACO 1.3 Wormholes
  15. mysql日志详细解析【转载】
  16. sqlserver智能提示插件-sql prompt(9.4.6)的安装及注册流程
  17. Cache架构设计
  18. LeetCode OJ 24. Swap Nodes in Pairs
  19. Django(框架、模板)
  20. 【React 资料备份】React v16.3之后的生命周期

热门文章

  1. Git版本控制系统VCS
  2. public,protected,privat区别
  3. C++ IPv4与IPv6的兼容编码(转,出自http://blog.csdn.net/ligt0610/article/details/18667595)
  4. rsh命令配置于使用
  5. 【HackerRank】 Find Digits
  6. freeswitch中集成使用ekho实现TTS功能一
  7. Linux进程中TIME_OUT解析
  8. 标准输出:1&gt;,2&gt;,1&gt;&amp;2,2&gt;&amp;1
  9. 12.常见模块time、json模块
  10. 在Linux系统下使用Docker以及Weave搭建Nginx反向代理