「算」 05. 字符串

本文最后更新于:24 天前

字符串相关题目。

1 概念

字符串在 Go 语言中是不可变的,这意味着一旦创建了一个字符串,就不能直接修改它的内容。如果需要修改字符串,可以先将其转换为字节数组([]byte)进行修改,然后再转换回字符串。

Go 使用的 UTF-8 变长编码,每个字符所占的字节是不一样的:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
func utf_8Test() {
str := "英文名为Jamison"
fmt.Println(len(str)) // 19
// 四个汉字每个汉字三个字节,其余英文每个一字节
// 所以含有一个字节以上的字符不能使用传统 for 循环遍历,应该使用 for range
for _, s := range str {
fmt.Printf("%c", s) // result: 英文名为Jamison
}

// 错误做法
for i := 0; i < len(str); i++ {
fmt.Printf("%c", str[i]) // result: è±æå为Jamison
}
}
  • 对字符串使用 len 方法得到的是字节数不是字符数

  • 对字符串直接使用下标访问,得到的是字节而不是字符

  • 字符串被 range 遍历时,被解码为 rune 类型的字符

  • UTF-8 编码解码算法位于 runtime/utf8.go

2 题目

2.1 leetcode 344 题

原地修改输入数组,反转字符串。

1
2
3
4
5
6
7
8
9
func reverseString(s []byte) {
left := 0
right := len(s)-1
for left < right {
s[left], s[right] = s[right], s[left]
left++
right--
}
}

2.2 [剑指 Offer 05]

把字符串 s 中的每个空格替换成”%20”。

如果想把这道题目做到极致,首先扩充数组到每个空格替换成”%20”之后的大小,然后双指针法从后向前替换空格。

从前向后填充就是 O(n^2) 的算法了,因为每次添加元素都要将添加元素之后的所有元素向后移动。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
func replaceSpace(s string) string {
b := []byte(s)
length := len(b)
spaceCount := 0
// 计算空格数量
for _, v := range b {
if v == ' ' {
spaceCount++
}
}
// 扩展原有切片
resizeCount := spaceCount * 2
tmp := make([]byte, resizeCount)
b = append(b, tmp...)
i := length - 1
j := len(b) - 1
for i >= 0 {
if b[i] != ' ' {
b[j] = b[i]
i--
j--
} else {
b[j] = '0'
b[j-1] = '2'
b[j-2] = '%'
i--
j = j - 3
}
}
return string(b)
}

2.3 leetcode 151 题

翻转字符串里的单词,不使用辅助空间,空间复杂度要求为 O(1):

  1. 移除多余空格
  2. 将整个字符串反转
  3. 将每个单词反转
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
func reverseWords(s string) string {
fast, slow := 0, 0
b := []byte(s)
lens := len(b)

// top sp
for ; fast < lens; fast++ {
if b[fast] != ' ' {
break
}
}

// mid sp
for ; fast < lens; fast++ {
if fast-1 > 0 && b[fast]==' ' && b[fast-1]==' ' {
continue
}
b[slow] = b[fast]
slow++
}

// end sp
if b[slow-1] == ' ' {
b = b[:slow-1]
} else {
b = b[:slow]
}

reverseStr(b, 0, len(b)-1)

for i, j := 0, 0; j <= len(b); j++ {
if j == len(b) {
reverseStr(b, i, len(b)-1)
break
}
if b[j] == ' ' {
reverseStr(b, i, j-1)
i = j + 1
}
}

return string(b)
}

func reverseStr(b []byte, begin int, end int) {
for i, j := begin, end; i < j; i, j = i+1, j-1 {
(b)[i], (b)[j] = (b)[j], (b)[i]
}
}

注:在 Go 中,字符串是不可变的,也就是说无法直接通过下标操作来修改字符串的内容。因此,无法直接通过下标操作来实现字符串的反转。要反转字符串,可以将字符串转换为字节数组([]byte)。

2.4 leetcode 3 题

无重复字符的最长子串。

滑动窗口。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
func lengthOfLongestSubstring(s string) (res int) {
b := []byte(s)
bmap := make(map[byte]int)
i := 0
for j := 0; j < len(b); j++ {
bmap[b[j]]++
if bmap[b[j]] > 1 {
for bmap[b[i]] != bmap[b[j]] {
bmap[b[i]]--
i++
}
bmap[b[i]]--
i++
}
if (j-i+1) > res {
res = j-i+1
}
}
return
}

「算」 05. 字符串
https://qanlyma.github.io/Algorithm-string/
作者
Qanly
发布于
2023年4月12日