字符串
严格来说,Go字符串并不是组合类型,但是Go提供了很多直接操作字符串的函数,第3章提到过Go字符串是值类型,并非像C字符串需要靠指针实现。另外,Go默认支持UTF-8字符串,处理Unicode编码的字符就会非常方便。下面的小节也会讲到字符、rune、字节之间的区别,以及字符串与字符串字面量的不同。
Go字符串实际上是一个字节切片,可以存储任意类型、任意长度的字节。
可以像下面一样定义一个字符串:
const sLiteral = “\x99\x42\x32”
也可以使用字符串字面量:
s2 := “rfsdf”
可以使用len()
得到字符串的长度。
strings.go
将会分4部分展示一些Go字符串的标准操作。
第一部分:
package main
import (
"fmt"
)
func main() {
const sLiteral= "\x99\x42\x32\x55\x50\x35\x23\x50\x29\x9c"
fmt.Println(sLiteral)
fmt.Printf("x: %x\n", sLiteral)
fmt.Printf("sLiteral length: %d\n", len(sLiteral))
每一个\xAB
都代表sLiteral
的一个字符,所以调用len()
就会得到sLiteral
的字符数量。%x
会得到\xAB
中的AB
部分。
第二部分代码:
for i := 0; i < len(sLiteral); i++ {
fmt.Printf("%x ", sLiteral[i])
}
fmt.Println()
fmt.Printf("q: %q\n", sLiteral)
fmt.Printf("+q: %+q\n", sLiteral)
fmt.Printf(" x: % x\n", sLiteral)
fmt.Printf("s: As a string: %s\n", sLiteral)
如代码所示,你可以像操作一个切片那样去操作字符串。使用%q
作为字符串格式化参数,可以安全地打印出带双引号的字符串,%+q
可以保证输出是ASCII格式。
最后,% x
(注意%与x之间的空格)将会在输出的字符之间加上空格,如果你想打印字符串格式,就要使用%s
。
第三部分代码:
s2 := "€£³"
for x, y := range s2 {
fmt.Printf("%#U starts at byte position %d\n", y, x)
}
fmt.Printf("s2 length: %d\n", len(s2))
在这里定义了字符串s2
,内容是3个unicode字符。使用%#U
可以打印出U+0058
格式的字符,range
关键字能够迭代包含Unicode字符的字符串,这样就可以逐个处理Unicode字符。
len(s2)
的输出可能会令你困惑,解释一下,s2
包含的是Unicode字符,Unicode字符的字节数量是大于该字符串中的元素数量的,而len()
函数计算的是字节数量,所以len(s2)
的值是7而不是3。
最后一部分:
const s3= "ab12AB"
fmt.Println("s3:", s3)
fmt.Printf("x: % x\n", s3)
fmt.Printf("s3 length: %d\n", len(s3))
for i := 0; i < len(s3); i++ {
fmt.Printf("%x ", s3[i])
}
fmt.Println()
}
执行strings.go
的输出是:
go run strings.go
�B2UP5#P)�
x: 9942325550352350299c
sLiteral length: 10
99 42 32 55 50 35 23 50 29 9c
q: “\x99B2UP5#P)\x9c”
+q: “\x99B2UP5#P)\x9c”
x: 99 42 32 55 50 35 23 50 29 9c
s: As a string: �B2UP5#P)�
U+20AC ‘€’ starts at byte position 0
U+00A3 ‘£’ starts at byte position 3
U+00B3 ‘³’ starts at byte position 5
s2 length: 7
s3: ab12AB
x: 61 62 31 32 41 42
s3 length: 6
61 62 31 32 41 42
在不清楚Unicode和UTF-8机制的情况下,你可能会对本节的输出干到困惑,不必恐慌,因为这些在平时的开发中很少使用,基本的fmt.Printf()
和fmt.Println()
就能满足你的需求。