SONG Shengjie

151.翻转字符串里的单词reverse-words-in-a-string卡码网:55.右旋转字符串28. 实现 strStr()find-the-index-of-the-first-occurrence-in-a-string459.重复的子字符串repeated-substring-pattern字符串总结双指针总结

151.翻转字符串里的单词reverse-words-in-a-string

Leetcode Learning Materials

image

class Solution:
    def reverse(self, s, start, end):
        while start < end:
            s[start], s[end] = s[end], s[start]
            start += 1
            end -= 1
            
    def reverseWords(self, s: str) -> str:
        result = ""
        ss = list(s)
        ss.reverse()
        fast = 0
        # 1. 首先将原字符串反转并且除掉空格, 并且加入到新的字符串当中
        # 由于Python字符串的不可变性,因此只能转换为列表进行处理
        while fast < len(ss):
            if ss[fast] != " ":
                if len(result) != 0:
                    result += " "
                while fast < len(ss) and ss[fast] !=" ":
                    result += ss[fast]
                    fast += 1
            else:
                fast += 1
        
        # 2.其次将每个单词进行翻转操作
        slow = 0
        fast = 0
        result = list(result)
        while fast <= len(result):
            if fast == len(result) or result[fast] == " ":
                self.reverse(result, slow, fast - 1)
                slow = fast + 1
                fast += 1
            else:
                fast += 1

        return "".join(result)

卡码网:55.右旋转字符串

Leetcode Learning Materials

image

k = int(input())
s = input()
print(s[len(s) - k:]+s[:len(s) - k])

28. 实现 strStr()find-the-index-of-the-first-occurrence-in-a-string

Leetcode Learning Materials

image

class Solution:
    def strStr(self, haystack: str, needle: str) -> int:
        next = [0] * len(needle)
        self.getNext(next, needle)
        j = 0

        for i in range(len(haystack)):
            while j > 0 and haystack[i] != needle[j]:
                j = next[j - 1] // j 寻找之前匹配的位置
            if haystack[i] == needle[j]:
                j += 1
            if j == len(needle):
                return i - j + 1
        return -1


    def getNext(self, next: List[int] , s: str) -> None:
        j = 0
        next[0] = 0
        for i in range(1, len(s)):
            while j > 0 and s[i] != s[j]:
                j = next[j - 1]
            if s[i] == s[j]:
                j += 1
            next[i] = j

459.重复的子字符串repeated-substring-pattern

Leetcode Learning Materials

image

class Solution:
    def repeatedSubstringPattern(self, s: str) -> bool:
        n = len(s)
        if n <= 1:
            return False
        substr = ""
        for i in range(1, n // 2 + 1): // 查一半因为如果子字符串长度超过原字符串长度的一半那么它重复后长度必然会超过原字符串所以只需要检查长度不超过原字符串一半的子字符串即可
            if n % i == 0 : //查整除如果 n 不能被 i 整除说明原字符串不可能由长度为 i 的子字符串重复构成
                substr = s[:i] //提取字符串 s 的前 i 个字符作为子字符串 substr
                if substr * (n // i) == s:
                    return True
        return False
class Solution:
    def repeatedSubstringPattern(self, s: str) -> bool:
        n = len(s)
        if n <= 1:
            return False
        
        ss = s + s # 将字符串 s 自身拼接一次
        ss = ss[1:-1] # 去掉首尾元素
        return s in ss
class Solution:
    def repeatedSubstringPattern(self, s: str) -> bool:
        next = [0] * len(s)
        self.getNext(next, s)
        if next[-1] != 0 and len(s) % (len(s) - next[-1]) == 0:
            return True
        return False


    def getNext(self, next, s):
        j = 0
        next[0] = 0
        for i in range(1, len(s)):
            while j > 0 and s[i] != s[j]:
                j = next[j - 1]
            if s[i] == s[j]:
                j += 1
            next[i] = j
        return next

字符串总结

Interpretation

前缀表:起始位置到下标i之前(包括i)的子串中,有多大长度的相同前缀后缀。

前缀:指不包含最后一个字符的所有以第一个字符开头的连续子串。

后缀:指不包含第一个字符的所有以最后一个字符结尾的连续子串。

双指针总结

十道题目:移除元素、反转字符串、替换数字、翻转字符串里的单词、翻转链表、删除链表的倒数第N个节点、链表相交、环形链表Ⅱ、三数之和、四数之和。

一般是将O(n^2)的时间复杂度,降为 $O(n)$ 。