最近我負責乙個im專案的開發,服務端和客戶端採用tcp協議連線。服務端採用c#開發,客戶端採用delphi開發。在服務端開發中我碰到了各種各樣的網路異常斷開現象。在處理這些異常的時候有了一些心得,現在寫出來和大家分享一下。
那網路異常斷開原因主要有那些呢?歸納起來主要有以下兩種:
1、客戶端程式異常。
對於這種情況,我們很好處理,因為客戶端程式異常退出會在服務端引發connectionreset的socket異常(就是winsock2中的10054異常)。只要在服務端處理這個異常就可以了。
2、網路鏈路異常。
如:網線拔出、交換機掉電、客戶端機器掉電。當出現這些情況的時候服務端不會出現任何異常。這樣的話上面的**就不能處理這種情況了。對於這種情況在msdn裡面是這樣處理的,我在這裡貼出msdn的原文:
如果您需要確定連線的當前狀態,請進行非阻止、零位元組的 send 呼叫。如果該呼叫成功返回或引發 waewouldblock 錯誤** (10035),則該套接字仍然處於連線狀態;否則,該套接字不再處於連線狀態。
但是我在實際應用中發現,msdn說的這種處理方法在很多時候根本無效,無法檢測出網路已經異常斷開了。那我們該怎麼辦呢?
我們知道,tcp有乙個連線檢測機制,就是如果在指定的時間內(一般為2個小時)沒有資料傳送,會給對端傳送乙個keep-alive資料報,使用的序列號是曾經發出的最後乙個報文的最後乙個位元組的序列號,對端如果收到這個資料,回送乙個tcp的ack,確認這個位元組已經收到,這樣就知道此連線沒有被斷開。如果一段時間沒有收到對方的響應,會進行重試,重試幾次後,向對端發乙個reset,然後將連線斷掉。
在windows中,第一次探測是在最後一次資料傳送的兩個小時,然後每隔1秒探測一次,一共探測5次,如果5次都沒有收到回應的話,就會斷開這個連線。但兩個小時對於我們的專案來說顯然太長了。我們必須縮短這個時間。那麼我們該如何做呢?我要利用socket類的iocontrol()函式。我們來看看這個函式能幹些什麼:
使用 iocontrolcode 列舉指定控制**,為 socket 設定低階操作模式。
命名空間:system.net.sockets
程式集:system(在 system.dll 中)
語法c#
public int iocontrol (
iocontrolcode iocontrolcode,
byte optioninvalue,
byte optionoutvalue
)引數
iocontrolcode
乙個 iocontrolcode 值,它指定要執行的操作的控制**。
optioninvalue
byte 型別的陣列,包含操作要求的輸入資料。
optionoutvalue
byte 型別的陣列,包含由操作返回的輸出資料。
返回值
optionoutvalue 引數中的位元組數。
如:socket.iocontrol(iocontrolcode.keepalivevalues, inoptionvalues, null);
我們要搞清楚的就是inoptionvalues的定義,在c++裡它是乙個結構體。我們來看看這個結構體:
struct tcp_keepalive
...;
在c#中,我們直接用乙個byte陣列傳遞給函式:
uint dummy = 0;
byte inoptionvalues = new byte[marshal.sizeof(dummy) * 3];
bitconverter.getbytes((uint)1).copyto(inoptionvalues, 0);//是否啟用keep-alive
bitconverter.getbytes((uint)5000).copyto(inoptionvalues, marshal.sizeof(dummy));//多長時間開始第一次探測
bitconverter.getbytes((uint)5000).copyto(inoptionvalues, marshal.sizeof(dummy) * 2);//探測時間間隔
具體實現**:
public static void acceptthread()
...catch ...}}
private static void accept(byte inoptionvalues)
...好了,這樣就成功了。
文章整理:、服務
碰撞檢測 判斷線段相交
3d 叉乘 3d 叉乘的結果是乙個 3d 向量,這個向量垂直於參與運算的2個向量的法向量。3d 叉乘計算公式 a.y b.z b.y a.z a.z b.x b.z a.x a.x b.y b.x a.y 2d叉乘 2d 叉乘的結果是乙個標量。2d 叉乘就是把 2d 點看作 3d 的點 z 軸的值為...
C 判斷檢測網路是否連線
完整的 using system.runtime.interopservices namespace internet public class class1 dllimport wininet.dll private extern static bool internetgetconnecteds...
檢測網路狀態
搞了一下午,什麼internetcheckconnection,internetconnect,internetqueryoption試了一堆,發現這些函式要不沒用,要不用起來很麻煩,根本檢測不了網路的狀態。因為機子上本來有wifi和本地網路,所以就用socket了 這是朋友給的乙個函式,方便以後使...