indy的服务器端和客户端心跳处理


服务端的心跳处理(idtcpserver)

//定义心跳常量
Const
IOC_IN = $80000000;
IOC_VENDOR = $18000000;
IOC_out = $40000000;
SIO_KEEPALIVE_VALS = IOC_IN or IOC_VENDOR or 4;
DATA_BUFSIZE = 8192;

//定义 KeepAlive 数据结构
Type
TTCP_KEEPALIVE = packed record
onoff: integer;
keepalivetime: integer;
keepaliveinterval: integer;
end;

在idtcpserver的OnConnect事件里面加入以下代码:

var
opt:Integer;
klive, outKlive: TTCP_KEEPALIVE;

begin
opt := 1;
if setsockopt(AThread.Connection.Socket.Binding.Handle,SOL_SOCKET, SO_KEEPALIVE, @opt, SizeOf(opt)) <> 0 then
begin
Showmessage('setsockopt KeepAlive Error!');
end;
klive.onoff := 1;
klive.keepalivetime := 3000;
klive.keepaliveinterval := 1;
if WSAIoctl(AThread.Connection.Socket.Binding.Handle, SIO_KEEPALIVE_VALS, @klive,
SizeOf(TTCP_KEEPALIVE), @outKlive,
SizeOf(TTCP_KEEPALIVE), @opt,0,nil) = SOCKET_ERROR then
begin
Showmessage('WSAIoctl KeepAlive Error!');
end

end

在idtcpserver的OnException事件里面就能捕获到这个异常,并进行异常处理了

procedure Tfrmmain.IdTCPServer1Exception(AThread: TIdPeerThread;
AException: Exception);
begin
suiMemo1.Lines.Add('客户端'+inttostr(athread.handle)+'异常断开');
if AThread.Connection.Connected then AThread.Connection.Disconnect;
end;





客户端的心跳处理(idtcpclient)
procedure Tfrmmain.IdTCPClient1Connected(Sender: TObject);
var
opt:Integer;
klive, outKlive: TTCP_KEEPALIVE;
begin
opt := 1;
if setsockopt(IdTCPClient1.Socket.Binding.Handle,SOL_SOCKET, SO_KEEPALIVE, @opt, SizeOf(opt)) <> 0 then
begin
Showmessage('setsockopt KeepAlive Error!');
end;
klive.onoff := 1;
klive.keepalivetime := 3000;
klive.keepaliveinterval := 1;
if WSAIoctl(IdTCPClient1.Socket.Binding.Handle, SIO_KEEPALIVE_VALS, @klive,
SizeOf(TTCP_KEEPALIVE), @outKlive,
SizeOf(TTCP_KEEPALIVE), @opt,0,nil) = SOCKET_ERROR then
begin
Showmessage('WSAIoctl KeepAlive Error!');
end;
end;



客户端的异常捕捉,我是放在客户端读取服务端发送过来数据的线程里面

procedure TReadThread.Execute;
var
TestBuffer:TSendDataNet;
begin
{ Place thread code here }
frmmain.log('Client Begin reading...');
while permit do
begin
try
frmmain.IdTCPClient1.ReadBuffer(TestBuffer,SizeOf(TestBuffer));
sleep(100);
except
on e:Exception do
begin
permit:=False;
if frmmain.IdTCPClient1.Connected then frmmain.IdTCPClient1.Disconnect;
frmmain.RzStatusPane1.Caption:='断开;
frmmain.LinkTimer.Enabled:=True;//这儿启动定时器重连服务端
end;
end;
end;
ThreadDestroy;
end;
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Delphi 7中,Indy 10是一个非常有用的组件库,它提供了一种在服务器和客户端之间进行TCP/IP通信的方法。 首先,要创建一个服务器端的IDTcpServer组件,可以在Delphi的组件面板上找到。通过设置一些属性,比如端口号和OnConnect / OnDisconnect事件来监听和处理客户端的连接和断开。 在OnConnect事件中,可以编写代码来处理客户端连接时的操作,比如给客户端发送欢迎信息。同样,在OnDisconnect事件中,可以编写代码来处理断开连接时的操作,比如释放客户端资源。 如果需要在客户端断开连接时发生报错,可以在回归事件(例如OnDisconnect)中获取异常信息并进行处理。可以通过获取异常的消息,堆栈跟踪和错误代码等来确定问题所在。一种常见的处理方法是使用try...except块来捕获异常并提供适当的错误处理。 解决报错的方法可能取决于具体的错误类型。一种常见的报错是EIdConnClosedGracefully,它表示连接被对等方正常关闭。可以通过在OnDisconnect事件中检查异常类型,并根据需要采取不同的处理方式来解决这个问题。 总之,在Delphi 7中,使用Indy 10的IDTcpServer组件,可以方便地处理服务器和客户端的连接和断开。当断开客户端时发生报错时,可以通过在适当的事件中捕获和处理异常来解决问题。根据具体的错误类型和需求,可以采取不同的处理方法来解决问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值