結果
問題 | No.2310 [Cherry 5th Tune A] Against Regret |
ユーザー | navel_tos |
提出日時 | 2023-05-20 19:07:04 |
言語 | PyPy3 (7.3.15) |
結果 |
AC
|
実行時間 | 3,530 ms / 6,000 ms |
コード長 | 3,047 bytes |
コンパイル時間 | 625 ms |
コンパイル使用メモリ | 82,176 KB |
実行使用メモリ | 89,932 KB |
最終ジャッジ日時 | 2024-12-21 11:42:16 |
合計ジャッジ時間 | 48,250 ms |
ジャッジサーバーID (参考情報) |
judge5 / judge1 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 43 ms
51,840 KB |
testcase_01 | AC | 46 ms
52,096 KB |
testcase_02 | AC | 44 ms
52,224 KB |
testcase_03 | AC | 98 ms
75,136 KB |
testcase_04 | AC | 81 ms
68,736 KB |
testcase_05 | AC | 81 ms
70,784 KB |
testcase_06 | AC | 98 ms
75,136 KB |
testcase_07 | AC | 84 ms
69,888 KB |
testcase_08 | AC | 400 ms
79,204 KB |
testcase_09 | AC | 653 ms
79,624 KB |
testcase_10 | AC | 746 ms
80,356 KB |
testcase_11 | AC | 990 ms
80,068 KB |
testcase_12 | AC | 1,026 ms
81,348 KB |
testcase_13 | AC | 515 ms
78,988 KB |
testcase_14 | AC | 651 ms
79,104 KB |
testcase_15 | AC | 1,094 ms
80,768 KB |
testcase_16 | AC | 224 ms
77,568 KB |
testcase_17 | AC | 646 ms
79,928 KB |
testcase_18 | AC | 3,435 ms
89,812 KB |
testcase_19 | AC | 3,188 ms
89,548 KB |
testcase_20 | AC | 3,175 ms
89,500 KB |
testcase_21 | AC | 3,161 ms
89,932 KB |
testcase_22 | AC | 3,157 ms
89,676 KB |
testcase_23 | AC | 3,187 ms
89,672 KB |
testcase_24 | AC | 3,196 ms
89,292 KB |
testcase_25 | AC | 3,185 ms
89,540 KB |
testcase_26 | AC | 3,187 ms
89,416 KB |
testcase_27 | AC | 3,198 ms
89,784 KB |
testcase_28 | AC | 3,530 ms
89,428 KB |
testcase_29 | AC | 702 ms
79,744 KB |
ソースコード
#yukicoder389F Against Regret ''' なんだか解けそう。 頂点iから頂点jへの移動経路として考えられるものの総数を前計算する。 DAGであることを利用すれば、ゴール側から順に計算してゆくことでO(N^3)に収まる。 事前に追加した辺集合を「下道での移動」と呼称する。 各クエリで追加する辺集合を「高速道路での移動」と呼称しよう。 頂点0から頂点Nへのパスとして考えられるものは ・下道だけで0→Nに移動する ・下道で0→i、高速でi→j、下道でj→N ・下道で0→i、高速でi→j、下道でj→k、高速にまた乗ってk→L、下道でL→N ・高速に一時的に乗って、下道に降りて1区間以上進んで、・・・ となる。 なので、高速道路の各地点において ・今高速道路を降りて、下道で次の高速道路の地点かゴールに向かう。 ・今高速道路に乗る。 とする場合の数を管理すればよい。 あとは 下道だけで0→Nの移動をする場合の数と足せばよい。 ''' import sys; input=sys.stdin.readline f=lambda:list(map(int,input().split())) N=int(input()); X=[f() for _ in range(N+1)]; MOD=998244353 #local[i][j]: iからjに移動を行う場合の数 local=[[0]*(N+1) for _ in range(N+1)] for i in range(N,-1,-1): for j in range(i+1,N+1): for k in range(i+1,j): #kを経由する場合 local[i][j]+=X[i][k]*local[k][j]; local[i][j]%=MOD local[i][j]+=X[i][j]; local[i][j]%=MOD #直接iからjに移動する場合の数 #クエリに回答。高速道路の頂点名は0~Nではなく、座圧したもので呼ぶので注意 for _ in range(int(input())): K=int(input()); Task=[f() for _ in range(K)] R=sorted(set([Task[x][y] for x in range(K) for y in [0,1]])) D={j:i for i,j in enumerate(R)} #G[x][y]: 座圧した頂点xからyに移動する、追加した有向辺の本数 G=[[0]*len(D) for _ in range(len(R))] for a,b,c in Task: G[D[a]][D[b]]+=c #highway[x][t]: 座圧した頂点xに対して、辺状態がtの辺を最後に使ってxに到達する場合の数 #t=0: 1回以上追加した有向辺を使ったうえで、既存の辺を最後に使ってxに到達 #t=1: 追加した有向辺を最後に使ってxに到達 highway=[[0]*2 for _ in range(len(R))] #初期化するが、頂点0と頂点Nだけは特別扱いする。 ans=local[0][N]; local[0][0]=local[N][N]=1 for x,i in enumerate(R): #1. 一度高速を降りて別の頂点に向かうケース for y in range(x+1,len(R)): highway[y][0]+=highway[x][1]*local[i][R[y]]%MOD; highway[y][0]%=MOD ans+=highway[x][1]*local[i][N]%MOD; ans%=MOD #2. ここから高速に乗って別の頂点に遷移するケース for y in range(x+1,len(R)): highway[y][1]+=(local[0][i]+highway[x][0]+highway[x][1])*G[x][y]%MOD highway[y][1]%=MOD print(ans%MOD)