当前位置: 首页 > article >正文

string详解

Golang详解string

文章目录

  • Golang详解string
    • Golang中为什么string是只读的?
    • stirng和[]byte的转化原理
    • []byte转string一定需要内存拷贝吗?
    • 字符串拼接性能测试

Golang中为什么string是只读的?

在Go语言中,string其实就是一个结构体,包含一个指向底层数组的指针和长度。字符串只读,在Go运行时能有效的管理内存分配,在创建字符串后不可修改,那么字符串就固定在内存中了,就可以消除跟踪和管理字符串修改的复杂性了。同时,在多线程的环境下,不可变性让字符串避免数据竞争和一致性问题,不需要额外的同步处理了。

stirng和[]byte的转化原理

从string的底层结构就知道是不可扩容的,string和[]byte的区别就是在[]byte中多了个容量,所以string转[]byte和[]byte转string都是进行内存的拷贝,指针数据和长度的匹配。

[]byte转string一定需要内存拷贝吗?

如果[]byte转string是临时场景,那么就不需要内存拷贝。就比如;

  1. 字符串拼接,临时使用
  2. 查找数据,临时使用
  3. 用于比较,临时使用

字符串拼接性能测试

Golang中常用的字符串拼接:

  1. strings.Builder
  2. strings.Join
  3. (加号) +
  4. fmt.Sprintf
  5. append
package main

import (
	"bytes"
	"fmt"
	"strings"
	"testing"
)

var loremIpsm = `It is a highly competitive world. One can feel the existence of competition everywhere, from the classroom to the job-hunting market. Looking for a fair opportunity to prove one's ability has become a matter of survival.If one wants to survive and to be successful in such a challenging society, one must learn to face the competition bravely`

var strSlice = make([]string, LIMIT)

const LIMIT = 1000

func init() {
	for i := 0; i < LIMIT; i++ {
		strSlice[i] = loremIpsm
	}
}

// 进行压力测试
// +
func BenchmarkOperator(b *testing.B) {
	for i := 0; i < b.N; i++ {
		var q string
		for _, s := range strSlice {
			q = q + s
		}
	}
	b.ReportAllocs()
}

// Sprintf
func BenchmarkSprintf(b *testing.B) {
	for i := 0; i < b.N; i++ {
		var q string
		for _, s := range strSlice {
			q = fmt.Sprintf(q, s)
		}
	}
	b.ReportAllocs()
}

// strings.Join
func BenchmarkJoin(b *testing.B) {
	for i := 0; i < b.N; i++ {
		strings.Join(strSlice, "")
	}
	b.ReportAllocs()
}

// bytes.Buffer
func BenchmarkBuffer(b *testing.B) {
	for i := 0; i < b.N; i++ {
		var q bytes.Buffer
		q.Grow(len(loremIpsm) * len(strSlice))
		for _, s := range strSlice {
			q.WriteString(s)
		}
	}
	b.ReportAllocs()
}

// append
func BenchmarkAppend(b *testing.B) {
	for i := 0; i < b.N; i++ {
		var q []byte
		for _, s := range strSlice {
			q = append(q, s...)
		}
	}
	b.ReportAllocs()
}

// strings.Builder
func BenchmarkBuilder(b *testing.B) {
	for i := 0; i < b.N; i++ {
		var q strings.Builder
		q.Grow(len(loremIpsm) * len(strSlice))
		for _, s := range strSlice {
			q.WriteString(s)
		}
	}
	b.ReportAllocs()
}

测试结果:
在这里插入图片描述

可以看到性能比较好的是strings.Builder、strings.Join、bytes.Buffer这三个性能相比之下比较高。

如果大量字符串进行拼接时建议使用以上性能好的拼接方式,如果是少量的字符串用+比较方便。fmt.Sprintf性能最差,它一般用于格式化返回字符串而不是拼接。


http://www.kler.cn/a/289559.html

相关文章:

  • 前端,location.reload刷新页面
  • C++中string的新特性
  • 【计算机网络】UDP网络程序
  • 【MySQL】约束
  • 基于Python+Django+Vue3+MySQL实现的前后端分类的商场车辆管理系统
  • 【深圳大学】数据结构A+攻略(计软版)
  • [HNCTF 2022 WEEK2]Canyource
  • 关于武汉芯景科技有限公司的实时时钟芯片XJ8340开发指南(兼容DS1340)
  • 密码学---对称加密和非对称加密
  • 【网络安全】IDOR之敏感数据泄露
  • 果浆产业的自动化、智能化离不开机器视觉的发展
  • Lua:条件断点
  • 第140天:内网安全-横向移动局域网ARP欺骗DNS劫持钓鱼中间人单双向
  • 【Xcode】Xcode基本使用指引
  • c++ for (const auto info : prerequisites) 解释这个语句中每个单词的含义
  • 【数据结构篇】~链表算法题2
  • 开发指南058-JPA多数据源
  • 项目经理成长路径
  • java中数据访问层userdao接口怎么写
  • Apache DolphinScheduler项目与社区7-8月发展报告
  • 冲击大厂算法面试=>链表专题【链表反转之局部反转升级版】
  • 1、正则表达式
  • C++ | Leetcode C++题解之第394题字符串解码
  • Elasticsearch检索原理
  • 2024.9.2 作业
  • Loadrunner12录制时,目标网站打不开的解决办法