-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathslide_circuits2.py
62 lines (57 loc) · 2.14 KB
/
slide_circuits2.py
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
# Copyright (c) 2021 kamyu. All rights reserved.
#
# Google Code Jam 2021 Virtual World Finals - Problem B. Slide Circuits
# https://codingcompetitions.withgoogle.com/codejam/round/0000000000436329/000000000084f7b2
#
# Time: O(B + N + SlogS)
# Space: O(B + SlogS)
#
# xor of random values solution (faster)
#
from random import seed, randint
from operator import xor
def random_id_gen(B, ids_set):
MAX_RAND_ID = MAX_UINT64//(B*2)
for _ in xrange(B):
x = randint(1, MAX_RAND_ID)
while True:
x = randint(1, MAX_RAND_ID)
if x not in ids_set:
break
ids_set.add(x)
yield x
def get_sum(prefix, L, R, M):
return sub(prefix[M][R//M], prefix[M][(L-1)//M])
def slide_circuits():
B, S, N = map(int, raw_input().strip().split())
ids_set = set()
in_ids = [i for i in random_id_gen(B, ids_set)]
out_ids = [i for i in random_id_gen(B, ids_set)]
total_hash = add(reduce(add, in_ids), reduce(add, out_ids))
slide_hashes = [0]*S
lookup = {}
for i in xrange(S):
X, Y = map(int, raw_input().strip().split())
slide_hashes[i] = add(out_ids[X-1], in_ids[Y-1])
lookup[sub(total_hash, slide_hashes[i])] = i # 0-indexed
prefix = [[0] for _ in xrange(S+1)]
for m in xrange(1, S+1): # sum of harmonic series: O(S/1 + S/2 + ... + S/S) = O(S * (1 + 1/2 + ... + 1/S)) = O(SlogS)
for i in xrange(m, S+1, m):
prefix[m].append(add(prefix[m][-1], slide_hashes[i-1]))
result = [0]*N
total = curr_hash = 0
for i in xrange(N):
A, L, R, M = raw_input().strip().split()
L, R, M = int(L), int(R), int(M)
total = TOTAL_OP[A](total, R//M-(L-1)//M)
curr_hash = HASH_OP[A](curr_hash, get_sum(prefix, L, R, M))
if total == B-1 and curr_hash in lookup: # check total to avoid hash collision
result[i] = lookup[curr_hash]+1
return " ".join(map(lambda x: str(x) if x else "X", result))
seed(0)
MAX_UINT64 = 2**64-1
add = sub = xor
TOTAL_OP = {'E':lambda a,b: a+b, 'D':lambda a, b:a-b}
HASH_OP = {'E':add, 'D':sub}
for case in xrange(input()):
print 'Case #%d: %s' % (case+1, slide_circuits())