

新闻资讯
技术学院UDP客户端用net.DialUDP复用连接收发,需解析目标地址、设读超时、处理无响应;服务端用net.ListenUDP监听,每包启goroutine并发处理;跨机丢包主因防火墙或绑定127.0.0.1;需按最大包长分配缓冲区并自行定义消息边界。
Go 的 net.D 可以复用同一个连接收发,但要注意它默认不支持广播,且必须指定目标地址。常见错误是调用
ialUDPWriteToUDP 时传入了 nil 地址,导致 panic:「write: invalid argument」。
实操建议:
net.ResolveUDPAddr 解析目标地址,避免手动拼接 "127.0.0.1:8080" 字符串出错SetReadDeadline),否则 ReadFromUDP 会永久阻塞conn, _ := net.DialUDP("udp", nil, &net.UDPAddr{IP: net.IPv4(127, 0, 0, 1), Port: 8080})
defer conn.Close()
conn.SetReadDeadline(time.Now().Add(2 * time.Second))
conn.Write([]byte("hello"))
buf := make([]byte, 1024)
n, addr, _ := conn.ReadFromUDP(buf)
fmt.Printf("received %s from %v\n", string(buf[:n]), addr)用 net.ListenUDP 启动的服务端本身是单 goroutine 的,ReadFromUDP 是阻塞调用。若在主 goroutine 中循环读取,就无法同时处理多个请求——这不是 TCP,没有 accept 概念,但也不意味着天然并发。
关键点:
ReadFromUDP 被卡住ReadFromUDP 返回的 n,避免读到空数据或越界写入":8080" 即可监听所有网卡,但生产环境建议绑定具体 IPaddr := &net.UDPAddr{Port: 8080}
conn, _ := net.ListenUDP("udp", addr)
defer conn.Close()
for {
buf := make([]byte, 1024)
n, clientAddr, err := conn.ReadFromUDP(buf)
if err != nil {
continue
}
go func(n int, addr *net.UDPAddr) {
// 处理逻辑,例如回写
conn.WriteToUDP([]byte("ack"), addr)
}(n, clientAddr)
}最常见原因是防火墙拦截或端口未开放。Linux 上可通过 sudo ufw status 查看;macOS 检查「系统设置 → 隐私与安全性 → 防火墙」;Windows 则看「高级安全 Windows 防火墙」入站规则。
其他可能性:
127.0.0.1,只能本机访问;应改用 "" 或 0.0.0.0
Go 的 UDP 连接底层复用系统 socket,但每次 ReadFromUDP 都需提供足够大的缓冲区。若固定用 make([]byte, 1024),而对方发来 2KB 数据,多余部分直接被截断——UDP 不重传,也不会通知你丢包。
稳妥做法:
sync.Pool 复用切片可降低 GC 压力make([]byte, ...),尤其高并发场景下易触发频繁堆分配复杂点在于:UDP 没有消息边界保证,应用层需自行定义帧头(比如前 4 字节存长度),否则无法判断一个完整逻辑包是否收齐。这点常被忽略,直到出现粘包或半包问题才意识到。