我的编程空间,编程开发者的网络收藏夹
学习永远不晚

一文带你了解Golang中的缓冲区Buffer

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

一文带你了解Golang中的缓冲区Buffer

作为一种常见的数据结构,缓冲区(Buffer)在计算机科学中有着广泛的应用。Go 语言标准库中提供了一个名为 bytes.Buffer 的缓冲区类型,它可以方便地进行字符串操作、IO 操作、二进制数据处理等。本篇博客将详细介绍 Go 中 Buffer 的用法,从多个方面介绍其特性和应用场景。

1. Buffer 是什么

在计算机科学中,缓冲区(Buffer)是一种数据结构,它用于临时存储数据,以便稍后进行处理。在 Go 语言中,bytes.Buffer 是一个预定义的类型,用于存储和操作字节序列。bytes.Buffer 类型提供了很多有用的方法,例如:读写字节、字符串、整数和浮点数等。

// 创建一个空的缓冲区
var buf bytes.Buffer
​
// 向缓冲区写入字符串
buf.WriteString("Hello, World!")
​
// 从缓冲区读取字符串
fmt.Println(buf.String()) // 输出:Hello, World!

2. 创建缓冲区

要使用 Buffer 类型,我们首先需要创建一个缓冲区。可以通过以下两种方式来创建一个 Buffer 对象。

2.1 使用 NewBuffer 函数创建

可以使用 bytes 包中的 NewBuffer 函数来创建一个新的缓冲区对象。它的方法如下:

func NewBuffer(buf []byte) *Buffer

其中,buf 参数是可选的,它可以用来指定缓冲区的初始容量。如果不指定该参数,则会创建一个默认容量为 64 字节的缓冲区。

下面是一个使用 NewBuffer 函数创建缓冲区的示例:

import (
    "bytes"
    "fmt"
)
​
func main() {
    buf := bytes.NewBufferString("hello world")
    fmt.Println(buf.String()) // 输出:hello world
}

2.2 使用 bytes.Buffer 结构体创建

另一种创建缓冲区对象的方式是直接声明一个 bytes.Buffer 类型的变量。这种方式比较简单,但是需要注意,如果使用这种方式创建的缓冲区没有被初始化,则其初始容量为 0,需要在写入数据之前进行扩容。

下面是一个使用 bytes.Buffer 结构体创建缓冲区的示例:

import (
    "bytes"
    "fmt"
)
​
func main() {
    var buf bytes.Buffer
    buf.WriteString("hello")
    buf.WriteString(" ")
    buf.WriteString("world")
    fmt.Println(buf.String()) // 输出:hello world
}

3. 写入数据

创建好缓冲区之后,我们可以向其中写入数据。Buffer类型提供了多种方法来写入数据,其中最常用的是Write方法。它的方法如下:

func (b *Buffer) Write(p []byte) (n int, err error)

其中,p 参数是要写入缓冲区的字节切片,返回值 n 表示实际写入的字节数,err 表示写入过程中可能出现的错误。

除了 Write 方法之外,Buffer 类型还提供了一系列其他方法来写入数据,例如 WriteString、WriteByte、WriteRune 等。这些方法分别用于向缓冲区写入字符串、单个字节、单个 Unicode 字符等。

下面是一个使用 Write 方法向缓冲区写入数据的示例:

import (
    "bytes"
    "fmt"
)
​
func main() {
    buf := bytes.NewBuffer(nil)
    n, err := buf.Write([]byte("hello world"))
    if err != nil {
        fmt.Println("write error:", err)
    }
    fmt.Printf("write %d bytes\n", n) // 输出:write 11 bytes
    fmt.Println(buf.String()) // 输出:hello world
}

4. 读取数据

除了写入数据之外,我们还可以从缓冲区中读取数据。Buffer 类型提供了多种方法来读取数据,其中最常用的是 Read 方法。它的方法如下:

func (b *Buffer) Read(p []byte) (n int, err error)

其中,p 参数是用于存放读取数据的字节切片,返回值 n 表示实际读取的字节数,err 表示读取过程中可能出现的错误。

除了 Read 方法之外,Buffer 类型还提供了一系列其他方法来读取数据,例如 ReadString、ReadByte、ReadRune 等。这些方法分别用于从缓冲区读取字符串、单个字节、单个 Unicode 字符等。

下面是一个使用 Read 方法从缓冲区读取数据的示例:

import (
    "bytes"
    "fmt"
)
​
func main() {
    buf := bytes.NewBufferString("hello world")
    data := make([]byte, 5)
    n, err := buf.Read(data)
    if err != nil {
        fmt.Println("read error:", err)
    }
    fmt.Printf("read %d bytes\n", n) // 输出:read 5 bytes
    fmt.Println(string(data)) // 输出:hello
}

