-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsushi_platter.py3
119 lines (113 loc) · 5.79 KB
/
sushi_platter.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
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
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
# Copyright (c) 2024 kamyu. All rights reserved.
#
# Meta Hacker Cup 2024 Final Round - Problem D. Sushi Platter
# https://www.facebook.com/codingcompetitions/hacker-cup/2024/final-round/problems/D
#
# Time: O(NlogN + MlogM + N^2 * (N * MAX_A) + M * min(L, N * MAX_A) + M! * ((M - 1) * 2^(M - 1))), MAX_A = max(A) = 100
# = O(N^3 * MAX_A + M! * ((M - 1) * 2^(M - 1))), if L >= N * MAX_A and N >= M
# Space: O(N^2 * MAX_A + M * min(L, N * MAX_A)
# = O(N^2 * MAX_A), if L >= N * MAX_A and N >= M
#
from itertools import permutations
def popcount(x):
return bin(x).count('1')
def sushi_platter():
# reference: https://codeforces.com/blog/entry/45593
# modified from template: https://ideone.com/bj9pm9
def fill_A():
A.sort()
dp = [[[] for _ in range(3)] for _ in range(N+1)]
mx = 0
if (nv := A[1]-A[0]) <= L:
if nv >= (l := len(dp[1][1])):
if nv > mx:
mx = nv
dp[1][1].extend((0 for _ in range((nv+1)-l)))
dp[1][1][nv] = 2
if (nv := 2*(A[1]-A[0])) <= L:
if nv >= (l := len(dp[1][0])):
if nv > mx:
mx = nv
dp[1][0].extend((0 for _ in range((nv+1)-l)))
dp[1][0][nv] = 1
for i in range(1, N):
new_dp = [[[] for _ in range(3)] for _ in range(N+1)]
diff = (A[i+1] if i+1 < N else A[-1]+1)-A[i]
for j in reversed(range(1, i+1)):
for k in range(3):
for v in reversed(range(len(dp[j][k]))):
if dp[j][k][v] == 0:
continue
if (nv := v+diff*(2*j-k+2)) <= L: # creating a new connected component with a non-endpoint
if nv >= (l := len(new_dp[j+1][k])):
if nv > mx:
mx = nv
new_dp[j+1][k].extend((0 for _ in range((nv+1)-l)))
new_dp[j+1][k][nv] = (new_dp[j+1][k][nv]+dp[j][k][v]) % MOD
if k <= 1 and (nv := v+diff*(2*j-k+1)) <= L: # creating a new connected component with a endpoint
if nv >= (l := len(new_dp[j+1][k+1])):
if nv > mx:
mx = nv
new_dp[j+1][k+1].extend((0 for _ in range((nv+1)-l)))
new_dp[j+1][k+1][nv] = (new_dp[j+1][k+1][nv]+dp[j][k][v]*(2-k)) % MOD
if (c := 2*j-k) >= 1 and (nv := v+diff*c) <= L: # appending a non-endpoint to an existing connected component
if nv >= (l := len(new_dp[j][k])):
if nv > mx:
mx = nv
new_dp[j][k].extend((0 for _ in range((nv+1)-l)))
new_dp[j][k][nv] = (new_dp[j][k][nv]+dp[j][k][v]*(2*j-k)) % MOD
if k <= 1 and (c := 2*j-k-1) >= 1 and (nv := v+diff*c) <= L: # appending a endpoint to an existing connected component.
if nv >= (l := len(new_dp[j][k+1])):
if nv > mx:
mx = nv
new_dp[j][k+1].extend((0 for _ in range((nv+1)-l)))
new_dp[j][k+1][nv] = (new_dp[j][k+1][nv]+dp[j][k][v]*(2-k)*(j-k)) % MOD
if j >= 2 and (c := 2*j-k-2) >= 1 and (nv := v+diff*c) <= L: # merging connected components
if nv >= (l := len(new_dp[j-1][k])):
if nv > mx:
mx = nv
new_dp[j-1][k].extend((0 for _ in range((nv+1)-l)))
new_dp[j-1][k][nv] = (new_dp[j-1][k][nv]+dp[j][k][v]*(j-k)*(j-1)) % MOD
dp = new_dp
return dp, mx
def fill_B():
B.sort()
prefix = [[[0]*((min(L, mx)+1)+1) for _ in range(4)] for _ in range(min(M, len(dp))+1)]
for cnt_B in range(1, len(prefix)):
for e in range((3 if cnt_B != 1 else 2)+1):
c = popcount(e)
cnt_A = cnt_B-1 if c == 2 else cnt_B if c == 1 else cnt_B+1
for i in range(min(L, mx)+1):
cnt = dp[cnt_A][2-c][i] if cnt_A < len(dp) and i < len(dp[cnt_A][2-c]) else 0
if e == 1 or e == 2:
cnt = (cnt*INV_2) % MOD
cnt = (cnt*FACT[cnt_B-1]) % MOD
prefix[cnt_B][e][i+1] = (prefix[cnt_B][e][i]+cnt) % MOD
result = 0
for p in permutations(B):
for mask in range(1<<(M-1)):
cnt_B = 1+popcount(mask)
total = sum(abs(p[i+1]-p[i]) for i in range(M-1) if mask&(1<<i) == 0)
for e in range((3 if cnt_B != 1 else 2)+1):
new_total = total
for i in range(M):
if (i > 0 and mask&(1<<(i-1))) or (i == 0 and e != 1 and e != 3):
new_total += p[i]-(A[-1]+1)
if (i < M-1 and mask&(1<<i)) or (i == M-1 and e != 2 and e != 3):
new_total += p[i]-(A[-1]+1)
if cnt_B < len(prefix) and (mn := min(L-new_total, mx)) >= 0:
result = (result+prefix[cnt_B][e][mn+1]) % MOD
return result
N, M, L = list(map(int, input().split()))
A = list(map(int, input().split()))
B = list(map(int, input().split()))
dp, mx = fill_A()
return fill_B()
MOD = 10**9+7
INV_2 = pow(2, MOD-2, MOD)
MAX_M = 5
FACT = [1]
while MAX_M-1 >= len(FACT):
FACT.append(FACT[-1]*len(FACT) % MOD)
for case in range(int(input())):
print('Case #%d: %s' % (case+1, sushi_platter()))