實現(xiàn)基于事件通知的.Net套接字
更新時間:2006年09月28日 00:00:00 作者:
以前學(xué)過MFC,我用過的它里面的包裝Winsock的兩個類CSocket和CAsyncSocket,我一直覺得它哪個事件通知的功能比較不錯,特別是在連接的兩方收發(fā)數(shù)據(jù)沒有一定的規(guī)律的時候比較有用,雖然不用它的事件通知也可以實現(xiàn)功能,但是你需要循環(huán)檢測套接字的狀態(tài)或者阻塞等待,如果每次都要做這些瑣碎的工作豈不是很麻煩,于是我就想對這些功能作一次封裝。當(dāng)然在.net里面采用delegate和event是最好的選擇了。下面就是我實現(xiàn)的一些細(xì)節(jié):
先講一下思路:其實這個還是很簡單了,估計高手們會不屑一顧了^_^。當(dāng)套接字開始工作,比如開始監(jiān)聽、開始連接的時候,啟動一個線程不停的檢測套接字的狀態(tài),當(dāng)某一事件的狀態(tài)條件滿足時就觸發(fā)此事件,要知道具體檢測套接字狀態(tài)的方法,請往下看。
平常用套接字編成的時候可能需要用到的功能就是:連接成功或者失敗的通知、在監(jiān)聽的套接字上有掛起的連接的通知、有數(shù)據(jù)可以接受的通知、套接字關(guān)閉的通知、還有套接字空閑可以發(fā)送數(shù)據(jù)的通知等,我也就使對上面幾個功能作了封裝,不過我覺得最后面一個功能用處不大???
我的Socket類是從System.Net.Sockets.Socket繼承下來的,類名TcpEventSocket,實現(xiàn)如下:
先申明事件的委托類型,具體看代碼
public delegate void AcceptConnectionHandler();
public delegate void ConnectCompletedHandler(bool connected); //connected表明連接成功與否
public delegate void DataCanSendHandler();
public delegate void DataCanReceiveHandler(int buffersize); //當(dāng)前可以接受的數(shù)據(jù)量
public delegate void SocketClosedHandler();
這些類型具體什么意思應(yīng)該能看出來吧,另外還要申明對應(yīng)的事件
public event AcceptConnectionHandler AcceptConnection;
public event ConnectCompletedHandler ConnectCompleted;
public event DataCanSendHandler DataCanSend;
public event DataCanReceiveHandler DataCanReceive;
public event SocketClosedHandler SocketClosed;
另外加了幾個虛方法,這幾個方法用來觸發(fā)事件
protected virtual void OnAcceptConnection();
protected virtual void OnConnectCompleted(bool connected)
protected virtual void OnDataCanSend()
protected virtual void OnDataCanReceive(int buffersize)
protected virtual void OnSocketClosed()
為了在適當(dāng)?shù)氖聶C(jī)啟動檢測線程,我重寫了幾個基類的方法:
new public void Listen(int backlog)
{
base.Listen(backlog);
sockState = SocketState.Listenning;
if (!checkThread.IsAlive)
checkThread.Start();
}
new public void Connect(EndPoint remoteEP)
{
try
{
base.Connect(remoteEP);
this.Blocking = false; //設(shè)置非阻塞狀態(tài),以便事件通知的效率
if (!checkThread.IsAlive)
checkThread.Start();
}
catch(SocketException)
{
OnConnectCompleted(false);
}
}
這兩個方法實際編成中都應(yīng)該首先被調(diào)用的,所以有他們啟動檢測線程比較合適,另外線程啟動了必然也要中止,于是我重寫了Close方法
new public void Close()
{
if(checkThread.IsAlive) //先中止線程再關(guān)閉連接
checkThread.Abort();
base.Close();
sockState = SocketState.Disconnected;
OnSocketClosed();
}
于是剩下的工作就是怎么檢測套接字了,Socket類有個Select的靜態(tài)方法,它可以檢測很多套接字的狀態(tài),不過這里只需要檢測一個,所以直接用Socket的Poll方法,Poll的具體用法可以看MSDN,我這里用代碼說明我對套接字的檢測方法
while(true) //循環(huán)檢查
{
if (sockState == SocketState.Disconnected) //如果當(dāng)前沒有連接
{
if (Poll(500, SelectMode.SelectWrite))
OnConnectCompleted(true); //如果為可寫狀態(tài),則表示連接成功
}
else if(sockState == SocketState.Listenning)
{
if (Poll(500, SelectMode.SelectRead)) //如果在監(jiān)聽狀態(tài)發(fā)現(xiàn)套接字有數(shù)據(jù)可讀則表示已經(jīng)有人連接上來可以調(diào)用Accept接受連接
OnAcceptConnection();
}
else //這里sockState = SocketState.Connected
{
if (Poll(500, SelectMode.SelectWrite)) //如果有可寫狀態(tài)則表示可以發(fā)送數(shù)據(jù)
OnDataCanSend();
if (Poll(500, SelectMode.SelectRead)) //如果有可讀狀態(tài)
{
if (Available > 0) //如果有數(shù)據(jù)可讀表示可以調(diào)用Receive接受數(shù)據(jù)
OnDataCanReceive(Available);
else
{
OnSocketClosed(); //沒數(shù)據(jù)可讀表示連接已經(jīng)關(guān)閉
break;
}
}
}
//如果沒有連接并且有錯誤狀態(tài),則連接失敗
if (sockState == SocketState.Disconnected && Poll(500, SelectMode.SelectError))
OnConnectCompleted(false);
}
這里的OnXXX方法就是執(zhí)行事件通知的方法,并且派生類中可以重載這些方法直接獲得事件通知而不需要掛界別的事件通知處理函數(shù)(類似于MFC的OnAccept等虛函數(shù))。不過派生的函數(shù)集的調(diào)用基類的相應(yīng)方法哦。很可惜沒有重寫Socket.Accept方法,讓它返回一個TcpEventSocket,這樣應(yīng)該就比較完整了,只可惜我不知道怎么做^_^,如果誰知道的話歡迎指教。
事件通知機(jī)制已經(jīng)基本完成,所缺乏的只是大量的測試(我做了幾下最簡單的測試,慚愧??!^_^)
如果有問題請聯(lián)系 yzx110@bit.edu.cn
先講一下思路:其實這個還是很簡單了,估計高手們會不屑一顧了^_^。當(dāng)套接字開始工作,比如開始監(jiān)聽、開始連接的時候,啟動一個線程不停的檢測套接字的狀態(tài),當(dāng)某一事件的狀態(tài)條件滿足時就觸發(fā)此事件,要知道具體檢測套接字狀態(tài)的方法,請往下看。
平常用套接字編成的時候可能需要用到的功能就是:連接成功或者失敗的通知、在監(jiān)聽的套接字上有掛起的連接的通知、有數(shù)據(jù)可以接受的通知、套接字關(guān)閉的通知、還有套接字空閑可以發(fā)送數(shù)據(jù)的通知等,我也就使對上面幾個功能作了封裝,不過我覺得最后面一個功能用處不大???
我的Socket類是從System.Net.Sockets.Socket繼承下來的,類名TcpEventSocket,實現(xiàn)如下:
先申明事件的委托類型,具體看代碼
public delegate void AcceptConnectionHandler();
public delegate void ConnectCompletedHandler(bool connected); //connected表明連接成功與否
public delegate void DataCanSendHandler();
public delegate void DataCanReceiveHandler(int buffersize); //當(dāng)前可以接受的數(shù)據(jù)量
public delegate void SocketClosedHandler();
這些類型具體什么意思應(yīng)該能看出來吧,另外還要申明對應(yīng)的事件
public event AcceptConnectionHandler AcceptConnection;
public event ConnectCompletedHandler ConnectCompleted;
public event DataCanSendHandler DataCanSend;
public event DataCanReceiveHandler DataCanReceive;
public event SocketClosedHandler SocketClosed;
另外加了幾個虛方法,這幾個方法用來觸發(fā)事件
protected virtual void OnAcceptConnection();
protected virtual void OnConnectCompleted(bool connected)
protected virtual void OnDataCanSend()
protected virtual void OnDataCanReceive(int buffersize)
protected virtual void OnSocketClosed()
為了在適當(dāng)?shù)氖聶C(jī)啟動檢測線程,我重寫了幾個基類的方法:
new public void Listen(int backlog)
{
base.Listen(backlog);
sockState = SocketState.Listenning;
if (!checkThread.IsAlive)
checkThread.Start();
}
new public void Connect(EndPoint remoteEP)
{
try
{
base.Connect(remoteEP);
this.Blocking = false; //設(shè)置非阻塞狀態(tài),以便事件通知的效率
if (!checkThread.IsAlive)
checkThread.Start();
}
catch(SocketException)
{
OnConnectCompleted(false);
}
}
這兩個方法實際編成中都應(yīng)該首先被調(diào)用的,所以有他們啟動檢測線程比較合適,另外線程啟動了必然也要中止,于是我重寫了Close方法
new public void Close()
{
if(checkThread.IsAlive) //先中止線程再關(guān)閉連接
checkThread.Abort();
base.Close();
sockState = SocketState.Disconnected;
OnSocketClosed();
}
于是剩下的工作就是怎么檢測套接字了,Socket類有個Select的靜態(tài)方法,它可以檢測很多套接字的狀態(tài),不過這里只需要檢測一個,所以直接用Socket的Poll方法,Poll的具體用法可以看MSDN,我這里用代碼說明我對套接字的檢測方法
while(true) //循環(huán)檢查
{
if (sockState == SocketState.Disconnected) //如果當(dāng)前沒有連接
{
if (Poll(500, SelectMode.SelectWrite))
OnConnectCompleted(true); //如果為可寫狀態(tài),則表示連接成功
}
else if(sockState == SocketState.Listenning)
{
if (Poll(500, SelectMode.SelectRead)) //如果在監(jiān)聽狀態(tài)發(fā)現(xiàn)套接字有數(shù)據(jù)可讀則表示已經(jīng)有人連接上來可以調(diào)用Accept接受連接
OnAcceptConnection();
}
else //這里sockState = SocketState.Connected
{
if (Poll(500, SelectMode.SelectWrite)) //如果有可寫狀態(tài)則表示可以發(fā)送數(shù)據(jù)
OnDataCanSend();
if (Poll(500, SelectMode.SelectRead)) //如果有可讀狀態(tài)
{
if (Available > 0) //如果有數(shù)據(jù)可讀表示可以調(diào)用Receive接受數(shù)據(jù)
OnDataCanReceive(Available);
else
{
OnSocketClosed(); //沒數(shù)據(jù)可讀表示連接已經(jīng)關(guān)閉
break;
}
}
}
//如果沒有連接并且有錯誤狀態(tài),則連接失敗
if (sockState == SocketState.Disconnected && Poll(500, SelectMode.SelectError))
OnConnectCompleted(false);
}
這里的OnXXX方法就是執(zhí)行事件通知的方法,并且派生類中可以重載這些方法直接獲得事件通知而不需要掛界別的事件通知處理函數(shù)(類似于MFC的OnAccept等虛函數(shù))。不過派生的函數(shù)集的調(diào)用基類的相應(yīng)方法哦。很可惜沒有重寫Socket.Accept方法,讓它返回一個TcpEventSocket,這樣應(yīng)該就比較完整了,只可惜我不知道怎么做^_^,如果誰知道的話歡迎指教。
事件通知機(jī)制已經(jīng)基本完成,所缺乏的只是大量的測試(我做了幾下最簡單的測試,慚愧??!^_^)
如果有問題請聯(lián)系 yzx110@bit.edu.cn
相關(guān)文章
Asp.net管理信息系統(tǒng)中數(shù)據(jù)統(tǒng)計功能的實現(xiàn)方法
這篇文章主要介紹了Asp.net管理信息系統(tǒng)中數(shù)據(jù)統(tǒng)計功能的實現(xiàn)方法,需要的朋友可以參考下2017-07-07uni-app結(jié)合.NET?7實現(xiàn)微信小程序訂閱消息推送
本文主要介紹了uni-app結(jié)合.NET?7實現(xiàn)微信小程序訂閱消息推送,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-02-02C# 自定義異??偨Y(jié)及嚴(yán)格遵循幾個原則
在C#中所有的異常類型都繼承自System.Exception,也就是說,System.Exception是所有異常類的基類. 總起來說,其派生類分為兩種,需要了解的朋友可以參考下2012-12-12ASP.NET MVC5網(wǎng)站開發(fā)之登錄、驗證和注銷管理員篇1(六)
這篇文章主要介紹了ASP.NET MVC5網(wǎng)站開發(fā)之管理員登錄、驗證和注銷,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-08-08