C#进阶之路(二):事件
2021-02-20 02:18
一、初步了解事件
事件是委托的一个子集,为了满足“广播/订阅”模式的需求而生。
事件就是限制委托字段的包装器。限制外界对委托字段内部的访问。相当于封装。
事件就是能够发生的什么事情,主要有以下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 = 5288 }; //订阅事件 PriceChanged为事件,iphone6_PriceChanged是事件处理器 iphone6.PriceChanged += iphone6_PriceChanged; // 调整价格(事件发生) iphone6.Price = 3999; 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 EventHandlerPriceChanged; //注意使用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 = 3999; 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
public event EventHandler
如果不需要参数,可以直接使用EventHandler(不需要
最后,事件标准模式还需要写一个受保护的虚方法来触发事件,这个方法必须以On为前缀,加上事件名(PriceChanged),还要接受一个EventArgs参数,如下:
public class IPhone6 {
...
public event EventHandler
protected virtual void OnPriceChanged(PriceChangedEventArgs e) {
if (PriceChanged != null) PriceChanged(this, e);
}
...
}
三、总结
直接用一个看到的教程的图片来总结啦。本博就是这么懒,包括上面的案例也是参照另外一个教程上面的,哈哈。
上一篇:C#实现自己主动升级(附源代码)
下一篇:WinForm 文件操作