温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

GoLang string与strings.Builder使用源码对比分析

发布时间:2023-03-17 14:23:59 来源:亿速云 阅读:310 作者:iii 栏目:开发技术

本文小编为大家详细介绍“GoLang string与strings.Builder使用源码对比分析”,内容详细,步骤清晰,细节处理妥当,希望这篇“GoLang string与strings.Builder使用源码对比分析”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

    一、strings.Builder的优势

    • 以存在的内容不可变,但可以拼接更多的内容;

    • 减少了内存分配和内存拷贝的次数;

    • 可将可容重叠,可重用值;

    二、string类型的值

    在Go语言中,string类型的值是不可变的。

    基于原字符串的裁剪和拼接操作:

    • 裁剪操作可以使用切片表达式;

    • 拼接操作可以使用操作符+实现;

    一个string的内容会被存储在一块连续的内存空间中,这个连续空间的字节数量也会被记录下来,并用于表示该string值的长度。

    对string类型的值,执行len() 方法,得到的是字节的数量。

    	var str01 string
    	str01 = "Go,你好"
    	// len(string) : 11
    	fmt.Printf("len(string) : %v \n", len(str01))
    	// len([]char): 5
    	fmt.Printf("len([]char): %v\n", len([]rune(str01)))
    	// len([]byte): 11
    	fmt.Printf("len([]byte): %v\n", len([]byte(str01)))

    对string类型的值,进行切片操作,相当于对底层的字节数组做切片

    	// 在string类型的值上进行切片操作,相当于对底层的字节数组做切片
    	fmt.Println(str01[0:3]) // Go�
    	fmt.Println(str01[0:5]) // Go,

    在执行字符串拼接的时候,会把所有被拼接的字符串依次拷贝到一个崭新且足够大的连续内存空间中,并把相应指针指的string值作为结果返回。

    三、与string相比Builder的优势体现在拼接方面

    与string相比,strings.Builder 的优势最要体现在字符串拼接方面。

    Builder有一个内容容器,它是一个以byte为元素类型的切片(简称字节切片)。

    Builder类型的值只能被拼接或完全覆盖。

    3.1 Builder的拼接与Builder的自动扩容

    可以通过Write、WriteByte、WriteRune、WriteString方法,把新的内容拼接到已存在的内容的尾部。

    Builder会自动地对自身的内容容器进行扩容。扩容策略与切片的扩容策略一致。

    3.2 手动扩容

    Builder的值还可以手动扩容,通过调用Builder的Grow方法,就可以做到。

    Grow方法接受一个int类型的参数n,这个参数表示要扩充的字节数量。

    如果未用容量大于或等于n,Grow方法可能什么都不做。

    3.3 Builder的重用

    通过调用Reset方法,可以让Builder值重新回到零值的状态,就像它从未被使用一样。

    四、strings.Builder的使用约束

    • springs.Builder 类型的值在已被真正使用后,就不可再被复制。

    • 由于其内容不是完全不可变的,需要使用方自行解决操作冲突和并发安全的问题。

    只要调用了Builder值的拼接或扩容方法,就意味着真正使用。一旦真正使用,Builder值就不能被复制,否则调用副本的方法会引起panic。

    	var sb01 strings.Builder
    	sb01.WriteString("Go")
    	sb02 := sb01
    	// sb02.Grow(1) // 这里会引起恐慌panic
    	_ = sb02

    虽然Builder值不可复制,但是它的指针指却可以复制。复制的指针值会指向同一个Builder值。

    对于处于零值状态的Builder值,复制不会有任何问题。所以,只要在传递之前调用Reset方法即可。

    	var sb01 strings.Builder
    	sb01.WriteString("Go")
        sb01.Reset()
    	sb03 := sb01
    	sb03.Grow(1)

    五、strings.Reader类型的值可以高效读取字符串

    strings.Reader 类型的值,可以让我们很方便的读取一个字符串中的内容。因为,在读取的过程中,Reader值会保存已读取的字节的计数。

    一读计数代表着下次读取的起始索引位置。

    Reader 正是依靠这样的一个计数,以及针对字符串的切片表达式,从而实现快读读取。

    通过Len方法和Size方法获取已读计数:

    	var reader1 strings.Reader
    	reader1 = *strings.NewReader("Go, 你好")
    	ch, size, err := reader1.ReadRune()
    	fmt.Printf("%v, %v, %v\n", string(ch), size, err)
    	readingIndex := reader1.Size() - int64(reader1.Len())
    	fmt.Printf("以读计数:%v\n", readingIndex)

    Reader 的Seek方法用于设定下一次读取的起始索引位置;如果把io.SeekCurrent 的值作为第二个参数值传递给该方法。那么它会依据当前的已读计数,以及第一个参数offset的值来计算新的计数值。

    读到这里,这篇“GoLang string与strings.Builder使用源码对比分析”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注亿速云行业资讯频道。

    向AI问一下细节

    免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

    AI