5. 截取缓冲区

Buffer 类型提供了 Bytes 方法和 String 方法,用于将缓冲区的内容转换为字节切片和字符串。另外,还可以使用 Truncate 方法来截取缓冲区的内容。它的方法如下:

func (b *Buffer) Truncate(n int) 

其中,n 参数表示要保留的字节数。如果缓冲区的内容长度超过了 n,则会从尾部开始截取,只保留前面的 n 个字节。如果缓冲区的内容长度不足 n,则不做任何操作。

下面是一个使用 Truncate 方法截取缓冲区的示例:

import (
    "bytes"
    "fmt"
)
​
func main() {
    buf := bytes.NewBufferString("hello world")
    buf.Truncate(5)
    fmt.Println(buf.String()) // 输出:hello
}

6. 扩容缓冲区

在写入数据的过程中,如果缓冲区的容量不够,就需要进行扩容。Buffer 类型提供了 Grow 方法来扩容缓冲区。它的方法如下:

func (b *Buffer) Grow(n int) 

其中,n 参数表示要扩容的字节数。如果 n 小于等于缓冲区的剩余容量,则不做任何操作。否则,会将缓冲区的容量扩大到原来的 2 倍或者加上 n,取两者中的较大值。

下面是一个使用 Grow 方法扩容缓冲区的示例:

import (
    "bytes"
    "fmt"
)
​
func main() {
    buf := bytes.NewBufferString("hello")
    buf.Grow(10)
    fmt.Printf("len=%d, cap=%d\n", buf.Len(), buf.Cap()) // 输出:len=5, cap=16
}

在上面的示例中,我们创建了一个包含 5 个字节的缓冲区,并使用 Grow 方法将其容量扩大到了 16 字节。由于 16 是大于 5 的最小的 2 的整数次幂,因此扩容后的容量为 16。

需要注意的是,Buffer 类型并不保证扩容后的缓冲区是连续的,因此在将缓冲区的内容传递给需要连续内存的接口时,需要先将缓冲区的内容拷贝到一个新的连续内存中。

7. 重置缓冲区

在有些情况下,我们需要重复使用一个缓冲区。此时,可以使用 Reset 方法将缓冲区清空并重置为初始状态。它的方法如下:

func (b *Buffer) Reset() 

下面是一个使用 Reset 方法重置缓冲区的示例:

import (
    "bytes"
    "fmt"
)
​
func main() {
    buf := bytes.NewBufferString("hello")
    fmt.Println(buf.String()) // 输出:hello
    buf.Reset()
    fmt.Println(buf.String()) // 输出:
}

在上面的示例中,我们首先创建了一个包含 hello 的缓冲区,并使用 Reset 方法将其重置为空缓冲区。注意,重置后的缓冲区长度和容量都变为了 0。

8. 序列化和反序列化

由于 bytes.Buffer 类型支持读写操作,它可以用于序列化和反序列化结构体、JSON、XML 等数据格式。这使得 bytes.Buffer 类型在网络通信和分布式系统中的应用变得更加便捷。

type Person struct {
    Name string
    Age  int
}
​
// 将结构体编码为 JSON
p := Person{"Alice", 25}
enc := json.NewEncoder(&buf)
enc.Encode(p)
fmt.Println(buf.String()) // 输出:{"Name":"Alice","Age":25}
​
// 从 JSON 解码为结构体
var p2 Person
dec := json.NewDecoder(&buf)
dec.Decode(&p2)
fmt.Printf("Name: %s, Age: %d\n", p2.Name, p2.Age) // 输出:Name: Alice, Age: 25

9. Buffer 的应用场景

9.1 网络通信

在网络通信中,bytes.Buffer 可以用于存储和处理 TCP/UDP 数据包、HTTP 请求和响应等数据。例如,我们可以使用 bytes.Buffer 类型来构造 HTTP 请求和响应:

// 构造 HTTP 请求
req := bytes.NewBufferString("GET / HTTP/1.0\r\n\r\n")
​
// 构造 HTTP 响应
resp := bytes.NewBuffer([]byte("HTTP/1.0 200 OK\r\nContent-Type: text/html\r\n\r\nHello, World!"))

9.2 文件操作

在文件操作中,bytes.Buffer 可以用于缓存文件内容,以避免频繁的磁盘读写操作。例如,我们可以使用 bytes.Buffer 类型来读取和写入文件:

// 从文件中读取数据
file, err := os.Open("example.txt")
if err != nil {
    log.Fatal(err)
}
defer file.Close()
​
var buf bytes.Buffer
_, err = io.Copy(&buf, file)
if err != nil {
    log.Fatal(err)
}
​
fmt.Println(buf.String())
​
// 将数据写入文件
out, err := os.Create("output.txt")
if err != nil {
    log.Fatal(err)
}
defer out.Close()
​
_, err = io.Copy(out, &buf)
if err != nil {
    log.Fatal(err)
}

