C#事件
想起來這個月還沒寫博客,隨便混更一下吧
事件
事件(event)指的是一些能夠發生的事情,它是一種使物件或者類能夠提供通知的成員,舉個例子,你打開了手機,點擊了一下“資訊”,然后就會彈出資訊的視窗,“點擊”這個動作就是一個事件,它向CPU傳達了一個訊息,打開資訊的視窗,然后CPU處理這個資訊,打開了“資訊”的視窗,
經過這番解釋,你應該大致知道事件這個模型的組成了:事件擁有者(event source)、事件本身(event)、事件回應者(event subscriber)、事件處理器(event handler)和事件訂閱,事件擁有者,顧名思義,就是這個事件是誰的,事件本身同理;事件回應者是指接收資訊的一方;事件處理器是對事件做出回應,一般是一個方法,而且是一個回呼方法;事件訂閱就是把事件處理器與事件關聯在一起,如果訂閱了這個事件,當它發生時就會呼叫事件處理器的方法,
其實事件在手機應用開發當中十分常見,不過Java里面沒有事件這種成員,Java使用介面來實作事件,
使用事件
在放代碼之前,先要添加一個參考,我們要用到一個新的程式集,具體方法為:對著參考右鍵,點擊添加參考,在程式集里面搜索System.Windows.Forms,勾選添加,

然后再來看例子:
using System;
using System.Windows.Forms;
namespace EventPractice
{
class Program
{
static void Main(string[] args)
{
MyForm form = new MyForm();
form.ShowDialog();
}
}
class MyForm : Form //繼承,MyForm這個類可以使用System.Windows.Forms.Form里面public和protected修飾的成員
{
private TextBox textBox; //TextBox和Button都是System.Windows.Forms里面的類
private Button button;
public MyForm() //建構式,初始化成員
{
this.textBox = new TextBox();
this.button = new Button();
this.Controls.Add(this.button); //添加控制元件
this.Controls.Add(this.textBox);
this.button.Click += this.ButtonClicked; //Click是一個在System.Windows.Forms里面定義的事件,button是事件的擁有者,MyForm的物件form是事件的回應者
//+=運算子訂閱該事件,ButtonClicked是事件處理器,在下文定義,但是記得不要在ButtonClicked后面加括號
this.button.Text = "Say Hello!";
this.button.Top = 50;
}
private void ButtonClicked(object sender, EventArgs e) //事件處理器,接收兩個引數:sender是事件源,e是事件引數,這是C#自動生成的事件處理器
{ //至于為什么這么寫,是因為它隱藏了一個“約定”——委托(delegate)
this.textBox.Text = "Hello World!"; //點擊按鈕會出現這句話
}
}
}
先看看跑出來的結果吧:

