|
如果我們想在一個(gè)窗體中訪問(wèn)另一個(gè)窗體中自定義的成員,必須把該成員的可見(jiàn)性設(shè)置為Public或者通過(guò)屬性公開(kāi),通過(guò)屬性公開(kāi)的話還說(shuō)得過(guò)去,但如果把可見(jiàn)性設(shè)置成Public的,這樣做就無(wú)可避免的破壞了類(lèi)型封裝性的原則,而這一做法也是我們?cè)?NET下開(kāi)發(fā)相當(dāng)樂(lè)意做的,特別是對(duì)于初次接觸.NET的開(kāi)發(fā)人員,實(shí)現(xiàn)訪問(wèn)另一類(lèi)型中成員的話最先想到的就是把該成員的可見(jiàn)性設(shè)置為Public,當(dāng)然這樣做算不上是錯(cuò)誤,但把這一做法作為自己的首要靈感,至少?gòu)拿嫦驅(qū)ο蟮慕嵌瘸霭l(fā)顯然是不合適的。
在.NET下,還為我們提供了另外一種強(qiáng)大的機(jī)制來(lái)實(shí)現(xiàn)窗體通信,這就是委托。委托可理解為一種類(lèi)型安全的函數(shù)指針,.NET下的事件的實(shí)現(xiàn)都是以委托做為基礎(chǔ)的。關(guān)于委托在這篇文章中我就不詳細(xì)介紹了,后邊會(huì)有文章專(zhuān)門(mén)介紹這一概念。 在此我演示通過(guò)在一個(gè)窗體里向另外一個(gè)窗體里的ListBox控件添加Item項(xiàng)來(lái)說(shuō)明這一方法。因此需要兩個(gè)窗體,一個(gè)MainFrm窗體,一個(gè)ChildFrm窗體,另外還需要一個(gè)Middle類(lèi),作為MainFrm和ChildFrm之間通信的橋梁。我也將給出VB.NET和C#兩種語(yǔ)言的代碼,以便大家可以做一下比較。
首先是MainFrm窗體,在MainFrm窗體中,拖一個(gè)ListBox控件即可,MainFrm.vb的代碼如下(為簡(jiǎn)單起見(jiàn),在此省去自動(dòng)生成的代碼):
Public Class Form3
Private Sub Form3_Load(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles MyBase.Load
AddHandler Middle.SendMessage, AddressOf DoMethod
End Sub
Private Sub DoMethod(ByVal getstr As String)
Me.ListBox1.Items.Add(getstr)
End Sub
End Class
再看ChildFrm窗體,在其中拖一個(gè)TextBox和一個(gè)Button控件,通過(guò)在TextBox中輸入值后,按Button按鈕向MainFrm窗體的ListBox控件中添加Item項(xiàng)。
Public Class Form2
Private Sub Button1_Click_1(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles Button1.Click
Middle.DoSendMessage(TextBox1.Text)
TextBox1.Text = ""
TextBox1.Focus()
End Sub
End Class
最后看Middle類(lèi):
Public Class Middle
Public Shared Event SendMessage(ByVal str As String)
Public Shared Sub DoSendMessage(ByVal str As String)
RaiseEvent SendMessage(str)
End Sub
End Class
為了更好的演示MainFrm和ChildFrm之間的獨(dú)立性,修改一下Application.Designer.vb的代碼:
<Global.System.Diagnostics.DebuggerStepThroughAttribute()>
Protected Overrides Sub OnCreateMainForm()
Me.MainForm = Global.WindowsApplication3.MainFrm
ChildFrm.show()
End Sub
好了,代碼完了,是不是很簡(jiǎn)單?通過(guò)上面的代碼可以看出來(lái),通過(guò)Middle類(lèi),MainFrm和ChildFrm都和Middle類(lèi)通信,它們之間除了參數(shù)的耦合外,已不再引用彼此的內(nèi)部成員,這樣就顯得更加獨(dú)立了。
下面是對(duì)應(yīng)的C#代碼,MainFrm.cs:
public partial class MainFrm: Form
{
private void MainFrm _Load(object sender, EventArgs e)
{
Middle.sendEvent += new Middle.SendMessage(this.DoMethod);
}
public void DoMethod(string getstr)
{
listBox1.Items.Add(getstr);
}
}
ChildFrm.cs:
public partial class ChildFrm: Form
{
public ChildFrm ()
{
InitializeComponent();
}
private void button1_Click(object sender, EventArgs e)
{
Middle.DoSendMessage(this.textBox1.Text);
textBox1.Text = "";
textBox1.Focus();
}
}
Middle.cs:
public static class Middle
{
public delegate void SendMessage(string str);
public static event SendMessage sendEvent;
public static void DoSendMessage(string str)
{
sendEvent(str);
}
}
同樣我們修改一下Program.cs的代碼:
static class Program
{
[STAThread]
static void Main()
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
// Application.Run(new Form1());
Form1 mainFrm = new Form1();
childFrm secondFrm = new childFrm();
secondFrm.Show();
Application.Run(mainFrm);
}
}
比較上面的VB.NET和C#代碼,我們可以看出VB.NET允許直接用Event關(guān)鍵字聲明事件,而C#則必須由我們自己首先聲明事件的委托原型,然后再基于該委托聲明事件,從這點(diǎn)看來(lái)VB.NET顯得更簡(jiǎn)潔,其實(shí)VB.NET編譯器在背后會(huì)自動(dòng)的為我們定義一個(gè)委托對(duì)象,而且該委托與C#代碼聲明的委托所生成IL代碼是一樣的,這點(diǎn)大家可以通過(guò)Ildasm中間代碼查看器來(lái)查看一下。引發(fā)事件,VB.NET是通過(guò)RaiseEvent關(guān)鍵字加上事件名稱,而C#則是通過(guò)直接使用事件名稱;最后是綁定事件的代碼,VB.NET是通過(guò)AddHandler關(guān)鍵字,C#通過(guò)重載的+=操作符,對(duì)于以上兩點(diǎn),編譯器同樣會(huì)為我們生成一致的IL代碼。
當(dāng)然,上面的例子比較簡(jiǎn)單,不過(guò)我們完全可以通過(guò)委托實(shí)現(xiàn)復(fù)雜的窗體通信,比如可以傳遞復(fù)雜的數(shù)據(jù)類(lèi)型,同時(shí),可以在設(shè)計(jì)結(jié)構(gòu)更加良好的中間通信類(lèi)。但也要提醒大家,不要?jiǎng)硬粍?dòng)就要用委托,它會(huì)增加程序的復(fù)雜性,應(yīng)該根據(jù)自己的需求考慮用何種方法。
AspNet技術(shù):在.NET中利用委托實(shí)現(xiàn)窗體間通信,轉(zhuǎn)載需保留來(lái)源!
鄭重聲明:本文版權(quán)歸原作者所有,轉(zhuǎn)載文章僅為傳播更多信息之目的,如作者信息標(biāo)記有誤,請(qǐng)第一時(shí)間聯(lián)系我們修改或刪除,多謝。