9.3 二进制数据处理

在处理二进制数据时,bytes.Buffer 可以用于存储和操作字节数组。例如,我们可以使用 bytes.Buffer 类型来读写字节数组、转换字节数组的大小端序等操作:

// 读取字节数组
data := []byte{0x48, 0x65,0x6c, 0x6c, 0x6f}
var buf bytes.Buffer
buf.Write(data)
​
// 转换大小端序
var num uint16
binary.Read(&buf, binary.BigEndian, &num)
fmt.Println(num) // 输出:0x4865
​
// 写入字节数组
data2 := []byte{0x57, 0x6f, 0x72, 0x6c, 0x64, 0x21}
buf.Write(data2)
fmt.Println(buf.Bytes()) // 输出:[72 101 108 108 111 87 111 114 108 100 33]

9.4 字符串拼接

在字符串拼接时,如果直接使用 + 运算符会产生大量的中间变量,影响程序的效率。使用 Buffer 类型可以避免这个问题。

import (
    "bytes"
    "strings"
)
​
func concatStrings(strs ...string) string {
    var buf bytes.Buffer
    for _, s := range strs {
        buf.WriteString(s)
    }
    return buf.String()
}
​
func main() {
    s1 := "hello"
    s2 := "world"
    s3 := "!"
    s := concatStrings(s1, s2, s3)
    fmt.Println(s) // 输出:hello world!
}

在上面的示例中,我们使用 Buffer 类型将多个字符串拼接成一个字符串。由于 Buffer 类型会动态扩容,因此可以避免产生大量的中间变量,提高程序的效率。

9.5 格式化输出

在输出格式化的字符串时,我们可以使用 fmt.Sprintf 函数,也可以使用 Buffer 类型。

import (
    "bytes"
    "fmt"
)
​
func main() {
    var buf bytes.Buffer
    for i := 0; i < 10; i++ {
        fmt.Fprintf(&buf, "%d\n", i)
    }
    fmt.Println(buf.String())
}

在上面的示例中,我们使用 Buffer 类型将 10 个整数格式化为字符串,并输出到标准输出。使用 Buffer 类型可以方便地组织格式化的字符串,同时也可以减少系统调用的次数,提高程序的效率。

9.6 图像处理

在图像处理中,我们经常需要将多个图像合成一个新的图像。使用 Buffer 类型可以方便地缓存多个图像的像素值,然后将它们合成为一个新的图像。

import (
    "bytes"
    "image"
    "image/png"
    "os"
)
​
func combineImages(images []image.Image) image.Image {
    width := images[0].Bounds().Dx()
    height := images[0].Bounds().Dy() * len(images)
    canvas := image.NewRGBA(image.Rect(0, 0, width, height))
    var y int
    for _, img := range images {
        for i := 0; i < img.Bounds().Dy(); i++ {
            for j := 0; j < img.Bounds().Dx(); j++ {
                canvas.Set(j, y+i, img.At(j, i))
            }
        }
        y += img.Bounds().Dy()
    }
    return canvas
}
​
func main() {
    images := make([]image.Image, 3)
    for i := 0; i < 3; i++ {
        f, _ := os.Open(fmt.Sprintf("image%d.png", i+1))
        img, _ := png.Decode(f)
        images[i] = img
    }
    combined := combineImages(images)
    f, _ := os.Create("combined.png")
    png.Encode(f, combined)
}

在上面的示例中,我们使用 Buffer 类型缓存多个图像的像素值,并将它们合成为一个新的图像。使用 Buffer 类型可以方便地缓存像素值,同时也可以减少系统调用的次数,提高程序的效率。

10. 总结

在 Go 语言中,bytes.Buffer 类型是一个十分实用的数据类型,它可以用于存储和操作二进制数据、网络通信数据、文件数据等。在实际开发中,我们经常会使用 bytes.Buffer 类型来缓存数据、序列化和反序列化数据、处理二进制数据等操作,以提高代码的可读性、可维护性和可扩展性。

除了 bytes.Buffer 类型之外,Go 语言中还有 bytes.Reader 和 bytes.Writer 类型,它们都是基于 bytes.Buffer 类型实现的,可以用于读取和写入数据,但 bytes.Reader 类型只能读取数据,而 bytes.Writer 类型只能写入数据。在实际开发中,我们可以根据不同的需求来选择不同的类型。

以上就是一文带你了解Golang中的缓冲区Buffer的详细内容,更多关于Golang缓冲区Buffer的资料请关注编程网其它相关文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

