Skip to content

UDP尝试

使用Golang分别编写UDP服务端和客户端,如下所示。

服务端

go
package main

import (
	"fmt"
	"net"
)

func main() {
	listen, err := net.ListenUDP("udp", &net.UDPAddr{
		IP:   net.IPv4(127, 0, 0, 1),
		Port: 30000,
	})
	if err != nil {
		fmt.Println("listen failed, err:", err)
		return
	}
	defer listen.Close()
	for {
		var data [1024]byte
		n, addr, err := listen.ReadFromUDP(data[:]) // 接收数据
		if err != nil {
			fmt.Println("read udp failed, err:", err)
			continue
		}
		fmt.Printf("data:%v addr:%v count:%v
", string(data[:n]), addr, n)
		_, err = listen.WriteToUDP(data[:n], addr) // 发送数据
		if err != nil {
			fmt.Println("write to udp failed, err:", err)
			continue
		}
	}
}

客户端

go
package main

import (
	"fmt"
	"net"
	"time"
)

func main() {
	socket, err := net.DialUDP("udp", nil, &net.UDPAddr{
		IP:   net.IPv4(127, 0, 0, 1),
		Port: 30000,
	})
	if err != nil {
		fmt.Println("连接服务端失败,err:", err)
		return
	}
	defer socket.Close()
	sendData := []byte("Hello server")
	for {
		_, err = socket.Write(sendData) // 发送数据
		if err != nil {
			fmt.Println("发送数据失败,err:", err)
			return
		}
		data := make([]byte, 4096)
		n, remoteAddr, err := socket.ReadFromUDP(data) // 接收数据
		if err != nil {
			fmt.Println("接收数据失败,err:", err)
			return
		}
		fmt.Printf("recv:%v addr:%v count:%v
", string(data[:n]), remoteAddr, n)
		time.Sleep(2 * time.Second)
		break
	}
}

Wireshark抓包

上述代码,先启动服务端,再运行客户端。客户端启动后,会连接服务端,然后向服务端发送字符串"Hello server"。服务端接收到客户端数据之后,会将客户端发来的数据原封不动地发回去。

通过Wireshark抓包,可以清晰的看到如下过程,共有两个UDP数据包。

image-20221205215111613

可以看到,UDP直接对数据进行了发送,且未对数据进行加密。

如果想对UDP数据进行加密,可以使用DTLS协议,这是针对UDP的认证和密钥协商协议。

总结

P2P网络常使用UDP连接传输,根据本次尝试,UDP服务端是可以获取客户端的addr和port的。

后面看一下P2P网络的发现机制是怎么做的。