LeetCode 3335. Total Characters in String After Transformations I
weekly contest 421。
竟然在 Q2 就有 dp,受到驚嚇。
題目
輸入字串 s 和 整數 t,代表轉換次數。
每次轉換須按照以下規則替換字元:
- 若字元是 ‘z’,則將其替換成 “ab”。
- 否則替換成字母表中的下個字元。例如 ‘a’ 變成 ‘b’,’b’ 變成 ‘c’,以此類推。
求字串進行正好 t 次轉換後的長度。
答案可能很大,先模 10^9 + 7 後回傳。
解法
字串中的所有字元都是獨立的,彼此互不影響。
例如 “aaa” 可以看做是三個 “a” 分別轉換 t 次。
而 “z” 轉換後則拆成兩個 “a” 和 “b” 分別轉換 t-1 次。
不同的字元都有相同的結果,是重疊的子問題,考慮 dp。
首先統計 s 中各自元的頻率,然後模擬轉換 t 次。
因為轉換是線性的,產生下一個時間點的字元,所以我覺得填表法比較直覺。
“z” 會額外對 “b” 產生貢獻,記得另外補上。
時間複雜度 O(N + 26t)。
空間複雜度 O(26)。
MOD = 10 ** 9 + 7
class Solution:
def lengthAfterTransformations(self, s: str, t: int) -> int:
f = [0] * 26
for c in s:
f[ord(c)-97] += 1
for _ in range(t):
f2 = [0] * 26
# a~z to next char
for i in range(26):
f2[(i+1)%26] = f[i]
# z to extra b
f2[1] += f[25]
f = f2
return sum(f) % MOD
改成熟悉的填表法,代表時間點 t 的字元 i 是由時間點 t-1 的哪些來源產生而來。
其實大同小異。
MOD = 10 ** 9 + 7
class Solution:
def lengthAfterTransformations(self, s: str, t: int) -> int:
f = [0] * 26
for c in s:
f[ord(c)-97] += 1
for _ in range(t):
f2 = [0] * 26
# a~z from prev char
for i in range(26):
f2[i] = f[i-1]
# extra b from z
f2[1] += f[25]
f = f2
return sum(f) % MOD