目录
Go语言TCP" >Go语言TCP
服务端" >服务端
客户端
" >客户端
执行结果
" >执行结果
粘包
" >粘包
执行结果" >执行结果
原因" >原因
解决粘包
" >解决粘包
工具函数" >工具函数
Go语言UDP" >Go语言UDP
首页 后端开发 Golang 一篇文章带你了解Go语言基础之网络编程

一篇文章带你了解Go语言基础之网络编程

Jul 21, 2023 pm 04:19 PM
go语言

socker编程

我们所学的TCPUDP,统称为Socker编程,也叫做套接字编程

多台机器要实现互相通讯,其实是一个非常复杂的过程,底层从铺设网线网线接口交换机路由器,在到规定各种协议

再到应用层QQ微信等软件。

如果没有一套标准,每次使用都要自己去实现,可能每个程序员都不是掉头发那么简单了!

一篇文章带你了解Go语言基础之网络编程

有了Socker之后,Socker会在应用层之前,将各种繁琐的的底层操作隐藏,我们可能只需要Socker.TCP就实现了TCP协议的通讯。

Go语言TCP

TCP属于稳定的,可靠的长连接,

既然要涉及通讯,必然有两个终端,最起码一个是服务端,一个是客户端,就像我们的淘宝,我们每次打开淘宝,都要去链接它,当然,淘宝可不直接是TCP

服务端

在Go中实现服务端,并且在服务端并发很简单,只需要将每个连接让一个协程处理即可!

代码

package main


import (
    "bufio"
    "fmt"
    "net"
)


func process(conn net.Conn) {
    defer conn.Close()
    for {
        reader := bufio.NewReader(conn)
        buf := make([]byte, 128)
        n, err := reader.Read(buf)
        if err != nil {
            fmt.Println("数据读取失败", err)
            return
        }
        recvStr := string(buf[:n])
        fmt.Println("客户端发送过来的值:", recvStr)
}


}
func main() {
    lister, err := net.Listen("tcp", "0.0.0.0:8008")
    if err != nil {
        fmt.Println("连接失败", err)
}
    for {
        fmt.Println("等待建立连接,此时会阻塞住")
        conn, err := lister.Accept() //等待建立连接
        fmt.Println("连接建立成功,继续")
        if err != nil {
            fmt.Println("建立连接失败", err)
            //继续监听下次链接
            continue
        }
        go process(conn)
}
}
登录后复制

客户端

客户端就很简单了,相对来说是不需要并发的,只需要连接就行。

代码

package main


import (
    "bufio"
    "fmt"
    "net"
    "os"
)


//客户端
func main() {
    conn, err := net.Dial("tcp", "192.168.10.148:8008")
    if err != nil {
        fmt.Println("连接服务器失败",err)
}
    defer conn.Close()
    inputReader:=bufio.NewReader(os.Stdin)
    for{
        fmt.Println(":")
        input,_:=inputReader.ReadString('\n')
        _, err = conn.Write([]byte(input))
        if err != nil {
            fmt.Println("发送成功")
        }
}
}
登录后复制

执行结果

就这样,我们实现了服务端并发的处理所有客户端的请求。

一篇文章带你了解Go语言基础之网络编程

粘包

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

服务端

package main


import (
    "bufio"
    "fmt"
    "io"
    "net"
)


func process(conn net.Conn) {
    defer conn.Close()
    reader := bufio.NewReader(conn)
    buf := make([]byte, 1024)
    for {
        n, err := reader.Read(buf)
        //读完了
        if err == io.EOF {
            fmt.Println("读完了")
            break
        }
        //读错了
        if err != nil {
            fmt.Println("数据读取失败", err)
            return
        }
        recvStr := string(buf[:n])
        fmt.Println("客户端发送过来的值:", recvStr)
}


}
func main() {
    lister, err := net.Listen("tcp", "0.0.0.0:8008")
    if err != nil {
        fmt.Println("连接失败", err)
        return
}
    defer lister.Close()
    for {
        fmt.Println("等待建立连接,此时会阻塞住")
        conn, err := lister.Accept() //等待建立连接
        fmt.Println("连接建立成功,继续")
        if err != nil {
            fmt.Println("建立连接失败", err)
            //继续监听下次链接
            continue
        }
        go process(conn)
}
}
登录后复制

客户端

package main


import (
    "fmt"
    "net"
)


//客户端
func main() {
    conn, err := net.Dial("tcp", "192.168.10.148:8008")
    if err != nil {
        fmt.Println("连接服务器失败", err)
}
    defer conn.Close()
    for i := 0; i < 10; i++ {
        sendStr := "hello world ads asdf asd fads fadsf ads ads asd asd ads "
        conn.Write([]byte(sendStr))
        time.Sleep(time.Second)
}
}
登录后复制

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

执行结果

一篇文章带你了解Go语言基础之网络编程

如果我注释了第18行代码

一篇文章带你了解Go语言基础之网络编程

执行结果

一篇文章带你了解Go语言基础之网络编程

