力扣第 362 场周赛第 4 题
题目
给你两个长度都为 n 的字符串 s 和 t 。你可以对字符串 s 执行以下操作:
- 将
s 长度为 l (0 < l < n)的 后缀字符串 删除,并将它添加在 s 的开头。
比方说,s = 'abcd' ,那么一次操作中,你可以删除后缀 'cd' ,并将它添加到 s 的开头,得到 s = 'cdab' 。
给你一个整数 k ,请你返回 恰好 k 次操作将 s 变为 t 的方案数。
由于答案可能很大,返回答案对 109 + 7 取余 后的结果。
示例 1:
输入:s = "abcd", t = "cdab", k = 2
输出:2
解释:
第一种方案:
第一次操作,选择 index = 3 开始的后缀,得到 s = "dabc" 。
第二次操作,选择 index = 3 开始的后缀,得到 s = "cdab" 。
第二种方案:
第一次操作,选择 index = 1 开始的后缀,得到 s = "bcda" 。
第二次操作,选择 index = 1 开始的后缀,得到 s = "cdab" 。
示例 2:
输入:s = "ababab", t = "ababab", k = 1
输出:2
解释:
第一种方案:
选择 index = 2 开始的后缀,得到 s = "ababab" 。
第二种方案:
选择 index = 4 开始的后缀,得到 s = "ababab" 。
提示:
2 <= s.length <= 5 * 105
1 <= k <= 1015
s.length == t.length
s 和 t 都只包含小写英文字母。
分析
- 先用字符串匹配求出使得 s[i:]+s[:i]=t 的 i 的个数 c
- 然后令 f(k,0/1) 分别代表 k 次操作后 s 等于/不等于 t 的方案数,可以递推
- f(k,0) = f(k-1,0)*(c-1)+f(k-1,1)*c
- f(k,1) = f(k-1,0)(n-c)+f(k-1,1)(n-c-1)
- 由于 k 很大,用矩阵快速幂优化
解答
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
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
|
mod = 10**9+7
def kmp(s):
n = len(s)
pi,j = [0]*n,0
for i in range(1,n):
while j>0 and s[i]!=s[j]:
j = pi[j-1]
j += s[i]==s[j]
pi[i] = j
return pi # pi[i]:i结尾的最大真前缀长度
class MatPow:
def __init__(self,A): # k 阶递推式需要给定前 k*2 项
k = len(A)//2
self.f = A[:k]
self.A = A
self.g = self.gen(A)[::-1]
def gen(self,A): # Berlekamp-Massey 算法,给定前 k*2 项 A,返回符合的最短系数组 g
pre_c = []
pre_i, pre_d = -1, 0
g = []
for i,a in enumerate(A):
d = (a-sum(x*A[i-1-j] for j,x in enumerate(g)))%mod
if d == 0:
continue
if pre_i<0: # 首次算错,初始化 g 为 i+1 个 0
g = [0]*(i+1)
pre_i,pre_d = i,d
continue
bias = i-pre_i
old_len = len(g)
new_len = bias + len(pre_c)
if new_len>old_len: # 递推式变长了
tmp = g[:]
g += [0]*(new_len-old_len)
delta = d*pow(pre_d,-1,mod)%mod
g[bias-1] = (g[bias-1]+delta)%mod
for j,c in enumerate(pre_c):
g[bias+j] = (g[bias+j]-delta*c)%mod
if new_len>old_len:
pre_c = tmp
pre_i,pre_d = i,d
return g
def get(self,n): # Kitamasa 算法,给定前 k 项 f 和系数组 g,求第 n 项
def compose(A,B): # 根据 g(n) 的系数组 A 和 g(m) 的系数组 B 计算 g(n+m) 的系数组
C = [0]*k
for a in A:
for j,b in enumerate(B):
C[j] = (C[j]+a*b)%mod
B = [((B[i-1] if i else 0)+B[-1]*g[i])%mod for i in range(k)]
return C
f,g = self.f,self.g
if n<len(f):
return f[n]%mod
k = len(g)
if k == 0:
return 0
if k == 1:
return f[0]*pow(g[0],n,mod)%mod
res = [0]*k
C = [0]*k
res[0] = C[1] = 1
while n:
res = compose(C,res) if n&1 else res
C = compose(C,C)
n >>= 1
return sum(a*b for a,b in zip(res,f))%mod
class Solution:
def numberOfWays(self, s: str, t: str, k: int) -> int:
n = len(s)
pi = kmp(t+'#'+s+s[:-1])
c = pi.count(n)
a = s==t
b = 1-a
A = [a]
for _ in range(3):
a,b = a*(c-1)+b*c,a*(n-c)+b*(n-1-c)
a,b = a%mod,b%mod
A.append(a)
mp = MatPow(A)
return mp.get(k)
|
947 ms