时间:2021-05-22
go 迭代string数组,直接拷贝去用即可
package mainimport ("fmt")func main() { subsCodes := []string{"aaaa", "vvvvv", "dddd", "eeeee", "gfgggg"} for _, s := range subsCodes { fmt.Println(s) }}补充:golang字符串string与字符数组[]byte高效转换
string与[]byte的直接转换是通过底层数据copy实现的
var a = []byte("hello boy")
var b = string(a)
这种操作在并发量达到十万百万级别的时候会拖慢程序的处理速度
通过gdb调试来看一下string和[]byte的数据结构
(gdb) l main.mainimport ( "fmt")func main() { s := "hello, world!" b := []byte(s) fmt.Println(s, b)(gdb) b 11Breakpoint 1 at 0x487cd9: file /export/home/machao/src/test/strbytes.go, line 11.(gdb) rStarting program: /export/home/machao/src/test/test1 Breakpoint 1, main.main () at /export/home/machao/src/test/strbytes.go:11 fmt.Println(s, b)(gdb) info localss = {str = 0x4b8ccf "hello, world!level 3 resetload64 failednil stackbaseout of memorys.allocCount=srmount errorstill in listtimer expiredtriggerRatio=unreachable: value method xadd64 failedxchg64 failed nmidlelocked= on "..., len = 13}b = {array = 0xc4200140e0 "hello, world!", len = 13, cap = 16}(gdb) ptype stype = struct string { uint8 *str; int len;}(gdb) ptype btype = struct []uint8 { uint8 *array; int len; int cap;}转换后 [ ]byte 底层数组与原 string 内部指针并不相同,以此可确定数据被复制。那么,如不修改数据,仅转换类型,是否可避开复制,从而提升性能?
从 ptype 输出的结构来看,string 可看做 [2]uintptr,而 [ ]byte 则是 [3]uintptr,这便于我们编写代码,无需额外定义结构类型。如此,str2bytes 只需构建 [3]uintptr{ptr, len, len},而 bytes2str 更简单,直接转换指针类型,忽略掉 cap 即可。
通过unsafe.Pointer(指针转换)和uintptr(指针运算)实现转换
package mainimport ( "fmt" "strings" "unsafe")func str2bytes(s string) []byte { x := (*[2]uintptr)(unsafe.Pointer(&s)) h := [3]uintptr{x[0], x[1], x[1]} return *(*[]byte)(unsafe.Pointer(&h))}func bytes2str(b []byte) string { return *(*string)(unsafe.Pointer(&b))}func main() { s := strings.Repeat("abc", 3) b := str2bytes(s) s2 := bytes2str(b) fmt.Println(b, s2)}没有出现逃逸现象
package mainimport ( "testing" "io/ioutil" "time" "fmt")var s, _ = ioutil.ReadFile("mydata4vipday.720.datx")func test() { b := string(s) _ = []byte(b)}func test2() { b := bytes2str(s) _ = str2bytes(b)}func BenchmarkTest(b *testing.B) { t1 := time.Now() for i := 0; i < b.N; i++ { test() } fmt.Println("test", time.Now().Sub(t1), b.N)}func BenchmarkTestBlock(b *testing.B) { t1 := time.Now() for i := 0; i < b.N; i++ { test2() } fmt.Println("test block", time.Now().Sub(t1), b.N)}对比一下优化前后的性能差异
没有额外开辟内存0B/op,执行效率:5亿次耗时1.6秒,而不用unsafe.Pointer和uintptr转换300次耗时久达到了1.1秒,效率对比高下立判
以上为个人经验,希望能给大家一个参考,也希望大家多多支持。如有错误或未考虑完全的地方,望不吝赐教。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
go使用动态数组还有点麻烦,比python麻烦一点,需要先定义。动态数组申明vardynaArr[]string动态数组添加成员dynaArr=append(d
flagflag是Go标准库提供的解析命令行参数的包。使用方式:flag.Type(name,defValue,usage)其中Type为String,Int,
1概述字符串,string,一串固定长度的字符连接起来的字符集合。Go语言的字符串是使用UTF-8编码的。UTF-8是Unicode的实现方式之一。Go语言原生
Go语言切片(Slice)Go语言切片是对数组的抽象。Go数组的长度不可改变,在特定场景中这样的集合就不太适用,Go中提供了一种灵活,功能强悍的内置类型切片("
Go语言切片是对数组的抽象。Go数组的长度不可改变,在特定场景中这样的集合就不太适用,Go中提供了一种灵活,功能强悍的内置类型切片("动态数组"),与数组相比切