

新闻资讯
技术学院视频|直播|、语音通话、在线游戏这类场景,延迟比可靠性更重要。UDP 不建立连接、不重传、不排序,发出去就完事,自然快。但网络抖动或拥塞时,ReadFromUDP 可能收不到某些包,应用层得自己处理乱序、丢包、重复——比如用序列号+时间戳判断是否过期,或直接忽略旧包。
net.ListenUDP 启动服务,WriteToUDP 发送,无连接状态管理sendto: message too long 或静默丢弃)HTTP、RPC、数据库连接、文件上传都依赖 TCP。Go 的 net.Listen + listener.Accept() 模型天然匹配:每个连接对应一个 net.Conn,读写自动保证字节流顺序和完整性。但连接建立(三次握手)、断开(四次挥手)、拥塞控制会引入额外延迟。
TIME_WAIT 状态数,需调优 net.ipv4.tcp_tw_reuse 或复用连接conn.Read() 可能一次读到多个逻辑消息,或半个消息,得靠协议头(如长度字段)拆包SetKeepAlive
如果你打算自己实现重传、确认、滑动窗口、心跳保活、加密协商……那其实是在重造 TCP 轮子,除非有极致性能或特殊定制需求(如 QUIC 底层用 UDP 但上层自建可靠机制),否则大概率得不偿失。
sendto 返回成功只表示进内核发送队列,不等于对方收到time.AfterFunc 定时 close 连接,不如直接用 SetDeadline 让底层自动处理TCP 错误往往发生在连接生命周期中:read: connection reset by peer 表示对方强制断连;i/o timeout 多半是 SetReadDeadline 触发;而 UDP 几乎只在绑定端口失败时报错(bind: address already in use),收发阶段出错通常返回 n, addr, err 中的 err,且多为临时性错误(如 io.ErrShortBuffer 表示缓冲区不够存完整包)。
conn, err := net.ListenUDP("udp", &net.UDPAddr{Port: 8080})
if err != nil {
log.Fatal(err) // 这里可能失败:端口被占、权限不足
}
defer conn.Close()
buf := make([]byte, 1500)
for {
n, addr, err := conn.ReadFromUDP(buf)
if err != nil {
if netErr, ok := err.(net.Error); ok && netErr.Temporary() {
continue // 临时错误,比如 ICMP 报错,可跳过
}
log.Printf("UDP read error: %v", err)
break
}
// 处理 buf[:n]
}
UDP 的轻量是真轻量,但容错和逻辑复杂度全甩给业务代码;TCP 的“重”是把共性难题收归内核,换你省心。选哪个,本质是权衡「谁来承担协议复杂度」。