對連接上來的連接,進行檢測,以防止客戶端異常關閉,或線路異常斷開,而服務器不知道,得到一個半連接這種情況。
當然可以在協議裡加一個心跳包,然後服務器端定時檢測,過一段時間就去輪訓一次,看哪些連接超過多少時間沒有反應。超時就關閉。但這樣有點不爽,要自己寫代碼來完成。還要鎖定連接列表,代價挺大的。
記得以前在網上看到可以用 TCP 的 KeepAlive 保活機制來做,但也看到人說沒有效果。我想定TCP協議的那此人,不會定一些無用的功能吧,但MSDN上卻沒看到什麼有價值的東東。難道MS沒有實現它們?我決定試試。
1. Delphi 的 WinSock2.pas 不全,要加入一些東東,如下:
const
IOC_IN =80000000 H; // 這裡是 80000000 十六進制,Blog 有問題
IOC_VENDOR =18000000 H; // 這裡是 80000000 十六進制,Blog 有問題
IOC_out =40000000 H; // 這裡是 80000000 十六進制,Blog 有問題
SIO_KEEPALIVE_VALS =IOC_IN or IOC_VENDOR or 4;
type
TTCP_KEEPALIVE = packed record
onoff : integer;
keepalivetime : integer;
keepaliveinterval : integer;
end;
2. 在 accept 得到新連接時,設定它的保活時間, 如下:
// Set KeepAlive 開啟保活機制
opt := 1;
if setsockopt(hClient,SOL_SOCKET, SO_KEEPALIVE, @opt, SizeOf(opt)) <> 0 then
begin
OutputDebugString('setsockopt KeepAlive Error!!!');
end;
// KeepAlive Time 設保活時間
klive.onoff := 1; // 啟用保話
klive.keepalivetime := 10000; // 保話超時
klive.keepaliveinterval := 1; // 超時次數
if WSAIoctl( hClient, SIO_KEEPALIVE_VALS,
@klive,
SizeOf(TTCP_KEEPALIVE),
@outKlive,
SizeOf(TTCP_KEEPALIVE),
@opt,
0,nil) = SOCKET_ERROR then
begin
OutputDebugString('WSAIoctl KeepAlive Error');
end;
=======================================================
有开发网络应用经历的人都知道,网络中的接收和发送数据都是使用WINDOWS中的SOCKET进行实现。但是如果此套接字已经断开,那发送数据和接收数据的时候就一定会有问题。可是如何判断这个套接字是否还可以使用呢?
有人一定想到使用Send函数中的返回结果来进行判断。如果返回的长度和自己发送出去的长度一致,那就说明这个套接字是可用的,否则此套接字一定出现了问题。但是我们并不是无时无刻的发送数据呀。如何解决呢?
其实TCP中已经为我们实现了一个叫做心跳的机制。如果你设置了心跳,那TCP就会在一定的时间(比如你设置的是3秒钟)内发送你设置的次数的心跳(比如说2次),并且此信息不会影响你自己定义的协议。
在VC中实现心跳的例子很多,可是在DLEPHI中一直没有相应的代码。下面我是我使用DELPHI编写的关于心跳的代码(以IOCP为例),希望对大家有帮助。
定义心跳常量
const
IOC_IN =$80000000;
IOC_VENDOR =$18000000;
IOC_out =$40000000;
SIO_KEEPALIVE_VALS =IOC_IN or IOC_VENDOR or 4;
var
inKeepAlive,OutKeepAlive:TTCP_KEEPALIVE;
实现代码是在Acceptsc:= WSAAccept(Listensc, nil, nil, nil, 0);代码的后面加入:
opt:=1;
if setsockopt(Acceptsc,SOL_SOCKET,SO_KEEPALIVE,@opt,sizeof(opt))=SOCKET_ERROR then
begin
closesocket(Acceptsc);
end;
inKeepAlive.onoff:=1;
//设置3秒钟时间间隔
inKeepAlive.keepalivetime:=3000;
//设置每3秒中发送1次的心跳
inKeepAlive.keepaliveinterval:=1;
insize:=sizeof(TTCP_KEEPALIVE);
outsize:=sizeof(TTCP_KEEPALIVE);
if WSAIoctl(Accept,SIO_KEEPALIVE_VALS,@inKeepAlive,insize,@outKeepAlive,outsize,@outByte,nil,nil)=SOCKET_ERROR then
begin
closesocket(Acceptsc);
end;
如果加入以上的代码以后,系统会每3秒中加入一次的心跳。并且如果客户端断线以后(网线断),函数GetQueuedCompletionStatus会返回FALSE。
if (GetQueuedCompletionStatus(CompletionPort, BytesTransferred,DWORD(PerHandleData), POverlapped(PerIoData), INFINITE) = False) then
begin
//在这里处理客户端断线信息。
continue;
end;
以上就是我使用心跳的方法,此方法我已经在我的网络游戏中使用。情况稳定!