0

0

如何在Golang中实现TCP长连接与心跳机制

P粉602998670

P粉602998670

发布时间:2026-02-25 12:26:03

|

196人浏览过

|

来源于php中文网

原创

tcp连接会悄无声息断开是因为nat/防火墙等中间设备存在5–30分钟空闲超时,而go的net.conn不感知此状态;需应用层主动发带验证响应的心跳包,并结合setreaddeadline实现可靠保活。

如何在golang中实现tcp长连接与心跳机制

为什么 TCP 连接会悄无声息地断开

不是网络突然挂了才断连,而是中间 NAT 设备、防火墙、负载均衡器普遍有连接空闲超时(通常 5–30 分钟),一动不动就回收连接。Go 的 net.Conn 本身不感知这个,Write 可能还成功(数据进了内核发送缓冲区),但对端早已关闭——你发出去的心跳包根本没走远。

  • 别依赖 conn.SetDeadline 单纯防本地阻塞,它不解决中间链路静默断连
  • 心跳必须由应用层主动发起,且要带可验证的响应(比如回传时间戳或序列号)
  • 服务端不能只靠 Read 返回 io.EOF 才清理连接——客户端可能卡在半路,连接状态已失效

SetReadDeadline + 定时 Write 实现可靠心跳

核心思路:每次成功读/写后重置读超时;心跳定时器只负责发包,不等响应;真正判断是否存活,看下一次 Read 是否超时。

  • 客户端侧:启动一个 time.Ticker,每 25 秒调用 conn.Write 发一个短心跳包(如 []byte{0x01}),同时确保每次 Read 前都调用 conn.SetReadDeadline(time.Now().Add(30 * time.Second))
  • 服务端侧:同样为每个连接维护读超时,收到心跳后立刻重置,不响应也行(减小开销),但必须更新 deadline
  • 注意 SetReadDeadline 影响后续所有 Read,包括心跳包的读取,所以心跳处理逻辑必须包含在正常读循环里,不能单独 goroutine 阻塞读
// 示例:服务端读循环片段
for {
    conn.SetReadDeadline(time.Now().Add(30 * time.Second))
    n, err := conn.Read(buf[:])
    if err != nil {
        if netErr, ok := err.(net.Error); ok && netErr.Timeout() {
            // 连续两次读超时才判定断连
            if lastReadTimeout {
                closeConn(conn)
                return
            }
            lastReadTimeout = true
            continue
        }
        closeConn(conn)
        return
    }
    lastReadTimeout = false
    handlePacket(buf[:n])
}

KeepAlive 系统级选项只能辅助,不能替代应用层心跳

Go 的 net.Conn 支持 SetKeepAliveSetKeepAlivePeriod,但这只是启用 TCP 层的 SO_KEEPALIVE,默认间隔是 2 小时(Linux),且无法穿透大部分中间设备——NAT 早把你踢了,TCP keepalive 根本发不到对端。

android rtsp流媒体播放介绍 中文WORD版
android rtsp流媒体播放介绍 中文WORD版

本文档主要讲述的是android rtsp流媒体播放介绍;实时流协议(RTSP)是应用级协议,控制实时数据的发送。RTSP提供了一个可扩展框架,使实时数据,如音频与视频,的受控、点播成为可能。数据源包括现场数据与存储在剪辑中数据。该协议目的在于控制多个数据发送连接,为选择发送通道,如UDP、组播UDP与TCP,提供途径,并为选择基于RTP上发送机制提供方法。希望本文档会给有需要的朋友带来帮助;感兴趣的朋友可以过来看看

下载
  • 开启方式:tcpConn.SetKeepAlive(true) + tcpConn.SetKeepAlivePeriod(30 * time.Second)(需 Go 1.19+)
  • 仅建议作为兜底:防止进程僵死连接占用资源,但业务逻辑绝不能依赖它保活
  • Windows 默认 keepalive 时间更长,行为不一致,跨平台部署时尤其不可靠

心跳包设计和连接管理容易被忽略的细节

心跳不是发个字节就完事,协议设计和状态同步才是关键。

立即学习go语言免费学习笔记(深入)”;

  • 心跳包要有唯一标识(如递增 seq 或时间戳),服务端可识别重复包、乱序包,避免误判
  • 不要在心跳 goroutine 里直接 conn.Write 并忽略错误——如果写失败(比如连接已关),得通知主读循环退出
  • 连接池或长连接管理器中,必须把心跳失败与业务读写失败同等对待:立即标记为“待驱逐”,不再分发新请求
  • 测试时用 iptables -A OUTPUT -p tcp --dport 8080 -j DROP 模拟单向丢包,比直接 kill -9 更真实

真正难的不是发心跳,而是让整个连接生命周期里的读、写、超时、错误、重连全部串成一条因果链——任何一个环节断掉,都会让“长连接”变成“伪长连接”。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

智谱清言 - 免费全能的AI助手
智谱清言 - 免费全能的AI助手

智谱清言 - 免费全能的AI助手

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

207

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

242

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

350

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

212

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

405

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

365

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

200

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

1111

2025.06.17

batoto漫画官网入口与网页版访问指南
batoto漫画官网入口与网页版访问指南

本专题系统整理batoto漫画官方网站最新可用入口,涵盖最新官网地址、网页版登录页面及防走失访问方式说明,帮助用户快速找到batoto漫画官方平台,稳定在线阅读各类漫画内容。

65

2026.02.25

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 5.5万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号