TCP连接真实断开需主动探测:Socket.Connected不可靠,应结合Socket.Poll()与Available判断,或发心跳包并严格超时处理,禁用Nagle、启用KeepAlive仅作兜底,异步心跳须捕获异常并原子更新状态。判断 TCP 连接是否真的断开,不能只看 Socket.ConnectedSocket.Connected 只是缓存上次操作的状态,底层连接早已静默断开时它仍可能返回 true。真实掉线往往发生在对方突然断电、NAT 超时、防火墙中断等场景,此时本地 socket 不会主动触发 Disconnected 事件或抛异常。必须主动探测:要么发数据看是否抛 SocketException(带 WSAETIMEDOUT 或 WSAENETRESET),要么用 Socket.Poll() + Socket.Available 组合试探:socket.Poll(1000, SelectMode.SelectRead) 返回 true,且 socket.Available == 0 → 对端已关闭连接(FIN)socket.Poll(1000, SelectMode.SelectRead) 返回 true,但读取时抛 SocketException(错误码 10054)→ 连接被重置socket.Poll(1000, SelectMode.SelectError) 返回 true → 底层异常,基本可判为断线心跳包不能只发空字节,得带业务语义和超时控制纯 Send(new byte[0]) 在某些中间设备(如企业级防火墙、云负载均衡)上会被直接丢弃;更糟的是,即使发了,对方不响应也不等于断线------可能只是卡在业务逻辑里。真正可用的心跳要满足三点:使用固定格式的轻量协议帧(例如 0xFF 0x01 表示心跳请求,0xFF 0x02 表示响应),避免被中间件误判为无效流量客户端发心跳后启动独立计时器(比如 System.Threading.Timer),超时(建议 3× 心跳间隔)未收到响应就标记连接异常服务端收到心跳请求必须立即回响应,不能排队或延迟;且服务端也要对每个连接维护最后心跳时间戳,超过阈值(如 5× 客户端心跳间隔)主动 CloseTcpClient 和 Socket 层做心跳的区别很实际用 TcpClient 封装看似简单,但它隐藏了底层 socket 的状态细节:TcpClient.Client.Poll() 可用,但 TcpClient.GetStream().ReadAsync() 在连接断开时可能长时间挂起(受 ReadTimeout 影响,而该属性默认为 0 即无限等待)。 Tellers AI Tellers是一款自动视频编辑工具,可以将文本、文章或故事转换为视频。
相关推荐
m0_609160496 分钟前
Golang怎么实现数据库连接重试_Golang如何在启动时重试连接直到数据库就绪【技巧】罗超驿7 分钟前
8.数据库约束学习笔记:从非空、默认、唯一与主键约束到主键自增花米徐17 分钟前
技术洞察精选 | 2026年4月28日 — 5月4日zxrhhm41 分钟前
PostgreSQL 大规模随机数据生成完整指南techdashen42 分钟前
Cloudflare + PlanetScale:在边缘运行全栈应用,数据库也不例外宝贝儿好1 小时前
【LLM】第三章:项目实操案例:智能输入法项目m0_624578591 小时前
如何在phpMyAdmin中导入GZIP压缩格式文件_加速传输并突破文件大小限制m0_495496411 小时前
mysql数据库表名区分大小写吗_通过lower case table names配置瀚高PG实验室1 小时前
PG的JDBC对SQL中绑定变量个数的限制lifewange1 小时前
NoSQL