直接都淦到一行了,what?这是啥情况,不应该跟原来一样吗???

每发送一个值,那边就接收一下,这怎么整到一块了!!!

原因

主要原因是因为我们是应用层软件,是跑在操作系统之上的软件,当我们向服务器发送一个数据时,是调用操作系统相关接口发送的,操作系统再经过各种复杂的操作,发送到对方机器

但是操作系统有一个发送数据缓冲区,默认情况如果缓冲区是有大小的,如果缓冲区没满,是不会发送数据的,所以上述客户端在发送数据时,系统的缓冲区都没满,一直压在了操作系统的缓冲区中,最后发现没数据了,才一次都发送到服务端

但是为什么sleep(1)又管用了呢?这是因为缓冲区不止一个程序在用,1s的时间足够其他程序将缓冲区打满,然后各自发各自的数据,这也是为什么第一次操作没问题,第二次有问题,因为第二次全部都是我们客户端打满的

一篇文章带你了解Go语言基础之网络编程

解决粘包

工具函数

我们将解包封包的函数封装一下

socker_sitck/stick.go

一篇文章带你了解Go语言基础之网络编程

package socker_stick


import (
    "bufio"
    "bytes"
    "encoding/binary"
    "fmt"
)


//Encode 将消息编码
func Encode(message string) ([]byte, error) {
    length := int32(len(message))
    var pkg = new(bytes.Buffer)
    //写入消息头
    err := binary.Write(pkg, binary.LittleEndian, length)
    if err != nil {
        fmt.Println("写入消息头失败", err)
        return nil, err
}
    //写入消息实体
    err = binary.Write(pkg, binary.LittleEndian, []byte(message))
    if err != nil {
        fmt.Println("写入消息实体失败", err)
        return nil, err
}
    return pkg.Bytes(), nil
}


//Decode解码消息
func Decode(reader *bufio.Reader) (string, error) {
    //读取信息长度
    lengthByte, _ := reader.Peek(4)
    lengthBuff := bytes.NewBuffer(lengthByte)
    var length int32
    err := binary.Read(lengthBuff, binary.LittleEndian, &length)
    if err != nil {
        return "", err
}
    //BuffRead 返回缓冲区现有的可读的字节数
    if int32(reader.Buffered()) < length+4 {
        return "", err
}
    pack := make([]byte, int(4+length))
    _, err = reader.Read(pack)
    if err != nil {
        return "", err
}
    return string(pack[4:]), nil
}
登录后复制

服务端

package main


import (
    "a3_course/socker_stick"
    "bufio"
    "fmt"
    "io"
    "net"
)


func process(conn net.Conn) {
    defer conn.Close()
    reader := bufio.NewReader(conn)


    for {
        msg, err := socker_stick.Decode(reader)
        //读完了
        if err == io.EOF {
            fmt.Println("读完了")
            break
        }
        //读错了
        if err != nil {
            fmt.Println("数据读取失败", err)
            return
        }


        fmt.Println("客户端发送过来的值:", msg)
}


}
func main() {
    lister, err := net.Listen("tcp", "0.0.0.0:8008")
    if err != nil {
        fmt.Println("连接失败", err)
        return
}
    defer lister.Close()
    for {
        fmt.Println("等待建立连接,此时会阻塞住")
        conn, err := lister.Accept() //等待建立连接
        fmt.Println("连接建立成功,继续")
        if err != nil {
            fmt.Println("建立连接失败", err)
            //继续监听下次链接
            continue
        }
        go process(conn)
}
}
登录后复制

客户端

package main


import (
    "a3_course/socker_stick"
    "fmt"
    "net"
)


//客户端
func main() {
    conn, err := net.Dial("tcp", "192.168.10.148:8008")
    if err != nil {
        fmt.Println("连接服务器失败", err)
}
    defer conn.Close()
    for i := 0; i < 10; i++ {
        sendStr := "hello world ads asdf asd fads fadsf ads ads asd asd ads "
        data, err := socker_stick.Encode(sendStr)
        if err != nil {
            fmt.Println("编码失败",err)
            return
        }
        conn.Write(data)
        //time.Sleep(time.Second)
}
}
登录后复制

执行结果

一篇文章带你了解Go语言基础之网络编程

这次真的不管执行几次,都是这样的结果

对了,只有TCP才有粘包

Go语言UDP

UDP是一个无连接协议,客户端不会在乎服务端有没有问题,客户端只管发,通常用于实时性比较高的领域

例如直播行业

服务端

package main


import (
    "fmt"
    "net"
)


func main() {
    listen, err := net.ListenUDP("udp", &net.UDPAddr{
        IP:   net.IPv4(0, 0, 0, 0),
        Port: 8009,
})
    if err != nil {
        panic(fmt.Sprintf("udp启动失败,err:%v", err))
}
    defer listen.Close()
    for{
        var data = make([]byte,1024)
        n, addr, err := listen.ReadFromUDP(data)
        if err != nil {
            panic(fmt.Sprintf("读取数据失败,err:%v", err))
        }
        fmt.Println(string(data[:n]),addr,n)
}
}
登录后复制

