

新闻资讯
技术学院本文详解解决 python socket 通信中因字节流混杂导致的 `unicodedecodeerror: 'utf-8' codec can't decode byte 0xb5` 错误,核心在于严格分离控制信息(如文件名、大小)与二进制数据流,并确保 utf-8 编码/解码仅作用于纯文本字段。
该错误的根本原因并非编码参数设置不当,而是协议设计缺陷:发送端在 client.send(str(file_size).encode()) 后紧接着调用 client.sendall(encrypted)(发送加密后的二进制密文),而接收端却用固定缓冲区 recv(1024) 读取 file_size —— 此时 TCP 流无消息边界,极大概率将部分密文字节(如 0xb5)误纳入 file_size 的接收缓冲区,导致
decode('utf-8') 失败。
✅ 正确做法是:严格分阶段收发,且每阶段使用确定性长度或明确分隔符。以下是推荐方案:
# 发送端(Server)
client.send(file_name.encode('utf-8')) # 纯文本,UTF-8 安全
client.send(f"{file_size}".encode('utf-8')) # 纯数字字符串,UTF-8 安全
client.sendall(encrypted) # 纯二进制,不参与 decode
client.send(b'') # 可选:标记控制信息结束 # 接收端(Client)
# ✅ 安全读取文件名(假设最长100字节)
file_name = client.recv(100).decode('utf-8').strip('\x00')
# ✅ 安全读取文件大小(同样限定长度,避免截断或混入)
file_size_bytes = client.recv(20).decode('utf-8').strip('\x00')
file_size = int(file_size_bytes)
# ✅ 跳过可选分隔符(如有)
if client.recv(5) != b'':
raise ValueError("Missing delimiter")
# ✅ 按确切字节数接收加密数据
received = 0
with open(file_name, 'wb') as f:
while received < file_size:
chunk = client.recv(min(8192, file_size - received))
if not chunk:
raise ConnectionError("Connection closed prematurely")
f.write(chunk)
received += len(chunk) 通过协议层面的清晰分层,即可根治此类 UnicodeDecodeError —— 本质不是编码问题,而是数据边界失控。