-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathpalindromic_deletions.py3
36 lines (33 loc) · 1.36 KB
/
palindromic_deletions.py3
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
# Copyright (c) 2022 kamyu. All rights reserved.
#
# Google Kick Start 2022 Round C - Problem D. Palindromic Deletions
# https://codingcompetitions.withgoogle.com/kickstart/round/00000000008cb4d1/0000000000b20d16
#
# Time: O(N^3), pass in PyPy3 but Python3
# Space: O(N^2)
#
def inv_nCr(n, k):
while len(INV) <= n:
FACT.append(FACT[-1]*len(INV) % MOD)
INV.append(INV[MOD%len(INV)]*(MOD-MOD//len(INV)) % MOD)
INV_FACT.append(INV_FACT[-1]*INV[-1] % MOD)
return (INV_FACT[n]*FACT[n-k] % MOD) * FACT[k] % MOD
def palindromic_deletions():
N = int(input())
S = input()
result = 1
dp = [[[0]*N for _ in range(N)] for _ in range(L)] # dp[l%L][i][j]: number of palindromes of length l in the substring of S[i, j]
for l in range(1, N):
for i in reversed(range(N)):
dp[l%L][i][i] = (1 if l == 1 else 0)
for j in range(i+1, N):
dp[l%L][i][j] = (dp[l%L][i+1][j]+dp[l%L][i][j-1]-dp[l%L][i+1][j-1])%MOD # inclusion-exclusion principle
if S[i] == S[j]:
dp[l%L][i][j] = (dp[l%L][i][j]+(1 if l == 2 else dp[(l-2)%L][i+1][j-1]))%MOD
result = (result+dp[l%L][0][N-1]*inv_nCr(N, l))%MOD
return result
L = 3
MOD = 10**9+7
FACT, INV, INV_FACT = [[1]*2 for _ in range(3)]
for case in range(int(input())):
print('Case #%d: %s' % (case+1, palindromic_deletions()))