客户端

package main


import (
    "fmt"
    "net"
)


func main() {
    socker, err := net.DialUDP("udp", nil, &net.UDPAddr{
        IP:   net.IPv4(0, 0, 0, 0),
        Port: 8009,
})
    if err != nil {
        panic(fmt.Sprintf("连接服务器失败,err:%v", err))
}
    defer socker.Close()
    sendStr:="你好呀"
    _, err = socker.Write([]byte(sendStr))
    if err != nil {
        panic(fmt.Sprintf("数据发送失败,err:%v", err))
}
}
登录后复制

执行结果

一篇文章带你了解Go语言基础之网络编程

总结

本次章节我们讲述了什么是TCP,什么是UDP。

并且编写了代码如何实现TCP服务端TCP客户端UDP服务端UDP客户端

讲述了为什么会出现粘包,该怎么解决粘包

以上是一篇文章带你了解Go语言基础之网络编程的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
2 周前 By 尊渡假赌尊渡假赌尊渡假赌
仓库:如何复兴队友
4 周前 By 尊渡假赌尊渡假赌尊渡假赌
Hello Kitty Island冒险:如何获得巨型种子
4 周前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

Golang技术在移动物联网开发中的作用 Golang技术在移动物联网开发中的作用 May 09, 2024 pm 03:51 PM

Go语言凭借着其高并发性、高效性和跨平台性,成为移动物联网(IoT)应用程序开发的理想选择。Go的并发模型通过goroutine(轻量级协程)实现高度并发,适合处理大量同时连接的IoT设备。Go的低资源消耗有助于在计算和存储有限的移动设备上高效运行应用程序。此外,Go的跨平台支持使IoT应用程序能够轻松部署在各种移动设备上。实战案例展示了用Go构建BLE温度传感器应用,通过BLE与传感器通信并处理传入数据,从而读取和显示温度读数。

golang框架的优势有哪些? golang框架的优势有哪些? Jun 06, 2024 am 10:26 AM

Golang框架的优势Golang是一种高性能、并发编程语言,特别适用于微服务和分布式系统。 Golang框架通过提供一组现成的组件和工具,使开发这些应用程序变得更加容易。以下是Golang框架的一些关键优势:1.高性能和并发性:Golang本身以其高性能和并发性着称。它使用goroutine,这是一种轻量级的线程机制,允许并发执行代码,从而提高应用程序的吞吐量和响应能力。 2.模块化和可重用性:Golang框架鼓励模块化和可重用代码。通过将应用程序分解为独立的模块,您可以轻松维护和更新代

Golang语言在区块链物联网场景中的应用探索 Golang语言在区块链物联网场景中的应用探索 May 09, 2024 pm 02:48 PM

Go语言在区块链物联网场景中广泛应用,主要优势包括:编写智能合约,管理区块链上商品流动的规则和操作。开发应用程序,管理资产交易并跟踪资产位置和状态。高效、并发和开源的特性,适合处理大量数据并易于扩展和定制。

Go 语言中的包依赖是如何工作的? Go 语言中的包依赖是如何工作的? Jun 01, 2024 pm 10:40 PM

在Go语言中,包依赖通过import语句管理。Go语言中的包依赖有直接依赖和间接依赖两种类型。Go模块系统通过gomod工具管理包依赖,包括模块化、依赖版本控制和依赖下载安装等任务。

如何使用 Go 语言中的管道实现生产者消费者模式? 如何使用 Go 语言中的管道实现生产者消费者模式? Jun 02, 2024 pm 03:28 PM

生产者消费者模式允许生产者将数据放入缓存,而消费者可同时从中提取数据处理。在Go中,管道是一种通信机制,可实现此模式:创建管道:make(chanT),其中T为传输数据类型。生产者函数:将数据放入管道(ch

如何使用 Go 语言中的管道创建并行任务? 如何使用 Go 语言中的管道创建并行任务? Jun 04, 2024 pm 02:46 PM

管道是一种无缓冲通信机制,可用于创建并行任务:创建管道:ch:=make(chanint)发送数据:ch

如何使用 Go 语言中的管道传递数据? 如何使用 Go 语言中的管道传递数据? Jun 01, 2024 pm 04:15 PM

Go语言管道用于在goroutine间传递数据,可通过以下步骤使用:创建管道:make(chanint)发送数据:channel&lt;-data接收数据:data:=&lt;-channel创建缓冲管道:make(chanint,10)(缓冲区大小为10)实战案例:并发计算整数平方和并在goroutine中使用管道传递结果

Go语言与区块链技术的融合及其未来发展展望 Go语言与区块链技术的融合及其未来发展展望 May 09, 2024 pm 01:09 PM

Go语言凭借高性能、可扩展性和易用性,成为区块链开发的首选。它用于实现关键HyperledgerFabric组件(链码、共识机制),并将在未来推动去中心化应用程序、物联网和可互操作性解决方案的发展。

See all articles