0

0

UDP 通信双向实现:Go 中 ListenUDP 的正确用法

心靈之曲

心靈之曲

发布时间:2025-12-26 19:50:01

|

156人浏览过

|

来源于php中文网

原创

UDP 通信双向实现:Go 中 ListenUDP 的正确用法

udp 是无连接协议,`net.listenudp` 返回的连接不维护对端地址,直接调用 `write()` 无法自动回发数据;必须使用 `readfromudp` 获取客户端地址,并通过 `writetoudp` 显式指定目标地址才能实现双向通信。

在 Go 中使用 net.ListenUDP 创建 UDP 服务端时,一个常见误区是误以为它像 TCP 连接一样具备“会话上下文”——即能自动记住发送方地址并支持无参数 Write()。但事实并非如此:UDP 本身无连接、无状态,*net.UDPConn 的 Write() 方法没有默认目标地址,调用它等价于向 nil 地址发送数据,内核将静默丢弃(Wireshark 抓不到包也正源于此)。

要实现真正的双向 UDP 通信,关键在于:
✅ 使用 ReadFromUDP(buf []byte) (n int, addr *net.UDPAddr, err error) —— 它不仅读取数据,还返回发送方的完整地址(IP + 端口);
✅ 使用 WriteToUDP(b []byte, addr *net.UDPAddr) (n int, err error) —— 显式将响应发回该地址。

以下是修正后的 Node 1(服务端)代码:

func main() {
    addr := &net.UDPAddr{Port: 7000, IP: net.ParseIP("127.0.0.1")}
    conn, err := net.ListenUDP("udp", addr)
    if err != nil {
        panic(err)
    }
    defer conn.Close()

    fmt.Println("UDP server listening on :7000")
    for {
        buf := make([]byte, 1024) // 建议增大缓冲区,避免截断
        n, clientAddr, err := conn.ReadFromUDP(buf)
        if err != nil {
            log.Printf("Read error: %v", err)
            continue
        }
        msg := strings.TrimSpace(string(buf[:n]))
        fmt.Printf("Received from %s: %q\n", clientAddr, msg)

        // 显式回发到 clientAddr
        _, err = conn.WriteToUDP([]byte("sending back"), clientAddr)
        if err != nil {
            log.Printf("WriteTo error to %s: %v", clientAddr, err)
        }
    }
}

同时,Node 2(客户端)可保持简洁,但建议补全错误处理:

Catimind
Catimind

专为行业应用打造的AI生产力工具

下载
func main() {
    conn, err := net.Dial("udp", "127.0.0.1:7000")
    if err != nil {
        panic(err)
    }
    defer conn.Close()

    _, _ = conn.Write([]byte("first send"))

    buf := make([]byte, 1024)
    n, _ := conn.Read(buf)
    fmt.Println("Response:", string(buf[:n]))
}

⚠️ 注意事项:

  • ReadFromUDP 和 WriteToUDP 是 UDP 服务端双向通信的标准配对,不可替换为 Read/Write;
  • 缓冲区大小应足够容纳实际报文(UDP 单包通常 ≤ 65507 字节),示例中 10 字节极易导致截断;
  • 生产环境务必检查所有 I/O 错误(示例中已标注 TODO),忽略错误会导致静默失败;
  • 若需并发处理多个客户端,可启动 goroutine 处理每个 ReadFromUDP,但注意 *net.UDPConn 是线程安全的,可被多 goroutine 共享。

总结:UDP 的“双向通信”本质是基于地址的请求-响应模型,而非连接状态。Go 的 net 包严格遵循这一语义——没有魔法,只有显式的地址传递。掌握 ReadFromUDP/WriteToUDP 的使用,是构建健壮 UDP 应用的第一步。

相关专题

更多
scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

184

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

258

2023.10.25

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

311

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

513

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

46

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

183

2025.08.29

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

463

2023.08.10

tcp和udp的区别
tcp和udp的区别

TCP和UDP的区别,在连接性、可靠性、速度和效率、数据报大小以及适用场景等方面。本专题为大家提供tcp和udp的区别的相关的文章、下载、课程内容,供大家免费下载体验。

115

2023.07.25

虚拟号码教程汇总
虚拟号码教程汇总

本专题整合了虚拟号码接收验证码相关教程,阅读下面的文章了解更多详细操作。

25

2025.12.25

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
HTML5/CSS3/JavaScript/ES6入门课程
HTML5/CSS3/JavaScript/ES6入门课程

共102课时 | 6.5万人学习

前端基础到实战(HTML5+CSS3+ES6+NPM)
前端基础到实战(HTML5+CSS3+ES6+NPM)

共162课时 | 18.4万人学习

第二十二期_前端开发
第二十二期_前端开发

共119课时 | 12.1万人学习

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

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