-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcoin_change.py3
58 lines (52 loc) · 1.81 KB
/
coin_change.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
# Copyright (c) 2024 kamyu. All rights reserved.
#
# Meta Hacker Cup 2024 Round 3 - Problem C: Coin Change
# https://www.facebook.com/codingcompetitions/hacker-cup/2024/round-3/problems/C
#
# Time: O(min(N, THRESHOLD))
# Space: O(1)
#
from math import log
# https://en.wikipedia.org/wiki/Harmonic_number#Calculation
GAMMA = 0.5772156649015328606065120900824
THRESHOLD = 10**8
def f(N):
return log(N)+GAMMA
def H(N):
return sum(1/i for i in range(1, N+1)) if N < THRESHOLD else f(N)
def ceil_divide(a, b):
return (a+b-1)//b
def coin_change():
N, P = list(map(int, input().split()))
p = P/100
# B1 = N/(1+p) = 100*N/(100+P)
B1 = ceil_divide(100*N, 100+P)
# 1 dollar for [0, B1-1]
# 1 * (N/(N-0) + N/(N-1) + ... + N/(N-(B1-1)))
# = 1 * (N * (1/(N-B1+1) + 1/(N-B1) + ... + 1/(N-0)))
# = 1 * (N * (H(N)-H(N-B1)))
result = 1*(N*(H(N)-H(N-B1)))
if not P:
return result
# (M-1)*p >= 1
# M >= 1/p+1 = 100/P+1
# D = M-1 = 100/P
D = ceil_divide(100, P)
c = 1-(D-1)*p
# D*N/(N-B2*c) >= D+1
# D*N/(D+1) >= N-B2*c
# B2*c >= N-D*N/(D+1) = N/(D+1)
# B2 >= N/(D+1)/c = 100*N/((D+1)*(100-(D-1)*P))
B2 = min(ceil_divide(100*N, (D+1)*(100-(D-1)*P)), N)
# D dollars for [B1, min(N, B2)-1]
# sum(D*N/(N-x*c) for x in range(B1, min(N, B2)))
# = D*N * (1/(N-B1*c) + 1/(N-(B1+1)*c) + ... + 1/(N-(min(N, B2)-1)*c))
# = D*N * (1/(N/c-min(N, B2)+1) + 1/(N/c-min(N, B2)+2)... + 1/(N/c-B1)) / c
# = D*N * (H(N/c-B1)-H(N/c-min(N, B2))) / c
result += D*N*(sum(1/(N-x*c) for x in range(B1, min(N, B2))) if min(N, B2)-B1 < THRESHOLD else (f(N/c-B1)-f(N/c-min(N, B2)))/c)
if B2 < N:
# (D+1) dollars for [B2, N-1]
result += (D+1)*(N-B2)
return result
for case in range(int(input())):
print('Case #%d: %s' % (case+1, coin_change()))