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数据包。
可以看到,UDP直接对数据进行了发送,且未对数据进行加密。
如果想对UDP数据进行加密,可以使用DTLS协议,这是针对UDP的认证和密钥协商协议。
总结
P2P网络常使用UDP连接传输,根据本次尝试,UDP服务端是可以获取客户端的addr和port的。
后面看一下P2P网络的发现机制是怎么做的。