心跳函数.doc

上传人:11****ws 文档编号:3012713 上传时间:2019-05-17 格式:DOC 页数:3 大小:24.50KB
下载 相关 举报
心跳函数.doc_第1页
第1页 / 共3页
心跳函数.doc_第2页
第2页 / 共3页
心跳函数.doc_第3页
第3页 / 共3页
亲,该文档总共3页,全部预览完了,如果喜欢就下载吧!
资源描述

1、偶来凑凑热闹。前些日子,和群里面的兄弟讨论过这个问题,为此,还说了另外的一些话题,比如三次握手,滑动窗口,消息边界,心跳函数之类。如果不清楚的,请补习一下相应的知识吧,这可是底层原理,不清楚,自然不晓得为什么,客户端异常断开,服务器端侦测不到。从根源上说,这是由于 TCP 协议的问题。TCP 协议的初衷是: 网络中断时,仍能维持通信的能力。美国国防部要求能在遭到核打击或其它灾害的时候,仍然能维持计算机之间可靠的网络通知协议。呵呵,如果客户端断开,服务器就马上释放连接。还维持什么?和初衷不符吧。要搞清楚客户端断开,为什么服务器端不能立即侦测到。这个问题,必须要理解三次握手机制。当然服务器也能侦测

2、到异常断开,KEEPALIVE 的默认时间是 2 小时哟。详细的 TCP 三次握手机制:客户端发送一个带 SYN 位的请求,向服务器表示需要连接,假设请求序号号为 SYN = 10, ACK=0服务器接收到这样的请求后,查看是否在 LISTEN 的是指定的端口,不然,就发送 RST=1 应答,拒绝建立连接。 如果接收连接,那么服务器发送确认,SYN 为服务器的一个内码,假设为 100,ACK 位则是客户端的请求序号加 1,本例中发送的数据是:SYN=100,ACK=11,用这样的数据发送给客户端。 客户端发送确认建立连接的消息给服务器。确认信息的 SYN 位是服务器发送的 ACK 位,ACK

3、位是服务器发送的 SYN 位加 1 。如果 TCP 连接第三次握手中,用户向服务器发送了一个 SYN 后就掉线了( 第一步)。服务器发出SYN+ACK 应答报文(第二步) 。然后就再也无法收到客户端的 ACK 报文的。这种情况下,服务器就不断的重试, (再次发送 SYN+ACK 给客户端) ,直到超时。注意:服务器是接收到 SYN 请求就立即与客户端建立连接,而是先为连接请求分配内存空间,建立会话,并放到一个等待队列中。如果,这个等待的队列已经满了,那么,服务器就不在为新的连接分配任何东西,直接丢弃新的请求。这就是服务器的拒绝服务了。 因此,如果有人写一个恶意程序来试试你的 Borland s

4、ocket server,后果实在不敢去想。知道了问题的原因,也就能找出解决办法。解决办法就是用心跳函数。在 TCP 中有一个 Keep-alive 的机制可以检测死连接,原理很简单,TCP 会在空闲了一定时间后发送数据给对方:1.如果主机可达,对方就会响应 ACK 应答,就认为是存活的。2.如果可达,但应用程序退出,对方就发 RST 应答,发送 TCP 撤消连接。3.如果可达,但应用程序崩溃,对方就发 FIN 消息。4.如果对方主机不响应 ack, rst,继续发送直到超时,就撤消连接。这个时间就是默认的二个小时。用 keep_alive 可以检测死连接,并撤消连接。但是是不是我们希望的呢?

5、可能是,可能不是。有时可能我们希望重新通过另一个连接,来实现系统正常运转。keep_alive 实现:type TCP_KeepAlive = record OnOff: Cardinal; KeepAliveTime: Cardinal; KeepAliveInterval: Cardinal end; var Val: TCP_KeepAlive; Ret: DWord; begin Val.OnOff:=1; Val.KeepAliveTime:=xxx; Val.KeepAliveInterval:=xxx; WSAIoctl(AThread.Connection.Socket.Bin

6、ding.Handle, IOC_IN or IOC_VENDOR or 4, Val, SizeOf(Val), nil, 0, Ret, nil, nil) end;心跳函数:客户端定时向服务器发送数据,表示自己存活着。在固定的时间内,服务器没有收到消息,就认为客户端已经断开,并释放连接。而这个过程也可以是双方的。即两边都有心跳机制。Borland socket server 具体代码实现:从派生 TServerClientThread 线程类。THeartBeatServerClientThread= class(TServerClientThread) privatecs : TCri

7、ticalSection; protected.procedure AddClient(vHeartBeat: THeartBeatServerClientThread);procedure RemoveClient(vHeartBeat: THeartBeatServerClientThread);end;WorkThread: Array 0.MaxLink-1 of THeartBeatServerClientThread; /连接队列 procedure THeartBeatServerClientThread.Execute;var theStream : TWinSocketStr

8、eam; begin . theStream := TWinSocketStream.Create(ClientSocket,120000); While (not Terminated) and (ClientSocket.Connected) do begin try if theStream.WaitForData(10) then begin Len := theStream.Read.if Len=0 then begin /客户关闭连接 ClientSocket.Close; end else begin theStream.Write(Buffer,nLen); end; end; except ClientSocket.Close; /Read 和 Write 出错,断开连接。end; end;

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 教育教学资料库 > 精品笔记

Copyright © 2018-2021 Wenke99.com All rights reserved

工信部备案号浙ICP备20026746号-2  

公安局备案号:浙公网安备33038302330469号

本站为C2C交文档易平台,即用户上传的文档直接卖给下载用户,本站只是网络服务中间平台,所有原创文档下载所得归上传人所有,若您发现上传作品侵犯了您的权利,请立刻联系网站客服并提供证据,平台将在3个工作日内予以改正。