這個例子用到了C#的WinForm,因為它比較容易解釋事件,看MyForm這個類里面,有一個叫Click的事件,它也十分直觀,就是在點擊的時候做出一些反應,我把它掛在了button上,同時訂閱ButtonClicked這個事件處理器,看下文對ButtonClicked的定義,當事件發生時,它會在文本框(textbox)里面列印這句話,然后再回到Main函式里面,new一個MyForm的物件,然后呼叫showDialog,彈出視窗,當我們執行這個程式的時候,會出現控制臺和這個Form,我們點擊Say Hello就會在文本框里面看見Hello World,
回過頭來想想,事件的五要素齊了嗎?
- 事件擁有者——
button - 事件——
Click - 事件回應者——
form - 事件處理器——
ButtonClicked - 事件訂閱——
+=
很完整,五要素都齊全,這是事件的回應者用自己的方法訂閱著自己的欄位成員的某個事件的模型,當然你也可以寫成其他的方式,留給讀者自己鉆研吧,
但是其實還有一個問題,為什么事件處理器ButtonClicked要接收兩個莫名其妙的引數?其實,舊版本的訂閱事件是這樣寫的:
this.button.Click += new EventHandler(this.ButtonClicked);
把滑鼠移到EventHandler上面,你會發現這竟然是一個委托,這說明了什么,我們用到的事件處理器,必須符合事件要求的一套規矩——符合EventHandler的規矩,接收兩個引數,一個object型別的引數,一個EventArgs的引數,所以看回ButtonClicked,它要接收那兩個引數,是因為委托的要求,但是現在基本不會用舊版本的訂閱寫法了,盡量都用上面例子的寫法,
回憶一下我在C#方法Extra中提到的lambda運算式,在這里能用嗎?當然可以,特別是ButtonClicked這種簡單的函式,lambda運算式是最喜歡的,使用方法為:
this.button.Click += (sender, e) =>
{
this.textBox.Text = "Hello World!";
};
希望你還記得lambda運算式的語法,這幾行代碼直接省去了宣告和定義ButtonClicked的功夫,省時省力,
自定義一個事件
如果C#提供給你的事件不適合當前的場景,那么我們可以自己定義一個事件,具體如下:
using System;
using System.Threading;
namespace EventDevelop
{
class Program
{
static void Main(string[] args)
{
Customer customer = new Customer(); //事件擁有者
Waiter waiter = new Waiter(); //事件回應者
customer.Order += Waiter.WaiterAction; //訂閱事件
customer.CustomerAction();
customer.PayTheBill();
Console.ReadLine();
}
}
public delegate void OrderEventHandler(Customer customer, OrderEventArgs e); //先宣告一個委托,作為事件訂閱的規定
public class OrderEventArgs : EventArgs //事件引數,繼承自EventArgs,有兩個屬性
{
public string DishName { get; set; }
public string Size { get; set; }
}
public class Customer //事件源,發送資訊的一方
{
private OrderEventHandler eventHandler;
public event OrderEventHandler Order //宣告事件,用OrderEventHandler約束
{
add //添加事件處理器
{
this.eventHandler += value; //value是傳進來的處理器
}
remove //移除事件處理器
{
this.eventHandler -= value; //移除用-=
}
}
public double Bill { get; set; }
public void PayTheBill()
{
Console.WriteLine("I will pay ${0}.", this.Bill);
}
public void WalkIn()
{
Console.WriteLine("Walk into the restaurant.");
Thread.Sleep(1000); //Sleep是讓控制臺等你一段時間,括號里面是毫秒數,1000毫秒就是1秒
}
public void SitDown()
{
Console.WriteLine("Sit down.");
Thread.Sleep(1000);
}
public void Think()
{
for (int i = 0; i < 5; i++)
{
Console.WriteLine("Let me think...");
Thread.Sleep(1000);
}
if (this.eventHandler != null) //呼叫之前判空,防止拋出例外
{ //有事件訂閱eventHandler就不為空
OrderEventArgs e = new OrderEventArgs();
e.DishName = "Kongpao Chiken";
e.Size = "large";
this.eventHandler.Invoke(this, e);
}
}
public void CustomerAction()
{
Console.ReadLine();
this.WalkIn();
this.SitDown();
this.Think();
}
}
public class Waiter
{
public static void WaiterAction(Customer sender, OrderEventArgs e) //事件處理器,根據委托規定,輸入兩個引數
{
Console.WriteLine("I will serve you the dish {0}.", e.DishName);
Thread.Sleep(1000);
double price = 10;
switch (e.Size)
{
case "small":
price = price * 0.5;
break;
case "large":
price = price * 1.5;
break;
default:
break;
}
sender.Bill += price;
}
}
}
這是完整的事件宣告,在宣告事件之前,你需要準備一個“規矩”,也就是先宣告一個委托,約束事件處理器,同時委托的宣告伴隨事件源和事件引數的宣告,自己宣告的OrderEventArgs一般要繼承Microsoft給你的EventArgs,
應該有人留意到我上次寫委托的時候,沒有給出宣告委托的語法,因為我想事件一定用得到(真不是忘了),宣告委托就在第22行,格式為:訪問修飾符+delegate+回傳值+名字識別符號+引數表,很簡單的,
最終輸出結果:

總結
事件一般不用自己宣告,Microsoft給你的一般都夠用了,事件的應用范圍真的很廣,就像你雙擊檔案夾就會打開,這就是一個事件,事件本事建立在委托的基礎上,實作一系列的功能,沒有委托就沒有事件,我當初說事件處理器是一個回呼方法,現在留給讀者自己細品一下吧,
后記
沒想到事件要寫這么復雜= =,有點出乎意料,不過它本身確實挺綜合的,接下來復習考試了,1月有空再想想水些別的吧,社團那邊好起來了,作業室能拿到是沒想到的,再接再厲吧,我月底才更博,就是2077的問題!
轉載請註明出處,本文鏈接:https://www.uj5u.com/net/241660.html
標籤:.NET技术
