目录
- 题目:最长公共前缀
- 解法1:纵向对比-循环内套循环写法
- 解法2:横向对比-两两对比(类似合并K个数组、合并K个链表)
- 题目:压缩字符串
- 解法1:读写指针 + 次数统计 + 顺向思维处理次数输出
- 解法2:读写起始三指针 + 逆向思维处理次数输出(优化解法1)
题目:最长公共前缀
题目链接:LeetCode-14. 最长公共前缀
解法1:纵向对比-循环内套循环写法
以第一个子字符串为标准,遍历其每个字符时,内嵌遍历其余子字符串的对应字符是否一致。不一致时,则返回当前字符之前的子字符串。
复杂度:时间复杂度 O ( n ∗ m ) O(n*m) O(n∗m)、空间复杂度 O ( 1 ) O(1) O(1)
- 时间复杂度:其中 n 是字符串平均长度,m 是字符串数组的长度。
Go代码
func longestCommonPrefix(strs []string) string {
length := len(strs)
if length==1 {
return strs[0]
}
chlen := len(strs[0])
for i:=0; i<chlen; i++ {
// 用第一个子字符串的字符作为比较
ch := strs[0][i]
for j, v := range strs {
if j == 0 {
continue
}
// 子字符串的长度比第一个短 或者 出现比较字符不同
if i == len(v) || v[i] != ch {
return string(strs[0][:i])
}
}
}
return strs[0]
}
解法2:横向对比-两两对比(类似合并K个数组、合并K个链表)
- 先让前两个子字符串对比得到公共前缀
- 再将此公共前缀作和下一个子字符串对比得到公共前缀
- 如此循环到末尾,返回最后的公共前缀即可
- 优化:遍历过程中如果公共前缀已经是空字符串了,则可直接返回空字符串。
相比于解法1的优点:将逻辑分解到两个函数中,使代码更加模块化和可维护。
复杂度:时间复杂度 O ( n ∗ m ) O(n*m) O(n∗m)、空间复杂度 O ( 1 ) O(1) O(1)
- 时间复杂度:其中 n 是字符串平均长度,m 是字符串数组的长度。
Go代码
func longestCommonPrefix(strs []string) string {
length := len(strs)
if length == 1 {
return strs[0]
}
str := strs[0]
for i:=1; i<length; i++ {
str = getCommonPrefix(str, strs[i])
if str == "" {
return ""
}
}
return str
}
func getCommonPrefix(str1, str2 string) string {
length2 := len(str2)
for i, v := range str1 {
val := byte(v)
if i == length2 || val != str2[i] {
return string(str1[:i])
}
}
return str1
}
题目:压缩字符串
题目链接:LeetCode-443. 压缩字符串
解法1:读写指针 + 次数统计 + 顺向思维处理次数输出
-
安排读写指针,分别指向要写的位置,和读的位置,两指针首先指向第一个字符
-
如果是相同时最后一个元素的标记,或者
读指针的字符和写指针不同时
:- 统计总次数,区分是否 >1
- 大于1:区分是否 >= 10
- 小于10:仅追加一个数字
- 大于10:不断除10,得到可除次数,和除10的最后余数
- 追加最后余数
- 遍历可除次数,追加数值0
- 根据可除次数,用总次数除了10的可除次数次方,得到余数,追加余数
- 大于1:区分是否 >= 10
- 如果是相同时最后一个元素的标记,到这里直接返回写指针+1即可
- 更新对比字符为当前字符,追加对比字符
- 总次数还原为1,读指针++
- 如果是最后一个元素,到这里直接返回写指针+1即可
- 统计总次数,区分是否 >1
-
相同时,总次数++
- 如果是最后一个元素,添加标记(这里只为加上其相同次数)
复杂度:时间复杂度 O ( n ) O(n) O(n)、空间复杂度 O ( 1 ) O(1) O(1)
Go代码
func compress(chars []byte) int {
length := len(chars)
if length == 0 || length == 1 {
return length
}
left, right := 0, 1 // left写 right读
num := 1 //次数统计
ch := chars[0]
lastone := false
for lastone || right < length {
if lastone || chars[right] != chars[left] {
// 追加长度
if num > 1 {
// 要追加多个长度
if num >= 10 {
tmpnum := num
count10 := 1
// 明确有几个10
for tmpnum >= 10 {
tmpnum = tmpnum/10
if tmpnum >= 10 {
count10++
}
}
// 追加大于10的首个数字
left++
chars[left] = byte(tmpnum + '0')
for i:=1; i<count10; i++ {
// 补零
left++
chars[left] = '0'
}
// 追加大于等于10时的余数
f10 := math.Pow(10, float64(count10))
yu := num % int(f10)
if yu >= 0 {
left++
chars[left] = byte(yu + '0')
}
} else { //只追加一个长度
left++
chars[left] = byte(num + '0')
}
}
if lastone {
return left+1
}
ch = chars[right]
// 追加新字符
left++
chars[left] = ch
num = 1
right++
// 如果是最后一个元素
if right == length {
return left+1
}
} else {
num++
right++
if right == length {
lastone = true
}
}
}
return left+1
}
解法2:读写起始三指针 + 逆向思维处理次数输出(优化解法1)
优化点:
- 无需用变量循环时累加统计总次数:让一个
起始指针
指向当前统计字符的首位置,当遍历到该字符末尾时,尾索引-首索引+1=总次数
。 - 将正向计算追加>10时的最后余数、可除10次数、余数改为:追加依次除10得到的余数,最后
反转
该段数字区间即可。 读指针和下一个不同时处理
当前,此时包含了最后一个字符的场景。解法1中读指针和写指针不同时处理的逻辑就没法包含最后一个字符,需要考虑最后一个字符时是和之前一致还是不同的情况。
复杂度:时间复杂度 O ( n ) O(n) O(n)、空间复杂度 O ( 1 ) O(1) O(1)
Go代码
func compress(chars []byte) int {
length := len(chars)
if length == 0 || length == 1 {
return length
}
left, right, start := 0, 0, 0
for ; right < length; right++ {
if right == length-1 || chars[right] != chars[right+1] {
chars[left] = chars[right]
left++
count := right-start+1
site := left
if count > 1 {
for count > 0 {
n := count%10
chars[left] = byte(n + '0')
left++
count = count/10
}
reverse(chars, site, left-1)
}
start = right+1
}
}
return left
}
func reverse(chars []byte, left int, right int) {
if left >= right {
return
}
for left <= right {
chars[left], chars[right] = chars[right], chars[left]
left++
right--
}
}