成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

一篇文章帶你了解Go語(yǔ)言基礎(chǔ)之網(wǎng)絡(luò)編程

開(kāi)發(fā) 前端
本次章節(jié)我們講述了什么是TCP,什么是UDP。并且編寫(xiě)了代碼如何實(shí)現(xiàn)TCP服務(wù)端,TCP客戶(hù)端,UDP服務(wù)端,UDP客戶(hù)端。講述了為什么會(huì)出現(xiàn)粘包,該怎么解決粘包。

[[361058]]

前言

Hi,大家好呀,我是碼農(nóng),星期八,我們身處21世紀(jì),我們的世界已經(jīng)在不知不覺(jué)中,就像很多網(wǎng)一樣在互聯(lián)互通。

互聯(lián)網(wǎng)是一個(gè)統(tǒng)稱(chēng),目前比較常用的有TCP,UDP協(xié)議。

當(dāng)然,還有很多其他的協(xié)議,但是本次主要講最常用的TCP和UDP協(xié)議。

socker編程

我們所學(xué)的TCP和UDP,統(tǒng)稱(chēng)為Socker編程,也叫做套接字編程。

多臺(tái)機(jī)器要實(shí)現(xiàn)互相通訊,其實(shí)是一個(gè)非常復(fù)雜的過(guò)程,底層從鋪設(shè)網(wǎng)線,網(wǎng)線接口,交換機(jī),路由器,在到規(guī)定各種協(xié)議。

再到應(yīng)用層QQ,微信等軟件。

如果沒(méi)有一套標(biāo)準(zhǔn),每次使用都要自己去實(shí)現(xiàn),可能每個(gè)程序員都不是掉頭發(fā)那么簡(jiǎn)單了!

有了Socker之后,Socker會(huì)在應(yīng)用層之前,將各種繁瑣的的底層操作隱藏,我們可能只需要Socker.TCP就實(shí)現(xiàn)了TCP協(xié)議的通訊。

Go語(yǔ)言TCP

TCP屬于穩(wěn)定的,可靠的長(zhǎng)連接,

既然要涉及通訊,必然有兩個(gè)終端,最起碼一個(gè)是服務(wù)端,一個(gè)是客戶(hù)端,就像我們的淘寶,我們每次打開(kāi)淘寶,都要去鏈接它,當(dāng)然,淘寶可不直接是TCP。

服務(wù)端

在Go中實(shí)現(xiàn)服務(wù)端,并且在服務(wù)端并發(fā)很簡(jiǎn)單,只需要將每個(gè)連接讓一個(gè)協(xié)程處理即可!

