結果
問題 | No.5019 Hakai Project |
ユーザー | Kiri8128 |
提出日時 | 2023-11-19 03:41:37 |
言語 | PyPy3 (7.3.15) |
結果 |
AC
|
実行時間 | 1,015 ms / 3,000 ms |
コード長 | 11,103 bytes |
コンパイル時間 | 278 ms |
コンパイル使用メモリ | 81,828 KB |
実行使用メモリ | 142,796 KB |
スコア | 2,710,400,537 |
最終ジャッジ日時 | 2023-11-19 03:42:27 |
合計ジャッジ時間 | 47,829 ms |
ジャッジサーバーID (参考情報) |
judge11 / judge15 |
純コード判定しない問題か言語 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 774 ms
108,336 KB |
testcase_01 | AC | 858 ms
118,124 KB |
testcase_02 | AC | 831 ms
113,048 KB |
testcase_03 | AC | 714 ms
107,572 KB |
testcase_04 | AC | 832 ms
116,720 KB |
testcase_05 | AC | 848 ms
118,512 KB |
testcase_06 | AC | 735 ms
105,328 KB |
testcase_07 | AC | 997 ms
139,708 KB |
testcase_08 | AC | 850 ms
119,264 KB |
testcase_09 | AC | 898 ms
125,288 KB |
testcase_10 | AC | 757 ms
105,200 KB |
testcase_11 | AC | 812 ms
115,868 KB |
testcase_12 | AC | 886 ms
128,016 KB |
testcase_13 | AC | 818 ms
118,820 KB |
testcase_14 | AC | 716 ms
104,372 KB |
testcase_15 | AC | 754 ms
108,212 KB |
testcase_16 | AC | 827 ms
111,068 KB |
testcase_17 | AC | 754 ms
98,464 KB |
testcase_18 | AC | 843 ms
122,820 KB |
testcase_19 | AC | 829 ms
110,928 KB |
testcase_20 | AC | 925 ms
123,444 KB |
testcase_21 | AC | 878 ms
115,076 KB |
testcase_22 | AC | 758 ms
103,308 KB |
testcase_23 | AC | 874 ms
115,968 KB |
testcase_24 | AC | 1,015 ms
142,796 KB |
testcase_25 | AC | 943 ms
129,324 KB |
testcase_26 | AC | 924 ms
129,472 KB |
testcase_27 | AC | 795 ms
111,428 KB |
testcase_28 | AC | 790 ms
107,700 KB |
testcase_29 | AC | 771 ms
100,524 KB |
testcase_30 | AC | 714 ms
107,948 KB |
testcase_31 | AC | 851 ms
120,888 KB |
testcase_32 | AC | 772 ms
109,180 KB |
testcase_33 | AC | 829 ms
115,232 KB |
testcase_34 | AC | 921 ms
126,992 KB |
testcase_35 | AC | 719 ms
103,192 KB |
testcase_36 | AC | 824 ms
111,884 KB |
testcase_37 | AC | 758 ms
109,004 KB |
testcase_38 | AC | 707 ms
106,744 KB |
testcase_39 | AC | 783 ms
103,600 KB |
testcase_40 | AC | 860 ms
119,472 KB |
testcase_41 | AC | 737 ms
106,592 KB |
testcase_42 | AC | 814 ms
112,040 KB |
testcase_43 | AC | 747 ms
101,688 KB |
testcase_44 | AC | 788 ms
109,828 KB |
testcase_45 | AC | 732 ms
105,896 KB |
testcase_46 | AC | 809 ms
105,844 KB |
testcase_47 | AC | 810 ms
115,984 KB |
testcase_48 | AC | 873 ms
122,480 KB |
testcase_49 | AC | 880 ms
126,952 KB |
ソースコード
def move(target_i, target_j): global cost global ANS global current_i, current_j b = sum(B) while (current_i, current_j) != (target_i, target_j): if current_i < target_i: ANS.append((1, "D")) current_i += 1 elif current_i > target_i: ANS.append((1, "U")) current_i -= 1 elif current_j < target_j: ANS.append((1, "R")) current_j += 1 elif current_j > target_j: ANS.append((1, "L")) current_j -= 1 cost += (1 if A[current_i][current_j] == "." else 2) * (b + 1) ** 2 def buy(k): global cost assert A[current_i][current_j] == "@" cost += price[k] B[k] += 1 ANS.append((2, k + 1)) if DEBUG: print("buy", k, "@", (current_i, current_j)) def bomb_check(k, ii, jj): s = 0 t = 0 for i, j in Z[k][1]: ni = ii + i nj = jj + j if 0 <= ni < N and 0 <= nj < N and A[ni][nj] != ".": if A[ni][nj] == "@": s += 1 else: t += 1 def bomb(k): assert B[k] > 0 global shop_list B[k] -= 1 ANS.append((3, k + 1)) for i, j in Z[k][1]: ni = current_i + i nj = current_j + j if 0 <= ni < N and 0 <= nj < N and A[ni][nj] != ".": if A[ni][nj] == "@": pass # shop_list.destroyed A[ni][nj] = "X" if DEBUG: print("bomb", k, "@", (current_i, current_j)) def answer(): print(len(ANS)) for a, b in ANS: print(a, b) if DEBUG: print("cost =", cost) writetext_output(ANS) def disp(): if DEBUG: print("") print("-" * 20) print("price =", price) print("inv =", price_inv) print("power =", power) building_cnt = sum(a.count("#") for a in A) shop_cnt = sum(a.count("@") for a in A) print("building cnt =", building_cnt) print("shop cnt =", shop_cnt) print("shop_list =", len(shop_list), shop_list) print("cost =", cost) print("A =") for a in A: print("".join(a)) print("-" * 20) print("") try: LOCAL except NameError: LOCAL = 0 if LOCAL: DEBUG = 1 # N, M, A, Z = random_testcase() N, M, A, Z = preset_testcase() else: DEBUG = 0 N, M = map(int, input().split()) A = [] for _ in range(N): A.append([a for a in input()]) Z = [] for _ in range(M): C, L = map(int, input().split()) pos = [] for _ in range(L): a, b = map(int, input().split()) pos.append((a, b)) Z.append((C, pos)) AA = [a[:] for a in A] cover_count_stay = [[-1 if aa == "." else 0 for aa in a] for a in A] if DEBUG: writetext_input(N, M, A, Z) price = [] price_inv = [] power = [] position = [] position_matrix = [] B = [0] * M cost = 0 current_i = 0 current_j = 0 ANS = [] if DEBUG: print("N =", N) print("M =", M) shop_list = [] for i, a in enumerate(A): for j, aa in enumerate(a): if aa == "@": shop_list.append((i, j)) shop_list_destroyed = [0] * len(shop_list) disp() for C, pos in Z: if DEBUG: print("C, cnt =", C, len(pos), "{:.4f}".format(len(pos) / C)) print("pos =", pos) price.append(C) price_inv.append(10 ** 6 // C) power.append(len(pos)) position.append(pos) if 0: print("") print("C =", C) print("cnt =", len(pos)) z = [[0] * 41 for _ in range(41)] for i, j in pos: z[i][j] = 1 position_matrix.append(z) if 0: print("pos =") for zz in z: print("".join(zz)) def best_shop(i, j): best_i = -1 best_j = -1 best = 1000 for ti, tj in shop_list: d = abs(ti - i) + abs(tj - j) if 0: d += (max(10 - ti, 0) + max(ti - 39, 0) + max(10 - tj, 0) + max(tj - 39, 0)) * 5 if 21 <= ti < 29: d += 15 if 21 <= tj < 29: d += 15 if d < best: best = d best_i = ti best_j = tj return best_i, best_j def calc_favorite_shops(): L = [] # for ti, tj in ((9, 9), (9, 40), (40, 40), (40, 9)): for ti, tj in ((9, 9), (9, 24), (9, 40), (24, 40), (40, 40), (40, 24), (40, 9)): best = 10 ** 10 best_k = -1 for k, (i, j) in enumerate(shop_list): d = (ti - i) ** 2 + (tj - j) ** 2 if d < best: best = d best_k = k if best_k not in L: L.append(best_k) return L def calc_weight(shops): W = [[1.0] * N for _ in range(N)] for i in range(N): for j in range(N): for k in shops: si, sj = shop_list[k] d = max(abs(i - si), abs(j - sj)) if d <= 20: for ss in (0.2, 0.6, 1.0): w = 1 / (1 + d * ss) W[i][j] *= (1 - w) for i in range(N): for j in range(N): W[i][j] = int(10 ** 4 * W[i][j]) + 1 return W def find_best_cand(): best = 0 best_cand = None for b, i, j, v in candidates_stay: k = favorite_shops[v] ev = 0 si, sj = shop_list[k] for bi, bj in position[b]: ni = si + bi nj = sj + bj if 0 <= ni < N and 0 <= nj < N: # if AA[ni][nj] != ".": if cover_count_stay[ni][nj] == 0: ev += weight[ni][nj] ev *= price_inv[b] if ev > best: best = ev best_cand = (b, i, j, v) return best_cand favorite_shops = calc_favorite_shops() if DEBUG: print("position =") for b, _pos in enumerate(position): print(b, _pos) print("favorite_shops =", favorite_shops) weight = calc_weight(favorite_shops) if DEBUG: print("weight =") for w in weight: print(w) favorite_shops_flg = [[-1] * N for _ in range(N)] for v, k in enumerate(favorite_shops): si, sj = shop_list[k] favorite_shops_flg[si][sj] = v candidates_stay = [] for v, k in enumerate(favorite_shops): si, sj = shop_list[k] for b in range(M): for bi, bj in position[b]: ni = si + bi nj = sj + bj if (bi or bj) and 0 <= ni < N and 0 <= nj < N and favorite_shops_flg[ni][nj] > v: break else: candidates_stay.append((b, si, sj, v)) strategy_stay = [[] for _ in range(len(favorite_shops))] for _ in range(20): b, i, j, v = find_best_cand() k = favorite_shops[v] strategy_stay[v].append((b, i, j)) si, sj = shop_list[k] for bi, bj in position[b]: ni = si + bi nj = sj + bj if 0 <= ni < N and 0 <= nj < N: # if AA[ni][nj] != ".": # AA[ni][nj] = "." if cover_count_stay[ni][nj] >= 0: cover_count_stay[ni][nj] += 1 if DEBUG: print("candidates_stay =", candidates_stay) print("strategy_stay =") for v in range(len(strategy_stay)): print(v, strategy_stay[v]) cover_count_away = [[0 if aa == 0 else -1 for aa in a] for a in cover_count_stay] candidates_away = [] candidates_efficiency = [] candidates_matrix = [[[] for _ in range(N)] for _ in range(N)] for i in range(N): for j in range(N): for b in range(M): mav = -1 for v, k in enumerate(favorite_shops): si, sj = shop_list[k] di, dj = si - i, sj - j if -20 <= di <= 20 and -20 <= dj <= 20 and position_matrix[b][di][dj]: mav = max(mav, v) best = 10 ** 9 best_v = None for v in range(mav + 1, len(favorite_shops)): k = favorite_shops[v] si, sj = shop_list[k] d = abs(i - si) + abs(j - sj) if d <= best: best = d best_v = v if best_v is None: continue v = best_v k = favorite_shops[v] si, sj = shop_list[k] if v < len(favorite_shops): u = len(candidates_away) candidates_away.append((b, i, j, v)) benefit = 0 for bi, bj in position[b]: ni, nj = i + bi, j + bj if 0 <= ni < N and 0 <= nj < N: if cover_count_away[ni][nj] == 0: candidates_matrix[ni][nj].append(u) benefit += weight[ni][nj] moving_cost = (abs(i - si) + abs(j - sj)) * 5 efficiency = 10 ** 6 // (price[b] + moving_cost) * benefit candidates_efficiency.append(efficiency) if DEBUG: print("CHECKPOINT") def find_best_cand_away(): best = 0 best_cand = None for efficiency, cand in zip(candidates_efficiency, candidates_away): if efficiency > best: best = efficiency best_cand = cand if DEBUG: print("BEST CAND AWAY =", (b, i, j, v), "Efficiency =", best) return best_cand def update_candidates_away(b, i, j, v): for bi, bj in position[b]: ni, nj = i + bi, j + bj if 0 <= ni < N and 0 <= nj < N and cover_count_away[ni][nj] >= 0: cover_count_away[ni][nj] += 1 if cover_count_away[ni][nj] == 1: for u in candidates_matrix[ni][nj]: _b, _i, _j, _v = candidates_away[u] k = favorite_shops[_v] si, sj = shop_list[k] moving_cost = (abs(_i - si) + abs(_j - sj)) * 5 candidates_efficiency[u] -= 10 ** 6 // (price[_b] + moving_cost) * weight[ni][nj] strategy_away = [[] for _ in range(len(favorite_shops))] while 1: cand = find_best_cand_away() if cand is None: break b, i, j, v = cand if DEBUG: print("BEST CAND AWAY =", (b, i, j, v)) update_candidates_away(b, i, j, v) strategy_away[v].append((b, i, j)) if DEBUG: print("strategy_away") for v, s in enumerate(strategy_away): print(v, s) for v, k in enumerate(favorite_shops): si, sj = shop_list[k] for b, i, j in strategy_away[v]: move(si, sj) buy(b) move(i, j) bomb(b) move(si, sj) for b, i, j in strategy_stay[v]: move(i, j) buy(b) for b, i, j in strategy_stay[v]: bomb(b) while 0: #shop_list: move(*best_shop(current_i, current_j)) ni = current_i + 1 if current_i < N - 1 else current_i - 1 nj = current_j + 1 if current_j < N - 1 else current_j - 1 for k in range(5): buy(k) # buy(k) # buy(k) for k in range(5): bomb(k) if 0: move(ni, current_j) for k in range(M): bomb(k) move(ni, nj) for k in range(M): bomb(k) answer() disp()