字符串类型string是一个不可变的字符序列,go从底层就采用UTF-8编码。
字符串在内存中是以byte数组存储的,如果是非ASCII字符的其他长度字符(如中文),每个rune就占多个byte。
字符串操作
- 字符串定义
//转义字符\n \r \t \u \U分别表示换行,回车,制表符,Unicode字符
var str string = "测试转义字符: \t hello \u0067\U0000006F \n"
//使用反引号来定义原生字符串,无需转义原样输出
var str1 string = `使用反引号来定义原生字符串: hello
go
你好
golang
`
fmt.Print(str, str1)
//一个ascii字符占1字节,一个中文字符占3字节。
fmt.Printf("字符串'你好,go'的长度:%d", len("你好,go"))
// 输出
测试转义字符: hello go
使用反引号来定义原生字符串:hello
go
你好
golang
字符串'你好,go'的长度:9
== != < <= > >=
s1, s2 := "你好,go", "你好,\u0067\U0000006F"
fmt.Println("s1 == s2? ", s1 == s2)
//输出:true
- 字符串长度
fmt.Printf("内存长度:%d\n", len("你好,go")) //输出 9
fmt.Printf("rune长度:%d\n", utf8.RuneCountInString("你好,go")) //输出 5
- 字符访问
fmt.Printf("字符串'你好,go'的第8个byte为:%c\n", "你好,go"[7]) //输出: g
//取rune,先将string转为rune数组,在按索引取
fmt.Printf("字符串'你好,go'的第2个rune为:%v\n", string([]rune("你好,go")[1])) //输出:好
- 字符串拼接
字符串拼接是将两个字符串拼接在一起,产生一个新的字符串并返回,注意字符串本身是不可变的
有三种方式进行拼接:
//方法一:使用+拼接,效率相对较低,适用于少量字符串拼接
fmt.Print("hello " + " go" + "\n")
//方法二:使用bytes.Buffer拼接,效率高,节省内存
var stringBuilder bytes.Buffer
stringBuilder.WriteString("你好")
stringBuilder.WriteString("go")
fmt.Println(stringBuilder.String())
//方法三:使用fmt.Sprintf
fmt.Println(fmt.Sprintf("%s %s", "你好,", "go"))
- 字符串遍历
对于ASCII序列,使用for或for…range遍历都可以,对于非ASCII的Unicode使用for…range循环遍历。
s3 := "hello,go"
s4 := "你好,go"
//遍历byte序列,对于ASCII序列,这样遍历与预期一致
fmt.Print("遍历byte序列:")
for i := 0; i < len(s3); i++ {
fmt.Printf("%c\t", s3[i])
}
fmt.Printf("\n%s", "遍历rune序列:")
//遍历rune序列,对于非ASCII遍历,需要使用这种方式遍历
//这里index返回的是rune字符第一个byte的index
for index, c := range s4 {
fmt.Printf("s4[%d]=%c\t", index, c)
}
fmt.Println()
- 子串查找
//子串查找
fmt.Printf("第一个lo的索引:%d,", strings.Index("hello hello go", "lo")) //输出 3
fmt.Printf("截取第一个lo之前的子串:%s, ", s5[:strings.Index(s5, "lo")]) //输出 hel
fmt.Printf("最后一个lo的索引:%d", strings.LastIndex("hello hello go", "lo")) //输出9
练习1 字符串修改
[ ]byte
//修改字符串,将s5的第一个hello改成xxxxx
bytes := []byte(s5)
for i := 0; i < 5; i++ {
bytes[i] = 'x'
}
s5 = string(bytes)
fmt.Println(s5)
练习2 字符串替换
//将"hello go,我正在学习"中的"go"替换为"world"
s6 := "hello go,我正在学习"
src := "go"
target := "world"
var stringBuilder1 bytes.Buffer
index := strings.Index(s6, src)
stringBuilder1.WriteString(s6[:index])
stringBuilder1.WriteString(target)
stringBuilder1.WriteString(s6[index+len(src):])
fmt.Println(stringBuilder1.String())
总结
主要介绍了字符串相关操作。
完整示例代码
package main
import (
"bytes"
"fmt"
"strings"
"unicode/utf8"
)
func main() {
//string使用,转义字符\n \r \t \u \U分别表示换行,回车,制表符,Unicode字符
var str string = "测试转义字符: \t 你好,\u0067\U0000006F\n"
//使用反引号来定义原生字符串,无需转义原样输出
var str1 string = `使用反引号来定义原生字符串: 你好,
go
`
fmt.Print(str, str1)
//字符串比较
s1, s2 := "你好,go", "你好,\u0067\U0000006F"
fmt.Println("s1 == s2? ", s1 == s2)
//一个ascii字符占1字节,一个中文字符占3字节。
fmt.Printf("字符串'你好,go'的内存长度:%d\n", len("你好,go"))
fmt.Printf("字符串'你好,go'的rune长度:%dv\n", utf8.RuneCountInString("你好,go"))
//字符访问
fmt.Printf("字符串'你好,go'的第8个byte为:%c\n", "你好,go"[7])
fmt.Printf("字符串'你好,go'的第2个rune为:%v\n", string([]rune("你好,go")[1]))
//字符串拼接
//方法一:使用+拼接,效率相对较低,适用于少量字符串拼接
fmt.Print("hello " + " go" + "\n")
//方法二:使用bytes.Buffer拼接,效率高,节省内存
var stringBuilder bytes.Buffer
stringBuilder.WriteString("你好")
stringBuilder.WriteString("go")
fmt.Println(stringBuilder.String())
//方法三:使用fmt.Sprintf
fmt.Println(fmt.Sprintf("%s %s", "你好,", "go"))
//遍历
s3 := "hello,go"
s4 := "你好,go"
//遍历byte序列,对于ASCII序列,这样遍历与预期一致
fmt.Print("遍历byte序列:")
for i := 0; i < len(s3); i++ {
fmt.Printf("%c\t", s3[i])
}
fmt.Printf("\n%s", "遍历rune序列:")
//遍历rune序列,对于非ASCII遍历,需要使用这种方式遍历
//这里index返回的是rune字符第一个byte的index
for index, c := range s4 {
fmt.Printf("s4[%d]=%c\t", index, c)
}
fmt.Println()
//子串查找
s5 := "hello hello go"
fmt.Printf("第一个lo的索引:%d, ", strings.Index(s5, "lo"))
fmt.Printf("截取第一个lo之前的子串:%s, ", s5[:strings.Index(s5, "lo")])
fmt.Printf("最后一个lo的索引:%d", strings.LastIndex(s5, "lo"))
}
输出
测试转义字符: 你好,go
使用反引号来定义原生字符串: 你好,
go
s1 == s2? true
字符串'你好,go'的内存长度:9
字符串'你好,go'的rune长度:5v
字符串'你好,go'的第8个byte为:g
字符串'你好,go'的第2个rune为:好
hello go
你好go
你好, go
遍历byte序列:h e l l o , g o
遍历rune序列:s4[0]=你 s4[3]=好 s4[6]=, s4[7]=g s4[8]=o
第一个lo的索引:3, 截取第一个lo之前的子串:hel, 最后一个lo的索引:9%