一文带你了解Golang中的缓冲区Buffer

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

一文带你了解Golang中的缓冲区Buffer

作为一种常见的数据结构,缓冲区(Buffer)在计算机科学中有着广泛的应用。这篇文章将详细介绍 Go 中 Buffer 的用法,从多个方面介绍其特性和应用场景,需要的可以参考一下
2023-05-18

一文带你深入了解Node中的Buffer类

本篇文章带大家深入了解下Node​中 Buffer(缓冲区)类,希望对大家有所帮助!
2023-05-14

一文带你了解Golang中的WaitGroups

WaitGroups是同步你的goroutines的一种有效方式。这篇文章主要来和大家聊聊Golang中WaitGroups的使用,感兴趣的小伙伴可以跟随小编一起了解一下
2023-03-14

一文带你了解Golang中的并发性

并发是一个很酷的话题,一旦你掌握了它,就会成为一笔巨大的财富。所以本文就来和大家一起来聊聊Golang中的并发性,感兴趣的可以了解一下
2023-03-15

一文带你了解Redis中RDB与AOF的区别

目录Redis 中的 RDB 与 AOFRDBAOF小结Redis 中的 RDB 与 AOF我们都知道,Redis 运行时是将数据保存在内存中的,如果服务器宕机或者重启,那么内存中的数据必然会丢失,从而影响正常的业务运行。所以,我们就必须
2023-06-10

一文带你了解Golang中interface的设计与实现

本文就来详细说说为什么说 接口本质是一种自定义类型,以及这种自定义类型是如何构建起 go 的 interface 系统的,感兴趣的小伙伴可以跟随小编一起学习一下
2023-01-04

一文带你了解Golang中reflect反射的常见错误

go 反射的错误大多数都来自于调用了一个不适合当前类型的方法, 而且,这些错误通常是在运行时才会暴露出来,而不是在编译时,如果我们传递的类型在反射代码中没有被覆盖到那么很容易就会 panic。本文就介绍一下使用 go 反射时很大概率会出现的错误,需要的可以参考一下
2023-01-05

一文带你了解MySQL中的事务

目录一.什么是事务二.事务操作演示小结三.事务的特性四.事务的隔离级别概述四种隔离级别脏读、不可重复读、幻读操作一.什么是事务在mysql中的事务(Transaction)是由存储引擎实现的,在MySQL中,只有InnoDB存储引擎才支持
2023-02-17

一文带你了解Java中数值与集合的区别

一文带你了解Java中数值与集合的区别?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。数组array和集合的区别:(1) 数值是大小固定的,同一数组只能存放一样的数据。(2) j
2023-05-31

一文带你深入理解Golang中的RWMutex

这篇文章主要为大家详细介绍了Golang中RWMutex的相关知识,知其然,更要知其所以然。文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下
2023-05-14

一文带你了解Golang中类型转换库cast的使用

你是否在使用Go的过程中因为类型转换的繁琐而苦恼过?你是否觉得Go语言中的类型断言可能会panic而对自己写的代码有那么一点点不放心?本文就为大家推荐一个用于类型转换的第三方库cast绝对是一个值得尝试的选择
2023-02-08

一文带你了解Python中的枚举(enum)

这篇文章一文带你了解Python中的枚举(enum),在Python中,枚举和我们在对象中定义的类变量时一样的,每一个类变量就是一个枚举项,需要的朋友可以参考下
2023-05-15

一文带你了解Node.js中的path模块

Node.js和Python技术类似, 都致力于能够实现跨平台的通用代码。 为此,针对路径的拼接, Node.js提供了path模块,本文就来讲讲path模块的使用
2023-03-21

一文带你了解C++中queue的使用

C++中的queue是一种容器,用于在FIFO(先进先出)原则下存储和管理元素。本篇文章将深入探讨C++中的queue,包括它的定义、使用、原理和示例,感兴趣的可以了解一下
2023-05-18

一文带你了解C++中deque的使用

C++中的deque是一种双端队列,可以在队列的前端和后端进行插入元素和删除操作,同时可以视作一个长度不定的数组,支持高效的插入和删除操作。本篇文章将深入探讨C++中的deque的使用,感兴趣的可以了解一下
2023-05-18

一文带你了解Node.js中的http模块

本篇文章给大家了解一下Node.js http模块,介绍一下使用http模块创建服务器的方法,希望对大家有所帮助!
2023-05-14

一文带你了解Python中pymysql的使用

目录前言一、pymysql用途二、下载1.下载依赖2.下载方式三、使用 1.连接Mysql数据库2.创建游标对象 3.执行函数4.获取查询结果集的方法前言首先使用python很大一部分人是用于数据分析或者是开发,而数据来源一般都是存储在数
2023-02-21

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录