当前位置:首页 > 行业动态 > 正文

socket中的短连接与长连接以及心跳包示例分析「tcp长连接 心跳包」

在计算机网络中,Socket是一种常用的通信方式,它提供了在不同主机之间进行数据传输的接口,在Socket编程中,有两种主要的连接方式:短连接和长连接,心跳包也是网络通信中常用的一种机制,用于检测连接的有效性,本文将对这三种技术进行详细的分析。

1. 短连接

短连接是指在建立连接后,数据发送完毕即断开连接,这种方式适用于数据量较小、传输频率较高的场景,短连接的优点是实现简单,资源消耗较少;缺点是频繁建立和断开连接,增加了网络开销。

在Python中,可以使用`socket`库来实现短连接,以下是一个简单的短连接示例:

import socket

# 创建socket对象
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# 连接到服务器
server_address = ('localhost', 8080)
s.connect(server_address)

# 发送数据
data = 'Hello, World!'
s.sendall(data.encode())

# 关闭连接
s.close() 

2. 长连接

长连接是指在建立连接后,保持连接状态,直到客户端或服务器主动关闭连接,这种方式适用于数据量较大、传输频率较低的场景,长连接的优点是减少了频繁建立和断开连接的开销,提高了传输效率;缺点是占用的资源较多。

在Python中,可以使用`socket`库来实现长连接,以下是一个简单的长连接示例:

import socket

# 创建socket对象
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

# 连接到服务器并保持连接状态
server_address = ('localhost', 8080)
s.connect(server_address)
s.settimeout(60)  # 设置超时时间,避免死锁

# 发送数据
data = 'Hello, World!'
while True:
    try:
        s.sendall(data.encode())
        break  # 发送成功,跳出循环
    except socket.error as e:
        if e.errno == socket.errno.EPIPE:  # 对方已关闭连接,重新建立连接
            print('Connection closed by server, reconnecting...')
            s.close()
            s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
            s.connect(server_address)
        else:
            raise e 

3. 心跳包

心跳包是一种用于检测连接有效性的数据包,在长连接中,客户端和服务器需要定期发送心跳包来确认对方仍然在线,如果一段时间内没有收到对方的心跳包,可以认为连接已经失效,此时可以主动关闭连接或者重新建立连接,心跳包通常包含一个序列号和一个时间戳,用于判断数据包的新旧程度。

在Python中,可以使用`socket`库来实现心跳包的发送和接收,以下是一个简单的心跳包示例:

import socket
import time
import struct
from hashlib import sha1

# 创建socket对象并连接到服务器
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server_address = ('localhost', 8080)
s.connect(server_address)
s.settimeout(60)  # 设置超时时间,避免死锁

# 初始化序列号和时间戳
seq = 0
timestamp = int(time.time()) & 0xFFFFFFFFFFFFFFFFL  # 使用长整数表示时间戳,避免溢出问题
hash_str = f'{seq}{timestamp}'.encode()  # 将序列号和时间戳拼接成字符串,并进行哈希处理,生成唯一的标识符
hash_value = sha1(hash_str).digest()[:4]  # 取哈希值的前4个字节作为校验码,与原始数据一起发送和接收,用于检测数据包的完整性和一致性
data = struct.pack('>I', timestamp) + hash_value + struct.pack('>I', seq) + b'heartbeat'  # 将数据按照一定的格式打包成二进制数据流,方便传输和解析

# 发送心跳包并接收响应
while True:
    s.sendall(data)  # 发送心跳包
    try:
        response = s.recv(1024)  # 接收响应数据,最大接收1024字节的数据包,防止缓冲区溢出问题
        if len(response) < len(data):  # 如果接收到的数据长度小于发送的数据长度,说明对方已经关闭了连接,跳出循环,关闭本地连接并退出程序
            print('Connection closed by server, exiting...')
            s.close()
            break
        received_timestamp, received_hash_value, received_seq, response_data = struct.unpack('>IIII', response[:20])  # 解析接收到的数据,获取序列号、时间戳、校验码和响应数据等字段的值,并进行比较和验证,判断数据包的新旧程度和一致性是否正确
        if received_timestamp != timestamp or received_hash_value != hash_value or received_seq != (seq + 1):  # 如果序列号、时间戳或校验码等字段的值与本地的值不一致,说明数据包已经被改动或者丢失,抛出异常并重新发送心跳包;否则,更新本地的序列号、时间戳和校验码等字段的值,并处理响应数据(例如打印出来)
            raise ValueError('Invalid heartbeat packet')
        print(f'Received response: {response_data}')  # 处理响应数据(例如打印出来)
    except socket.error as e:  # 如果发生网络错误(例如对方已经关闭了连接),跳出循环,关闭本地连接并退出程序;否则,继续发送心跳包并接收响应数据(例如打印出来)
        print('Network error:', e)
        s.close()
        break 
0