
网络编程(tcp,udp,http)
UDP客户端
发布日期:2021-05-06 23:58:22
浏览次数:23
分类:精选文章
本文共 5074 字,大约阅读时间需要 16 分钟。
1.Go语言实现TCP通信
一个TCP服务端可以同时连接很多个客户端,例如世界各地的用户使用自己电脑上的浏览器访问淘宝网。因为Go语言中创建多个goroutine实现并发非常方便和高效,所以我们可以每建立一次链接就创建一个goroutine去处理。
TCP服务端程序的处理流程:
- 监听端口
- 接收客户端请求建立链接
- 创建goroutine处理链接。
我们使用Go语言的net包实现的TCP服务端代码如下:
func process (coon net.Conn){ for { //3.与客户端通信 var tmp [128]byte n,err :=coon.Read(tmp[:]) if err != nil { fmt.Println("Read Form coon ,err: ",err) return } fmt.Println(string(tmp[:n])) }}func main(){ //1.本地端口启动服务 listener,err :=net.Listen("tcp","127.0.0.1:20000") if err != nil { fmt.Println("start server failed ,err: ",err) return } //2.等待别人来跟我建立链接 for { coon ,err := listener.Accept() if err != nil { fmt.Println("listener.Accept() ,err: ",err) return } go process(coon) }}
一个TCP客户端进行TCP通信的流程如下:
- 建立与服务端的链接
- 进行数据收发
- 关闭链接
使用Go语言的net包实现的TCP客户端代码如下:
func main() { //1、与sever端建立链接 coon ,err := net.Dial("tcp","127.0.0.1:20000") if err != nil { fmt.Println("dial 127.0.0.1:20000 failed ,err: ",err) return } //2.发送数据,监听控制台输入,没有用fmt.Scanln,因为不支持空格 for { fmt.Println("请说话:") reader := bufio.NewReader(os.Stdin) content ,_:=reader.ReadString('\n') msg := strings.TrimSpace(content) if msg == "exit" { break } _,err :=coon.Write([]byte(msg)) if err != nil { fmt.Println("客户端发送数据 failed ,err: ",err) return } } error := coon.Close() if error != nil { fmt.Println("coon.Close()failed ,err: ",err) return }}
2.Go语言实现UDP通信
UDP协议
UDP协议(User Datagram Protocol)中文名称是用户数据报协议,是OSI(Open System Interconnection,开放式系统互联)参考模型中一种无连接的传输层协议,不需要建立连接就能直接进行数据发送和接收,属于不可靠的、没有时序的通信,但是UDP协议的实时性比较好,通常用于视频直播相关领域。
UDP服务端
func main() { listen, err:= net.ListenUDP("udp",&net.UDPAddr{ IP: net.IPv4(127,0,0,1), Port: 40000, }) if err != nil { fmt.Println("listen failed, err:", err) return } //接受对方回传消息 var data [1024]byte for { n,udpAddr ,err := listen.ReadFromUDP(data[:]) //ReadFromUDP只接受byte切片参数 if err != nil { fmt.Println("read udp failed, err:", err) continue } fmt.Printf("data:%v addr:%v count:%v\n", string(data[:n]), udpAddr, n) //addr 就知道了是谁发过来的数据,回复消息 msg := strings.ToUpper(string(data[:n])) _,err = listen.WriteToUDP([]byte(msg),udpAddr) if err != nil { fmt.Println("write to udp failed, err:", err) continue } }}
net.ListenUDP 参数,传“udp” , 和UDPAddr的一个指针


func main() { udpClientCoon, err :=net.DialUDP("udp",nil,&net.UDPAddr{ IP: net.IPv4(127,0,0,1), Port: 40000, }) if err != nil { fmt.Println("连接服务端失败,err:", err) return } defer udpClientCoon.Close() reader :=bufio.NewReader(os.Stdin) var reply [1024]byte for { fmt.Println("请输入内容:") msg, _ :=reader.ReadString('\n') udpClientCoon.Write([]byte(msg)) //收到server端回复的消息 n, _ ,err :=udpClientCoon.ReadFromUDP(reply[:]) if err != nil { fmt.Println("接收数据失败,err:", err) return } fmt.Println("收到sever回复的消息:",string(reply[:n])) }}
net.DialUDP的()得到对方的coon, 将来回复消息,直接coon,Write()即可,不需要和sever端一样需要传addr地址。

3. net/http
Go语言内置的net/http
包提供了HTTP客户端和服务端的实现。
1.htttp服务端
// 必须符合 http.HandleFunc 所要求的函数参数。func HandleIndexPage(w http.ResponseWriter , r *http.Request) { context,err := ioutil.ReadFile("D:/go/path/src/test/test/net/http/sever/index.html") if err != nil { fmt.Printf("打开文件失败, err:%v\n", err) return } w.Write(context)}func main() { http.HandleFunc("/",HandleIndexPage) //访问根目录调用的函数 err := http.ListenAndServe("127.0.0.1:9091",nil) if err != nil { fmt.Printf("http server failed, err:%v\n", err) return }}
ListenAndServe使用指定的监听地址和处理器启动一个HTTP服务端。处理器参数通常是nil,这表示采用包变量DefaultServeMux作为处理器。
http.HandleFunc()第一个参数为路径,第二个参数为处理函数名,(对参数有要求)

2.http客户端(带参数)
package mainimport ( "fmt" "io/ioutil" "net/http")func main() { resp ,err := http.Get("http://127.0.0.1:9091/?name=guo&age=25") 带参数的GET请求 if err != nil { fmt.Printf("get failed, err:%v\n", err) return } defer resp.Body.Close() content , err := ioutil.ReadAll(resp.Body) if err != nil { fmt.Printf("read from resp.Body failed, err:%v\n", err) return } fmt.Println(string(content))}
sever端
func HandleIndexPage2(w http.ResponseWriter , r *http.Request) { queryParam :=r.URL.Query() 解析参数 name := queryParam.Get("name") age := queryParam.Get("age") fmt.Println("name:",name) //name: guo fmt.Println("age:",age) //age: 25 fmt.Println(r.URL) ///?name=guo&age=25 fmt.Println(r.Method) //GET fmt.Println(ioutil.ReadAll(r.Body)) //GET GET请求在head里,Body里没东西 w.Write([]byte("hello"))}func main() { http.HandleFunc("/",HandleIndexPage1) err := http.ListenAndServe("127.0.0.1:9091",nil) if err != nil { fmt.Printf("http server failed, err:%v\n", err) return }}
发表评论
最新留言
能坚持,总会有不一样的收获!
[***.219.124.196]2025年03月25日 20时43分09秒
关于作者

喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!
推荐文章
Oracle常用SQL
2019-03-05
技术美术面试问题整理
2019-03-05
Redis分布式锁原理
2019-03-05
【备份】求极限笔记
2019-03-05
C++学习记录 四、基于多态的企业职工系统
2019-03-05
C++学习记录 五、C++提高编程(2)
2019-03-05
面试问道nginx优化怎么做的
2019-03-05
自学linux毕业shell面试题
2019-03-05
4 Java 访问控制符号的范围
2019-03-05
第9章 - 有没有替代原因(检验证据)
2019-03-05
VUE3(八)setup与ref函数
2019-03-05
Vue之Element标签页保留用户操作缓存。
2019-03-05
智能合约开发实践(1)
2019-03-05
2. Spring Boot学习——Yaml等配置文件教程
2019-03-05
MATLAB——操作矩阵的常用函数
2019-03-05
CMake自学记录,看完保证你知道CMake怎么玩!!!
2019-03-05
Eigen库中vector.transpose()函数什么意思
2019-03-05
ORB-SLAM2:LocalMapping线程学习随笔【李哈哈:看看总有收获篇】
2019-03-05
ORB-SLAM2:LoopClosing线程学习随笔【李哈哈:看看总有收获篇】
2019-03-05
牛客练习赛56 D 小翔和泰拉瑞亚(线段树)
2019-03-05