值得一看
广告
彩虹云商城
广告

热门广告位

Golang TCP长连接服务实现示例

Go语言通过net包实现TCP长连接服务,用于即时通讯等场景。首先使用net.Listen监听端口,Accept接受连接并为每个客户端启动goroutine处理读写。在handleConnection中,开启读协程接收数据,通过SetReadDeadline设置读超时实现心跳检测,收到消息后重置超时,若接收到”ping”则回复”pong”。主协程等待done信号关闭连接。客户端示例通过Dial连接服务器,循环发送消息并读取回显,验证通信正常。整体结构包含连接管理、读写分离与心跳机制,具备扩展性,可结合context、协议编码等提升生产环境稳定性。

golang tcp长连接服务实现示例

在Go语言中实现TCP长连接服务,通常用于即时通讯、设备上报、心跳维持等场景。核心是通过net包监听TCP连接,为每个客户端维护一个持久连接,并支持读写分离、心跳检测、并发安全处理。下面是一个简洁但完整的TCP长连接服务示例。

1. 基础TCP服务器结构

使用net.Listen创建监听,接受客户端连接并为每个连接启动独立的goroutine处理数据收发。

package main
<p>import (
"fmt"
"log"
"net"
)</p><p>func main() {
listener, err := net.Listen("tcp", ":8080")
if err != nil {
log.Fatal("监听端口失败:", err)
}
defer listener.Close()
fmt.Println("TCP服务器启动,监听 :8080")</p><pre class="brush:php;toolbar:false;"><pre class="brush:php;toolbar:false;">for {
conn, err := listener.Accept()
if err != nil {
log.Println("接受连接失败:", err)
continue
}
go handleConnection(conn)
}

}

2. 处理长连接的读写逻辑

每个连接开启两个goroutine:一个负责读取客户端数据,另一个可选地处理消息发送。使用sync.Mutex或通道协调状态。

func handleConnection(conn net.Conn) {
defer conn.Close()
clientIP := conn.RemoteAddr().String()
fmt.Printf("客户端 %s 已连接\n", clientIP)
<pre class="brush:php;toolbar:false;"><pre class="brush:php;toolbar:false;">// 模拟存储连接状态(实际可用map+锁或context管理)
done := make(chan bool)
// 读协程
go func() {
buffer := make([]byte, 1024)
for {
n, err := conn.Read(buffer)
if err != nil {
fmt.Printf("客户端 %s 断开: %v\n", clientIP, err)
done <- true
return
}
msg := string(buffer[:n])
fmt.Printf("来自 %s 的消息: %s\n", clientIP, msg)
// 回显消息(可替换为业务逻辑)
conn.Write([]byte("echo: " + msg))
}
}()
// 可扩展:写协程通过channel接收消息主动下发
<-done

}

3. 添加心跳机制保活

TCP连接可能因网络中断静默断开,需加入心跳检测。常见做法是设定读超时,或让客户端定期发送ping。

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

芦笋演示

芦笋演示

一键出成片的录屏演示软件,专为制作产品演示、教学课程和使用教程而设计。

芦笋演示34

查看详情
芦笋演示

import (
"time"
)
<p>func handleConnection(conn net.Conn) {
defer conn.Close()
clientIP := conn.RemoteAddr().String()
fmt.Printf("客户端 %s 已连接\n", clientIP)</p><pre class="brush:php;toolbar:false;"><pre class="brush:php;toolbar:false;">conn.SetReadDeadline(time.Now().Add(30 * time.Second)) // 初始超时
done := make(chan bool)
go func() {
buffer := make([]byte, 1024)
for {
n, err := conn.Read(buffer)
if err != nil {
done <- true
return
}
msg := string(buffer[:n])
fmt.Printf("收到: %s\n", msg)
// 收到消息重置超时时间
conn.SetReadDeadline(time.Now().Add(30 * time.Second))
// 简单协议判断是否为心跳
if msg == "ping" {
conn.Write([]byte("pong"))
} else {
conn.Write([]byte("echo: " + msg))
}
}
}()
// 超时或异常退出
<-done
fmt.Printf("连接 %s 关闭\n", clientIP)

}

4. 客户端测试示例

使用简单客户端连接并发送数据验证服务稳定性。

package main
<p>import (
"fmt"
"net"
"time"
)</p><p>func main() {
conn, _ := net.Dial("tcp", "127.0.0.1:8080")
defer conn.Close()</p><pre class="brush:php;toolbar:false;"><pre class="brush:php;toolbar:false;">for i := 0; i < 5; i++ {
msg := fmt.Sprintf("hello %d", i)
conn.Write([]byte(msg))
buf := make([]byte, 1024)
n, _ := conn.Read(buf)
fmt.Println("回显:", string(buf[:n]))
time.Sleep(2 * time.Second)
}
// 发送心跳
conn.Write([]byte("ping"))
buf := make([]byte, 1024)
n, _ := conn.Read(buf)
fmt.Println("心跳响应:", string(buf[:n]))

}

基本上就这些。这个示例展示了如何用Go构建一个基础的TCP长连接服务,包含连接管理、读写分离和心跳机制。实际生产环境中可结合context取消、连接池、协议编码(如protobuf)、日志监控等进一步增强健壮性。

相关标签:

go golang go语言 编码 端口 ai 日志监控 golang 循环 Go语言 并发

大家都在看:

Go语言中解析JSON时保留64位整型值的策略
如何在 Go 中解析转义的 JSON 字符串
Go语言中Goroutine与标准库及第三方包的并发使用指南
Go语言子字符串的内存管理与高效复制:避免潜在的内存泄露
Go语言中切片指针的预分配与填充:惯用方法解析
温馨提示: 本文最后更新于2025-10-13 22:39:30,某些文章具有时效性,若有错误或已失效,请在下方留言或联系在线客服
文章版权声明 1 本网站名称: 创客网
2 本站永久网址:https://new.ie310.com
1 本文采用非商业性使用-相同方式共享 4.0 国际许可协议[CC BY-NC-SA]进行授权
2 本站所有内容仅供参考,分享出来是为了可以给大家提供新的思路。
3 互联网转载资源会有一些其他联系方式,请大家不要盲目相信,被骗本站概不负责!
4 本网站只做项目揭秘,无法一对一教学指导,每篇文章内都含项目全套的教程讲解,请仔细阅读。
5 本站分享的所有平台仅供展示,本站不对平台真实性负责,站长建议大家自己根据项目关键词自己选择平台。
6 因为文章发布时间和您阅读文章时间存在时间差,所以有些项目红利期可能已经过了,能不能赚钱需要自己判断。
7 本网站仅做资源分享,不做任何收益保障,创业公司上收费几百上千的项目我免费分享出来的,希望大家可以认真学习。
8 本站所有资料均来自互联网公开分享,并不代表本站立场,如不慎侵犯到您的版权利益,请联系79283999@qq.com删除。

本站资料仅供学习交流使用请勿商业运营,严禁从事违法,侵权等任何非法活动,否则后果自负!
THE END
喜欢就支持一下吧
点赞9赞赏 分享
评论 抢沙发

请登录后发表评论

    暂无评论内容