代碼

  1. package main 
  2.  
  3. import ( 
  4.     "bufio" 
  5.     "fmt" 
  6.     "net" 
  7.  
  8. func process(conn net.Conn) { 
  9.     defer conn.Close() 
  10.     for { 
  11.         reader := bufio.NewReader(conn) 
  12.         buf := make([]byte, 128) 
  13.         n, err := reader.Read(buf) 
  14.         if err != nil { 
  15.             fmt.Println("數(shù)據(jù)讀取失敗", err) 
  16.             return 
  17.         } 
  18.         recvStr := string(buf[:n]) 
  19.         fmt.Println("客戶(hù)端發(fā)送過(guò)來(lái)的值:", recvStr) 
  20.  
  21. func main() { 
  22.     lister, err := net.Listen("tcp", "0.0.0.0:8008"
  23.     if err != nil { 
  24.         fmt.Println("連接失敗", err) 
  25.     for { 
  26.         fmt.Println("等待建立連接,此時(shí)會(huì)阻塞住"
  27.         conn, err := lister.Accept() //等待建立連接 
  28.         fmt.Println("連接建立成功,繼續(xù)"
  29.         if err != nil { 
  30.             fmt.Println("建立連接失敗", err) 
  31.             //繼續(xù)監(jiān)聽(tīng)下次鏈接 
  32.             continue 
  33.         } 
  34.         go process(conn) 

客戶(hù)端

客戶(hù)端就很簡(jiǎn)單了,相對(duì)來(lái)說(shuō)是不需要并發(fā)的,只需要連接就行。

代碼

  1. package main 
  2.  
  3. import ( 
  4.     "bufio" 
  5.     "fmt" 
  6.     "net" 
  7.     "os" 
  8.  
  9. //客戶(hù)端 
  10. func main() { 
  11.     conn, err := net.Dial("tcp", "192.168.10.148:8008"
  12.     if err != nil { 
  13.         fmt.Println("連接服務(wù)器失敗",err) 
  14.     defer conn.Close() 
  15.     inputReader:=bufio.NewReader(os.Stdin) 
  16.     for
  17.         fmt.Println(":"
  18.         input,_:=inputReader.ReadString('\n'
  19.         _, err = conn.Write([]byte(input)) 
  20.         if err != nil { 
  21.             fmt.Println("發(fā)送成功"
  22.         } 

執(zhí)行結(jié)果

就這樣,我們實(shí)現(xiàn)了服務(wù)端并發(fā)的處理所有客戶(hù)端的請(qǐng)求。


粘包

我們先看一下什么是粘包。

服務(wù)端

  1. package main 
  2.  
  3. import ( 
  4.     "bufio" 
  5.     "fmt" 
  6.     "io" 
  7.     "net" 
  8.  
  9. func process(conn net.Conn) { 
  10.     defer conn.Close() 
  11.     reader := bufio.NewReader(conn) 
  12.     buf := make([]byte, 1024) 
  13.     for { 
  14.         n, err := reader.Read(buf) 
  15.         //讀完了 
  16.         if err == io.EOF { 
  17.             fmt.Println("讀完了"
  18.             break 
  19.         } 
  20.         //讀錯(cuò)了 
  21.         if err != nil { 
  22.             fmt.Println("數(shù)據(jù)讀取失敗", err) 
  23.             return 
  24.         } 
  25.         recvStr := string(buf[:n]) 
  26.         fmt.Println("客戶(hù)端發(fā)送過(guò)來(lái)的值:", recvStr) 
  27.  
  28. func main() { 
  29.     lister, err := net.Listen("tcp", "0.0.0.0:8008"
  30.     if err != nil { 
  31.         fmt.Println("連接失敗", err) 
  32.         return 
  33.     defer lister.Close() 
  34.     for { 
  35.         fmt.Println("等待建立連接,此時(shí)會(huì)阻塞住"
  36.         conn, err := lister.Accept() //等待建立連接 
  37.         fmt.Println("連接建立成功,繼續(xù)"
  38.         if err != nil { 
  39.             fmt.Println("建立連接失敗", err) 
  40.             //繼續(xù)監(jiān)聽(tīng)下次鏈接 
  41.             continue 
  42.         } 
  43.         go process(conn) 

客戶(hù)端

  1. package main 
  2.  
  3. import ( 
  4.     "fmt" 
  5.     "net" 
  6.  
  7. //客戶(hù)端 
  8. func main() { 
  9.     conn, err := net.Dial("tcp", "192.168.10.148:8008"
  10.     if err != nil { 
  11.         fmt.Println("連接服務(wù)器失敗", err) 
  12.     defer conn.Close() 
  13.     for i := 0; i < 10; i++ { 
  14.         sendStr := "hello world ads asdf asd fads fadsf ads ads asd asd ads " 
  15.         conn.Write([]byte(sendStr)) 
  16.         time.Sleep(time.Second

注意:18行代碼睡眠了1s

執(zhí)行結(jié)果


如果我注釋了第18行代碼


執(zhí)行結(jié)果


直接都淦到一行了,what?這是啥情況,不應(yīng)該跟原來(lái)一樣嗎???

每發(fā)送一個(gè)值,那邊就接收一下,這怎么整到一塊了!!!

原因

主要原因是因?yàn)槲覀兪菓?yīng)用層軟件,是跑在操作系統(tǒng)之上的軟件,當(dāng)我們向服務(wù)器發(fā)送一個(gè)數(shù)據(jù)時(shí),是調(diào)用操作系統(tǒng)的相關(guān)接口發(fā)送的,操作系統(tǒng)再經(jīng)過(guò)各種復(fù)雜的操作,發(fā)送到對(duì)方機(jī)器

但是操作系統(tǒng)有一個(gè)發(fā)送數(shù)據(jù)緩沖區(qū),默認(rèn)情況如果緩沖區(qū)是有大小的,如果緩沖區(qū)沒(méi)滿(mǎn),是不會(huì)發(fā)送數(shù)據(jù)的,所以上述客戶(hù)端在發(fā)送數(shù)據(jù)時(shí),系統(tǒng)的緩沖區(qū)都沒(méi)滿(mǎn),一直壓在了操作系統(tǒng)的緩沖區(qū)中,最后發(fā)現(xiàn)沒(méi)數(shù)據(jù)了,才一次都發(fā)送到服務(wù)端

但是為什么sleep(1)又管用了呢?這是因?yàn)榫彌_區(qū)不止一個(gè)程序在用,1s的時(shí)間足夠其他程序?qū)⒕彌_區(qū)打滿(mǎn),然后各自發(fā)各自的數(shù)據(jù),這也是為什么第一次操作沒(méi)問(wèn)題,第二次有問(wèn)題,因?yàn)榈诙稳慷际俏覀兛蛻?hù)端打滿(mǎn)的


解決粘包

工具函數(shù)

我們將解包封包的函數(shù)封裝一下

  1. socker_sitck/stick.go 

 

  1. package socker_stick 
  2.  
  3. import ( 
  4.     "bufio" 
  5.     "bytes" 
  6.     "encoding/binary" 
  7.     "fmt" 
  8.  
  9. //Encode 將消息編碼 
  10. func Encode(message string) ([]byte, error) { 
  11.     length := int32(len(message)) 
  12.     var pkg = new(bytes.Buffer) 
  13.     //寫(xiě)入消息頭 
  14.     err := binary.Write(pkg, binary.LittleEndian, length) 
  15.     if err != nil { 
  16.         fmt.Println("寫(xiě)入消息頭失敗", err) 
  17.         return nil, err 
  18.     //寫(xiě)入消息實(shí)體 
  19.     err = binary.Write(pkg, binary.LittleEndian, []byte(message)) 
  20.     if err != nil { 
  21.         fmt.Println("寫(xiě)入消息實(shí)體失敗", err) 
  22.         return nil, err 
  23.     return pkg.Bytes(), nil 
  24.  
  25. //Decode解碼消息 
  26. func Decode(reader *bufio.Reader) (string, error) { 
  27.     //讀取信息長(zhǎng)度 
  28.     lengthByte, _ := reader.Peek(4) 
  29.     lengthBuff := bytes.NewBuffer(lengthByte) 
  30.     var length int32 
  31.     err := binary.Read(lengthBuff, binary.LittleEndian, &length) 
  32.     if err != nil { 
  33.         return "", err 
  34.     //BuffRead 返回緩沖區(qū)現(xiàn)有的可讀的字節(jié)數(shù) 
  35.     if int32(reader.Buffered()) < length+4 { 
  36.         return "", err 
  37.     pack := make([]byte, int(4+length)) 
  38.     _, err = reader.Read(pack) 
  39.     if err != nil { 
  40.         return "", err 
  41.     return string(pack[4:]), nil 

服務(wù)端

  1. package main 
  2.  
  3. import ( 
  4.     "a3_course/socker_stick" 
  5.     "bufio" 
  6.     "fmt" 
  7.     "io" 
  8.     "net" 
  9.  
  10. func process(conn net.Conn) { 
  11.     defer conn.Close() 
  12.     reader := bufio.NewReader(conn) 
  13.  
  14.     for { 
  15.         msg, err := socker_stick.Decode(reader) 
  16.         //讀完了 
  17.         if err == io.EOF { 
  18.             fmt.Println("讀完了"
  19.             break 
  20.         } 
  21.         //讀錯(cuò)了 
  22.         if err != nil { 
  23.             fmt.Println("數(shù)據(jù)讀取失敗", err) 
  24.             return 
  25.         } 
  26.  
  27.         fmt.Println("客戶(hù)端發(fā)送過(guò)來(lái)的值:", msg) 
  28.  
  29. func main() { 
  30.     lister, err := net.Listen("tcp", "0.0.0.0:8008"
  31.     if err != nil { 
  32.         fmt.Println("連接失敗", err) 
  33.         return 
  34.     defer lister.Close() 
  35.     for { 
  36.         fmt.Println("等待建立連接,此時(shí)會(huì)阻塞住"
  37.         conn, err := lister.Accept() //等待建立連接 
  38.         fmt.Println("連接建立成功,繼續(xù)"
  39.         if err != nil { 
  40.             fmt.Println("建立連接失敗", err) 
  41.             //繼續(xù)監(jiān)聽(tīng)下次鏈接 
  42.             continue 
  43.         } 
  44.         go process(conn) 

客戶(hù)端

  1. package main 
  2.  
  3. import ( 
  4.     "a3_course/socker_stick" 
  5.     "fmt" 
  6.     "net" 
  7.  
  8. //客戶(hù)端 
  9. func main() { 
  10.     conn, err := net.Dial("tcp", "192.168.10.148:8008"
  11.     if err != nil { 
  12.         fmt.Println("連接服務(wù)器失敗", err) 
  13.     defer conn.Close() 
  14.     for i := 0; i < 10; i++ { 
  15.         sendStr := "hello world ads asdf asd fads fadsf ads ads asd asd ads " 
  16.         data, err := socker_stick.Encode(sendStr) 
  17.         if err != nil { 
  18.             fmt.Println("編碼失敗",err) 
  19.             return 
  20.         } 
  21.         conn.Write(data) 
  22.         //time.Sleep(time.Second

執(zhí)行結(jié)果


這次真的不管執(zhí)行幾次,都是這樣的結(jié)果

對(duì)了,只有TCP才有粘包

Go語(yǔ)言UDP

UDP是一個(gè)無(wú)連接協(xié)議,客戶(hù)端不會(huì)在乎服務(wù)端有沒(méi)有問(wèn)題,客戶(hù)端只管發(fā),通常用于實(shí)時(shí)性比較高的領(lǐng)域

例如直播行業(yè)

服務(wù)端

  1. package main 
  2.  
  3. import ( 
  4.     "fmt" 
  5.     "net" 
  6.  
  7. func main() { 
  8.     listen, err := net.ListenUDP("udp", &net.UDPAddr{ 
  9.         IP:   net.IPv4(0, 0, 0, 0), 
  10.         Port: 8009, 
  11. }) 
  12.     if err != nil { 
  13.         panic(fmt.Sprintf("udp啟動(dòng)失敗,err:%v", err)) 
  14.     defer listen.Close() 
  15.     for
  16.         var data = make([]byte,1024) 
  17.         n, addr, err := listen.ReadFromUDP(data) 
  18.         if err != nil { 
  19.             panic(fmt.Sprintf("讀取數(shù)據(jù)失敗,err:%v", err)) 
  20.         } 
  21.         fmt.Println(string(data[:n]),addr,n) 

客戶(hù)端

  1. package main 
  2.  
  3. import ( 
  4.     "fmt" 
  5.     "net" 
  6.  
  7. func main() { 
  8.     socker, err := net.DialUDP("udp", nil, &net.UDPAddr{ 
  9.         IP:   net.IPv4(0, 0, 0, 0), 
  10.         Port: 8009, 
  11. }) 
  12.     if err != nil { 
  13.         panic(fmt.Sprintf("連接服務(wù)器失敗,err:%v", err)) 
  14.     defer socker.Close() 
  15.     sendStr:="你好呀" 
  16.     _, err = socker.Write([]byte(sendStr)) 
  17.     if err != nil { 
  18.         panic(fmt.Sprintf("數(shù)據(jù)發(fā)送失敗,err:%v", err)) 

執(zhí)行結(jié)果


總結(jié)

本次章節(jié)我們講述了什么是TCP,什么是UDP。

并且編寫(xiě)了代碼如何實(shí)現(xiàn)TCP服務(wù)端,TCP客戶(hù)端,UDP服務(wù)端,UDP客戶(hù)端。

講述了為什么會(huì)出現(xiàn)粘包,該怎么解決粘包。

逆水行舟,不進(jìn)則退!

 

責(zé)任編輯:姜華 來(lái)源: Go語(yǔ)言進(jìn)階學(xué)習(xí)
相關(guān)推薦

2020-11-05 09:58:16

Go語(yǔ)言Map

2022-02-16 10:03:06

對(duì)象接口代碼

2020-10-22 08:33:22

Go語(yǔ)言

2020-11-11 10:52:54

Go語(yǔ)言C語(yǔ)言

2021-10-09 07:10:31

Go語(yǔ)言基礎(chǔ)

2022-04-27 10:01:43

切片Go封裝

2020-10-23 08:38:19

Go語(yǔ)言

2020-10-25 07:33:13

Go語(yǔ)言

2020-12-09 09:59:32

Go語(yǔ)言技術(shù)

2020-12-27 10:15:44

Go語(yǔ)言channel管道

2020-12-07 05:59:02

語(yǔ)言Go接口

2021-10-30 10:43:04

語(yǔ)言Go函數(shù)

2021-11-03 10:02:07

Go基礎(chǔ)函數(shù)

2021-09-29 10:00:07

Go語(yǔ)言基礎(chǔ)

2021-10-13 10:00:52

Go語(yǔ)言基礎(chǔ)

2020-10-22 11:15:47

Go語(yǔ)言變量

2021-10-16 10:17:51

Go語(yǔ)言數(shù)據(jù)類(lèi)型

2020-12-23 08:39:11

Go語(yǔ)言基礎(chǔ)技術(shù)

2021-02-20 10:06:14

語(yǔ)言文件操作

2021-01-13 08:40:04

Go語(yǔ)言文件操作
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 91久久久久久久久 | av网站在线播放 | 国产激情视频在线免费观看 | 久久久综合网 | 国产精品毛片无码 | 在线不卡视频 | 国产第一页在线播放 | 在线免费看毛片 | 亚洲日韩中文字幕一区 | 一区二区精品 | 精品一区二区三区不卡 | 色婷婷一区二区三区四区 | 一区二区三区高清在线观看 | 中文字幕亚洲区一区二 | 国产一区二区在线看 | 精品欧美一区二区三区久久久 | 国产精品久久久久久久久久不蜜臀 | 欧美日韩精品免费 | 欧美激情国产日韩精品一区18 | 久久999| 国产精品久久久久久吹潮 | 成人影院网站ww555久久精品 | 欧区一欧区二欧区三免费 | 久久久久91 | 盗摄精品av一区二区三区 | 日本三级视频 | 日韩在线播放av | 中文字幕日韩欧美一区二区三区 | 日韩欧美国产综合 | 国产精品成人一区二区三区吃奶 | 久久久久国产一级毛片 | 亚洲精品视频免费观看 | 日韩一区二区在线观看视频 | 国产精品视频网 | 荷兰欧美一级毛片 | 成人在线视频观看 | 欧美一区二区久久 | 中文字幕一区二区三区在线观看 | 999热精品 | 久久久精选 | 黄色毛片在线看 |