|
先講一下思路:其實這個還是很簡單了,估計高手們會不屑一顧了^_^。當(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啟動檢測線程,我重寫了幾個基類的方法:
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īng)基本完成,所缺乏的只是大量的測試(我做了幾下最簡單的測試,慚愧!!^_^)
如果有問題請聯(lián)系 yzx110@bit.edu.cn
AspNet技術(shù):實現(xiàn)基于事件通知的.Net套接字,轉(zhuǎn)載需保留來源!
鄭重聲明:本文版權(quán)歸原作者所有,轉(zhuǎn)載文章僅為傳播更多信息之目的,如作者信息標(biāo)記有誤,請第一時間聯(lián)系我們修改或刪除,多謝。