0

0

如何在Golang中实现基于TCP的实时聊天系统_Golang TCP聊天系统开发实战

P粉602998670

P粉602998670

发布时间:2026-02-07 11:51:53

|

203人浏览过

|

来源于php中文网

原创

net.Conn 不能直接复用在多个 goroutine 中读写,因其底层读写共享缓冲与状态机,并发调用易导致 io.ErrUnexpectedEOF 或静默丢包;正确做法是读写分离并加锁串行化写操作。

如何在golang中实现基于tcp的实时聊天系统_golang tcp聊天系统开发实战

为什么 net.Conn 不能直接复用在多个 goroutine 中读写

很多初学者会把同一个 net.Conn 同时交给两个 goroutine:一个负责 Read,一个负责 Write。这看似合理,但 TCP 连接本身不是线程安全的——ReadWrite 共享底层缓冲与状态机,一旦并发调用,可能触发 io.ErrUnexpectedEOF 或静默丢包。

正确做法是:每个连接启动两个明确分工的 goroutine,且用互斥控制写操作(读可独立,写需排队):

// 示例:写入前加锁
var mu sync.Mutex
go func() {
    scanner := bufio.NewScanner(conn)
    for scanner.Scan() {
        msg := scanner.Text()
        mu.Lock()
        conn.Write([]byte("echo: " + msg + "\n"))
        mu.Unlock()
    }
}()
  • 读 goroutine 可以无锁运行,但别让它阻塞在 Read 上太久;建议用 SetReadDeadline 防僵死
  • 写操作必须串行化,否则可能错乱或 panic;用 sync.Mutex 最轻量,chan []byte 更适合高吞吐场景
  • 不要在 handler 里直接 conn.Close(),应通知读/写 goroutine 自行退出,再统一关闭

如何避免客户端断连后服务器 goroutine 泄漏

常见错误是只监听 conn.Read 返回 io.EOF 就结束,却没处理网络中断、超时、RST 等情况,导致 goroutine 卡在 ReadWrite 上不退出。

关键要结合连接生命周期管理:

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

conn.SetReadDeadline(time.Now().Add(30 * time.Second))
n, err := conn.Read(buf)
if netErr, ok := err.(net.Error); ok && netErr.Timeout() {
    // 主动断开闲置连接
    conn.Close()
    return
}
if err == io.EOF || strings.Contains(err.Error(), "broken pipe") {
    conn.Close()
    return
}
  • 每次读/写前都调用 SetReadDeadline / SetWriteDeadline,值随业务逻辑动态更新(比如心跳间隔)
  • 检查 err 时不能只判 io.EOF,还要覆盖 "use of closed network connection""broken pipe""connection reset by peer"
  • 为每个连接配一个 context.WithCancel,当检测到异常时调用 cancel(),让所有关联 goroutine 快速退出

怎么给每个 TCP 连接分配唯一 ID 并支持广播

Go 没有内置连接 ID,靠 conn.RemoteAddr().String() 不可靠(NAT 后端地址重复),也不能用指针(GC 可能移动)。必须自己生成并维护映射。

Ribbet.ai
Ribbet.ai

免费在线AI图片处理编辑

下载

推荐方案:服务端启动时用 atomic.Int64 递增生成 ID,同时用 sync.Map 存活连接:

var nextID atomic.Int64
var clients sync.Map // map[int64]*client

type client struct { conn net.Conn id int64 }

func newClient(conn net.Conn) *client { id := nextID.Add(1) c := &client{conn: conn, id: id} clients.Store(id, c) return c }

  • 广播时遍历 clients.Range,对每个 *client 尝试写入,遇到错误立即 clients.Delete(id) 并关闭 conn
  • 不要在广播循环里做耗时操作(如 JSON 序列化),先序列化好再发,或提前缓存格式化后的字节
  • 客户端重连时,旧 ID 要主动清理;可在新连接握手阶段发送 token,服务端比对并踢掉旧连接

为什么不用 bufio.Scanner 做粘包处理

bufio.Scanner 默认按行切割,但聊天消息未必换行;更严重的是它内部有 64KB 缓冲上限,超长消息直接报 scanner: token too long,且无法自定义分隔符长度。

真实场景中必须自己处理粘包,最简方式是「定长头 + 变长体」:

// 发送端:先写 4 字节长度,再写字节流
length := uint32(len(msg))
binary.Write(conn, binary.BigEndian, length)
conn.Write([]byte(msg))

// 接收端:先读 4 字节,再读指定长度 var length uint32 binary.Read(conn, binary.BigEndian, &length) buf := make([]byte, length) io.ReadFull(conn, buf) // 阻塞直到读满

  • 别用 ReadString('\n'),换行符可被用户输入,不可信
  • io.ReadFull 比反复 Read 更稳,它保证读够字节数或返回错误
  • 如果想兼容 WebSocket 或后续升级,协议头里加个 magic number 和版本字段,方便未来扩展

实际跑起来后,最常被忽略的是连接数突增时的文件描述符耗尽问题。Linux 默认单进程最多打开 1024 个 fd,而每个 TCP 连接占一个。上线前务必 ulimit -n 65536,并在代码里用 net.ListenConfig{LimitListener: ...} 做软限流。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

184

2024.02.23

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

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

232

2024.02.23

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

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

344

2024.02.23

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

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

210

2024.03.05

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

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

398

2024.05.21

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

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

282

2025.06.09

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

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

196

2025.06.10

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

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

641

2025.06.17

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

2

2026.02.06

热门下载

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

精品课程

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

共48课时 | 8.6万人学习

Git 教程
Git 教程

共21课时 | 3.4万人学习

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

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