結果
問題 | No.5015 Escape from Labyrinth |
ユーザー | Yu_212 |
提出日時 | 2023-04-16 18:12:57 |
言語 | Java (openjdk 23) |
結果 |
AC
|
実行時間 | 1,428 ms / 3,000 ms |
コード長 | 26,630 bytes |
コンパイル時間 | 3,476 ms |
コンパイル使用メモリ | 107,512 KB |
実行使用メモリ | 72,036 KB |
スコア | 153,550 |
最終ジャッジ日時 | 2023-04-16 18:15:25 |
合計ジャッジ時間 | 146,961 ms |
ジャッジサーバーID (参考情報) |
judge12 / judge14 |
純コード判定しない問題か言語 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 1,378 ms
69,816 KB |
testcase_01 | AC | 1,361 ms
69,864 KB |
testcase_02 | AC | 1,408 ms
72,036 KB |
testcase_03 | AC | 1,319 ms
69,740 KB |
testcase_04 | AC | 1,299 ms
71,812 KB |
testcase_05 | AC | 1,397 ms
69,984 KB |
testcase_06 | AC | 1,372 ms
69,636 KB |
testcase_07 | AC | 1,354 ms
69,960 KB |
testcase_08 | AC | 1,332 ms
70,144 KB |
testcase_09 | AC | 1,312 ms
69,712 KB |
testcase_10 | AC | 1,390 ms
69,856 KB |
testcase_11 | AC | 1,372 ms
70,264 KB |
testcase_12 | AC | 1,367 ms
69,856 KB |
testcase_13 | AC | 1,328 ms
69,748 KB |
testcase_14 | AC | 1,276 ms
69,732 KB |
testcase_15 | AC | 1,395 ms
70,308 KB |
testcase_16 | AC | 1,377 ms
69,956 KB |
testcase_17 | AC | 1,355 ms
69,884 KB |
testcase_18 | AC | 1,284 ms
70,500 KB |
testcase_19 | AC | 1,286 ms
69,792 KB |
testcase_20 | AC | 1,364 ms
69,848 KB |
testcase_21 | AC | 1,370 ms
70,204 KB |
testcase_22 | AC | 1,369 ms
69,844 KB |
testcase_23 | AC | 1,348 ms
69,808 KB |
testcase_24 | AC | 1,327 ms
69,648 KB |
testcase_25 | AC | 1,379 ms
70,128 KB |
testcase_26 | AC | 1,355 ms
70,116 KB |
testcase_27 | AC | 1,367 ms
69,940 KB |
testcase_28 | AC | 1,360 ms
69,968 KB |
testcase_29 | AC | 1,304 ms
69,804 KB |
testcase_30 | AC | 1,370 ms
69,960 KB |
testcase_31 | AC | 1,353 ms
71,944 KB |
testcase_32 | AC | 1,368 ms
70,064 KB |
testcase_33 | AC | 1,337 ms
69,968 KB |
testcase_34 | AC | 1,292 ms
69,860 KB |
testcase_35 | AC | 1,391 ms
69,660 KB |
testcase_36 | AC | 1,379 ms
70,204 KB |
testcase_37 | AC | 1,386 ms
69,928 KB |
testcase_38 | AC | 1,376 ms
70,120 KB |
testcase_39 | AC | 1,268 ms
69,840 KB |
testcase_40 | AC | 1,416 ms
70,408 KB |
testcase_41 | AC | 1,384 ms
69,968 KB |
testcase_42 | AC | 1,386 ms
67,792 KB |
testcase_43 | AC | 1,352 ms
70,076 KB |
testcase_44 | AC | 1,371 ms
69,904 KB |
testcase_45 | AC | 1,406 ms
70,328 KB |
testcase_46 | AC | 1,413 ms
69,960 KB |
testcase_47 | AC | 1,377 ms
70,252 KB |
testcase_48 | AC | 1,393 ms
70,000 KB |
testcase_49 | AC | 1,343 ms
69,692 KB |
testcase_50 | AC | 1,379 ms
70,124 KB |
testcase_51 | AC | 1,360 ms
69,988 KB |
testcase_52 | AC | 1,373 ms
70,064 KB |
testcase_53 | AC | 1,341 ms
71,752 KB |
testcase_54 | AC | 1,300 ms
69,764 KB |
testcase_55 | AC | 1,376 ms
69,768 KB |
testcase_56 | AC | 1,381 ms
70,388 KB |
testcase_57 | AC | 1,373 ms
69,672 KB |
testcase_58 | AC | 1,354 ms
70,136 KB |
testcase_59 | AC | 1,296 ms
69,916 KB |
testcase_60 | AC | 1,366 ms
69,468 KB |
testcase_61 | AC | 1,281 ms
69,616 KB |
testcase_62 | AC | 1,359 ms
69,596 KB |
testcase_63 | AC | 1,355 ms
70,108 KB |
testcase_64 | AC | 1,323 ms
69,784 KB |
testcase_65 | AC | 1,428 ms
70,488 KB |
testcase_66 | AC | 1,352 ms
69,956 KB |
testcase_67 | AC | 1,353 ms
69,656 KB |
testcase_68 | AC | 1,393 ms
69,808 KB |
testcase_69 | AC | 1,332 ms
69,800 KB |
testcase_70 | AC | 1,412 ms
69,604 KB |
testcase_71 | AC | 1,389 ms
69,884 KB |
testcase_72 | AC | 1,376 ms
68,116 KB |
testcase_73 | AC | 1,299 ms
69,692 KB |
testcase_74 | AC | 1,291 ms
69,468 KB |
testcase_75 | AC | 1,407 ms
70,216 KB |
testcase_76 | AC | 1,418 ms
69,912 KB |
testcase_77 | AC | 1,253 ms
70,216 KB |
testcase_78 | AC | 1,341 ms
69,744 KB |
testcase_79 | AC | 1,285 ms
70,100 KB |
testcase_80 | AC | 1,425 ms
69,932 KB |
testcase_81 | AC | 1,350 ms
69,868 KB |
testcase_82 | AC | 1,345 ms
69,560 KB |
testcase_83 | AC | 1,305 ms
70,164 KB |
testcase_84 | AC | 1,316 ms
69,616 KB |
testcase_85 | AC | 1,389 ms
70,448 KB |
testcase_86 | AC | 1,374 ms
69,704 KB |
testcase_87 | AC | 1,393 ms
69,840 KB |
testcase_88 | AC | 1,368 ms
69,616 KB |
testcase_89 | AC | 1,339 ms
70,460 KB |
testcase_90 | AC | 1,401 ms
69,584 KB |
testcase_91 | AC | 1,367 ms
69,744 KB |
testcase_92 | AC | 1,323 ms
69,904 KB |
testcase_93 | AC | 1,359 ms
70,136 KB |
testcase_94 | AC | 1,345 ms
70,064 KB |
testcase_95 | AC | 1,391 ms
69,856 KB |
testcase_96 | AC | 1,381 ms
70,404 KB |
testcase_97 | AC | 1,396 ms
70,340 KB |
testcase_98 | AC | 1,314 ms
69,884 KB |
testcase_99 | AC | 1,345 ms
69,652 KB |
ソースコード
import java.io.*; import java.util.*; import java.util.concurrent.ThreadLocalRandom; import java.util.function.IntUnaryOperator; import java.util.function.LongUnaryOperator; import java.util.stream.Collectors; public class Main { static boolean isLocal; Random rand = new Random(); In in = new In(System.in); PrintStream out = System.out; long inf = 0x1fffffffffffffffL; int iinf = 0x3fffffff; long startTime; public static final int N = 60; public int D; public static final int H = 1500; public int M; public int J; public char[] grid; public Pos[] ps; public int[] ds; public Pos[] jps; int[][] distances; int[][] beam = new int[60][N * N]; int[] avgWait = new int[N * N]; int[] jid = new int[N * N]; public static final String[] ss = {"M L", "M U", "M R", "M D", "S"}; long solve(int seed) { startTime = System.currentTimeMillis(); in(); distances = new int[J+3][J+3]; for (int i = 0; i < J + 3; i++) { Arrays.fill(distances[i], iinf); } for (int i = 0; i < M; i++) { for (int d = 0; d < 4; d++) { Pos cur = ps[i]; while (true) { cur = cur.moved(d); if (cur == null || grid[cur.z] == '#') { break; } for (int j = 0; j < 60; j += ds[i]) { beam[j][cur.z]++; } } } } for (int i = 0; i < N * N; i++) { int c = 1; for (int j = 0; j < 60; j++) { if (beam[j][i] > 0) { c++; } else { c = 1; } avgWait[i] += c; } } for (int i = 0; i < J + 3; i++) { calcDist(i); } // BitSet visited = new BitSet(J + 3); // visited.set(J); // visited.set(J + 1); // visited.set(J + 2); // List<Integer> routes = new ArrayList<>(); // routes.add(J); // routes.add(J + 1); // routes.add(J + 2); // int dsum = distances[J][J+1] + distances[J+1][J+2]; // int c = 0; // while (elapsed() < 2000) { // int p; // do { // p = rand.next(0, J); // } while (visited.get(p)); // int bi = -1; // int bd = iinf; // for (int i = 1; i < routes.size(); i++) { // int f = routes.get(i - 1); // int t = routes.get(i); // int diff = distances[f][p] + distances[p][t] - distances[f][t]; // if (bd > diff) { // bd = diff; // bi = i; // } // } // if (bi == -1) { // continue; // } //// debug(p, bi, bd, dsum); // if (routes.size() >= 200) { // break; // } //// if (dsum + bd * MUL[D] >= 60 * 500) { ////// c++; ////// if (c >= 100) { ////// break; ////// } else { ////// continue; ////// } //// continue; //// } // visited.set(p); // dsum += bd; // routes.add(bi, p); // } debug("?", elapsed()); State result = annealing(new State(), 180*10, 180, 1000); List<Integer> routes = result.routes; debug("?", elapsed()); Pos p = jps[J]; int tt = 0; int ds1 = 0; int ds2 = 0; totalCost = 0; List<Result> results = new ArrayList<>(); BitSet aa = new BitSet(J); for (int i = 1; i < routes.size(); i++) { Result res = route(jps[routes.get(i - 1)], jps[routes.get(i)], tt); // debug(distances[routes.get(i - 1)][routes.get(i)], res.damage * 60); ds1 += distances[routes.get(i - 1)][routes.get(i)]; ds2 += res.damage; // debug(p, res.from, res.to, res.damage, (tt+res.time)%60); results.add(res); tt = (tt + res.time) % 60; p = res.to; if (i > 1 && aa.get(routes.get(i - 1))) { debug("!!"); throw new RuntimeException(); } aa.set(routes.get(i - 1)); } debug(D, result.dsum, ds1, result.dsum/60, ds1/60, ds2, ds2*60./result.dsum); if (ds2 >= 1500) { throw new RuntimeException(); // debug("!!!!"); // p = jps[J]; // tt = 0; // int dm = 0; // for (Result res : results) { // if (res.from == jps[J + 1]) { // Result v = route(res.from, jps[J + 2], tt); // v.out(p, tt); // break; // } // dm += res.damage; // res.out(p, tt); // tt = (tt + res.time) % 60; // p = res.to; // } } else { p = jps[J]; tt = 0; for (Result res : results) { res.out(p, tt); tt = (tt + res.time) % 60; p = res.to; } } debug(ds2, totalCost, routes.size() * 10); return routes.size() * 10L; // List<Result> routes = new ArrayList<>(); // routes.add(route(jps[J], jps[J + 1], 0)); // routes.add(route(jps[J + 1], jps[J + 2], routes.get(0).time)); // int dsum = routes.get(0).damage + routes.get(1).damage; // while (true) { // int p; // do { // p = rand.next(0, J); // } while (visited.get(p)); // int tt = 0; // int bi = 0; // int bd = iinf; // Result br1 = null, br2 = null; // for (int i = 0; i < routes.size(); i++) { // Result r = routes.get(i); // Result r1 = route(r.from, jps[p], tt); // Result r2 = route(jps[p], r.to, (tt + r.time) % 60); // int diff = r1.damage + r2.damage - r.damage; // if (bd > diff) { // bd = diff; // bi = i; // br1 = r1; // br2 = r2; // } // tt = (tt + r.time) % 60; // } // debug(p, bi, bd, dsum); // if (dsum + bd >= 700) { // break; // } // visited.set(p); // dsum += bd; // routes.set(bi, br1); // routes.add(bi + 1, br2); // } // Pos p = jps[J]; // int tt = 0; // for (Result res : routes) { // debug(p, res.from, res.to); // res.out(p, tt); // tt = (tt + res.time) % 60; // p = res.to; // } // visited.set(J + 2); // visited.set(J); // tt = moveTo(J, J+1, tt, visited); // visited.clear(J + 2); // tt = moveTo(J+1, J+2, tt, visited); } //1678 int totalCost = 0; // int moveTo(int s, int g, int tt, BitSet visited) { // int cur = s; // int ds = 0; // while (cur != g) { // int md = iinf; // int mi = -1; // for (int i = 0; i < J + 3; i++) { // if (visited.get(i)) { // continue; // } //// debug(i, cur, jps[i].dist(jps[g]), distances[cur * (J + 3) + i]); // int d = distances[cur * (J + 3) + i] + jps[i].dist(jps[g]) * 50; // if (md > d) { // md = d; // mi = i; // } // } // ds += distances[cur * (J + 3) + mi]; // visited.set(mi); // tt = moveStraight(cur, mi, tt); // cur = mi; // } // debug(ds, totalCost); // return tt; // } public static final double[] MUL = {0, 0, 0, 1.25, 1.33, 1.4, 1.6, 1.8}; class State { BitSet visited = new BitSet(J + 3); List<Integer> routes = new ArrayList<>(); int dsum; double score; boolean hasScore; State prevCopy; State() { visited.set(J); visited.set(J + 1); visited.set(J + 2); routes.add(J); routes.add(J + 1); routes.add(J + 2); dsum = distances[J][J+1] + distances[J+1][J+2]; } State(State prev) { this.visited = (BitSet)prev.visited.clone(); this.routes = new ArrayList<>(prev.routes); this.dsum = prev.dsum; this.score = prev.score; this.hasScore = true; } int real() { int damage = 0; int tt = 0; for (int i = 1; i < routes.size(); i++) { Result res = route(jps[routes.get(i - 1)], jps[routes.get(i)], tt); // debug(distances[routes.get(i - 1)][routes.get(i)], res.damage * 60); damage += res.damage; // debug(p, res.from, res.to, res.damage); tt = (tt + res.time) % 60; } return damage; } State rollback() { return prevCopy; } int neighbor0(double stage) { while (true) { int p; do { p = rand.next(0, J); } while (visited.get(p)); int bi = -1; int bd = iinf; for (int i = 1; i < routes.size(); i++) { int f = routes.get(i - 1); int t = routes.get(i); int diff = distances[f][p] + distances[p][t] - distances[f][t]; if (bd > diff) { bd = diff; bi = i; } } if (bi == -1) { continue; } visited.set(p); dsum += bd; routes.add(bi, p); score = calcScore(0); break; } return 0; } int neighbor1(double stage) { int bd = -iinf; int bp = 0; for (int p = 1; p + 1 < routes.size(); p++) { int f = routes.get(p - 1); int t = routes.get(p + 1); int r = routes.get(p); if (r >= J) { continue; } int diff = distances[f][r] + distances[r][t] - distances[f][t]; if (bd < diff) { bd = diff; bp = p; } } visited.clear(routes.get(bp)); dsum -= bd; routes.remove(bp); score = calcScore(0); return 1; } int neighbor(double stage, double rs) { hasScore = false; prevCopy = copy(); if (routes.size() >= 60 && rand.next(3) == 0 || dsum * MUL[D] >= 60 * 1000 * rs) { // debug(dsum * MUL[D], getScore(stage), routes.size(), 36000, 1); return neighbor1(stage); } else { // debug(dsum * MUL[D], getScore(stage), routes.size(), 36000, 0); return neighbor0(stage); } } State copy() { return new State(this); } double getScore(double stage) { if (!hasScore) { score = calcScore(stage); hasScore = true; } return score; } double calcScore(double stage) { double score = (routes.size() - 3) * 180*5 - dsum * MUL[D]; return score; } long realScore() { int score = 0; return score; } } State annealing(State state, double startTemp, double endTemp, long timeLimit) { State bestState = state.copy(); double bestScore = state.getScore(0); int itr = 0; int rb = 0; int c0 = 0; double rs = 1; boolean calc = false; while (true) { long nowTime = System.currentTimeMillis() - startTime; if (nowTime >= timeLimit) { break; } double stage = (double)nowTime / timeLimit; if (stage > 0.95 && !calc) { calc = true; rs = 1300. / state.real(); debug("??", rs); } itr++; double prevScore = state.getScore(stage); double temp = startTemp + (endTemp - startTemp) * nowTime / timeLimit; double thres = prevScore + rand.nextLog() * temp; int nid = state.neighbor(stage, rs); if (nid == -1) { continue; } if (nid == 0) { c0++; } if (bestScore < state.getScore(stage)) { bestScore = state.getScore(stage); bestState = state.copy(); } if (state.getScore(stage) <= thres) { state = state.rollback(); rb++; } } debug(itr, rb, c0); return bestState; } class Result { List<Integer> route; int damage; int time; Pos from; Pos to; public Result(List<Integer> route, int damage, Pos from, Pos to) { this.route = route; this.damage = damage; this.time = route.size() % 60; this.from = from; this.to = to; } public void out(Pos p, int tt) { for (int v : route) { if (v < 4) p = p.moved(v); tt = (tt + 1) % 60; totalCost += D * beam[tt][p.z]; totalCost++; out.println(ss[v]); } } } Result route(Pos s, Pos g, int tt) { class S { private final Pos p; private final int t; private final int d; public S(Pos p, int t, int d) { this.p = p; this.t = t; this.d = d; } } PriorityQueue<S> queue = new PriorityQueue<>(Comparator.comparingInt(v -> v.d)); queue.add(new S(s, tt, 0)); int[][] dist = new int[60][N * N]; int[][] prev = new int[60][N * N]; for (int i = 0; i < 60; i++) { Arrays.fill(dist[i], iinf); Arrays.fill(prev[i], -1); } dist[tt][s.z] = 0; while (!queue.isEmpty()) { S v = queue.remove(); if (dist[v.t][v.p.z] < v.d) { continue; } if (v.p.z == g.z) { List<Integer> route = new ArrayList<>(); Pos cur = v.p; int ct = v.t; int damage = dist[ct][cur.z]; while (prev[ct][cur.z] != -1) { route.add(prev[ct][cur.z]); if (prev[ct][cur.z] < 4) { cur = cur.moved((prev[ct][cur.z] + 2) % 4); } ct = (ct + 59) % 60; } Collections.reverse(route); return new Result(route, damage, s, g); } int nt = (v.t + 1) % 60; for (int d = 0; d < 4; d++) { Pos q = v.p.moved(d); if (q == null || grid[q.z] == '#') { continue; } int cost = dist[v.t][v.p.z] + 1 + beam[nt][q.z] * D; if (dist[nt][q.z] > cost) { dist[nt][q.z] = cost; prev[nt][q.z] = d; queue.add(new S(q, nt, cost)); } } int cost = dist[v.t][v.p.z] + 1 + beam[nt][v.p.z] * D; if (dist[nt][v.p.z] > cost) { dist[nt][v.p.z] = cost; prev[nt][v.p.z] = 4; queue.add(new S(v.p, nt, cost)); } } throw new RuntimeException(); } void calcDist(int s) { class S { private final Pos p; private final int d; public S(Pos p, int d) { this.p = p; this.d = d; } } PriorityQueue<S> queue = new PriorityQueue<>(Comparator.comparingInt(v -> v.d)); queue.add(new S(jps[s], 0)); int[] dist = new int[N * N]; Arrays.fill(dist, iinf); dist[jps[s].z] = 0; while (!queue.isEmpty()) { S v = queue.remove(); if (dist[v.p.z] < v.d) { continue; } if (jid[v.p.z] != -1) { distances[s][jid[v.p.z]] = v.d; } for (int d = 0; d < 4; d++) { Pos q = v.p.moved(d); if (q == null || grid[q.z] == '#') { continue; } int cost = dist[v.p.z] + avgWait[q.z]; if (dist[q.z] > cost) { dist[q.z] = cost; queue.add(new S(q, cost)); } } } } void in() { in.nextInt(); D = in.nextInt(); in.nextInt(); grid = new char[N * N]; int jc = 0; for (int i = 0; i < N; i++) { char[] s = in.nextCharArray(); for (int j = 0; j < N; j++) { grid[i * N + j] = s[j]; if (grid[i * N + j] == 'J') { jc++; } } } M = in.nextInt(); J = jc; jps = new Pos[J + 3]; ps = new Pos[M]; ds = new int[M]; for (int i = 0; i < M; i++) { int y = in.nextInt(); int x = in.nextInt(); int d = in.nextInt(); ps[i] = pos(x, y); ds[i] = d; } jc = 0; Arrays.fill(jid, -1); for (int i = 0; i < N * N; i++) { if (grid[i] == 'S') { jps[J] = pos(i); jid[i] = J; grid[i] = '.'; } else if (grid[i] == 'K') { jps[J + 1] = pos(i); jid[i] = J + 1; grid[i] = '.'; } else if (grid[i] == 'G') { jps[J + 2] = pos(i); jid[i] = J + 2; grid[i] = '.'; } else if (grid[i] == 'J') { jid[i] = jc; jps[jc++] = pos(i); } else if (grid[i] == 'E') { grid[i] = '#'; } } } static Pos pos(int x, int y) { return Pos.instances[y * N + x]; } static Pos pos(int z) { return Pos.instances[z]; } static class Timing { Pos p; int t, z; public Timing(Pos p, int t) { this.p = p; this.t = t; this.z = p.z * 60 + t; } } static class Pos implements Comparable<Pos> { static Pos[] instances; static Pos[] moved; static { instances = new Pos[N * N]; moved = new Pos[4 * N * N]; for (int i = 0; i < N * N; i++) { instances[i] = new Pos(i % N, i / N, i); } for (int i = 0; i < N * N; i++) { for (int d = 0; d < 4; d++) { // LURD int nx = i % N + (d - 1) % 2; int ny = i / N + (d - 2) % 2; if (0 <= nx && nx < N && 0 <= ny && ny < N) { moved[i * 4 + d] = instances[ny * N + nx]; } } } } final int x, y, z; private Pos(int x, int y, int z) { this.x = x; this.y = y; this.z = z; } Pos moved(int d) { return moved[z * 4 + d]; } int dist(Pos o) { return Math.abs(x - o.x) + Math.abs(y - o.y); } Pos[] neighbors() { Pos[] p = new Pos[4]; int i = 0; if ((p[i] = moved(0)) != null) i++; if ((p[i] = moved(1)) != null) i++; if ((p[i] = moved(2)) != null) i++; if ((p[i] = moved(3)) != null) i++; return i == 4 ? p : Arrays.copyOf(p, i); } @Override public int compareTo(Pos o) { return Integer.compare(z, o.z); } @Override public boolean equals(Object o) { return o instanceof Pos && z == ((Pos)o).z; } @Override public int hashCode() { return z; } @Override public String toString() { return String.format("(%d, %d)", x, y); } } static void debug(Object... args) { if (!isLocal) { return; } if (args == null || args.getClass() != Object[].class) { args = new Object[] {args}; } System.err.println(Arrays.stream(args).map(obj -> { Class<?> clazz = obj == null ? null : obj.getClass(); return clazz == byte[].class ? Arrays.toString((byte[])obj) : clazz == short[].class ? Arrays.toString((short[])obj) : clazz == int[].class ? Arrays.toString((int[])obj) : clazz == long[].class ? Arrays.toString((long[])obj) : clazz == char[].class ? Arrays.toString((char[])obj) : clazz == float[].class ? Arrays.toString((float[])obj) : clazz == double[].class ? Arrays.toString((double[])obj) : clazz == boolean[].class ? Arrays.toString((boolean[])obj) : obj instanceof Object[] ? Arrays.deepToString((Object[])obj) : String.valueOf(obj); }).collect(Collectors.joining(" "))); } long elapsed() { return (long)((System.currentTimeMillis() - startTime) * (isLocal ? 1.7 : 1)); } boolean elapsed(long timeLimit) { return elapsed() >= timeLimit; } public static void main(String[] args) { new Main().solve(0); } static class Random { private long a, b, c, d; Random() { this(ThreadLocalRandom.current().nextLong()); } Random(long seed) { this.a = seed; for (int i = 0; i < 20; i++) { rand(); } } private int rand() { long t = a ^ (a << 11); a = b; b = c; c = d; d = d ^ (d >>> 19) ^ t ^ (t >>> 8); return (int)(a & 0x7fffffff); } long next() { return (long)rand() << 31 | rand(); } int next(int n) { return rand() % n; } int next(int l, int r) { return rand() % (r - l) + l; } double nextDouble() { return rand() / 2147483648.0; } double nextLog() { return Math.log(rand() + 1) - 21.487562597358306; } } static class In { private final BufferedReader reader; private StringTokenizer tokenizer; In(InputStream input) { this.reader = new BufferedReader(new InputStreamReader(input), 0x10000); } String next() { try { while (tokenizer == null || !tokenizer.hasMoreTokens()) { tokenizer = new StringTokenizer(reader.readLine()); } } catch (IOException ignored) { } return tokenizer.nextToken(); } int nextInt() { return Integer.parseInt(next()); } long nextLong() { return Long.parseLong(next()); } double nextDouble() { return Double.parseDouble(next()); } char[] nextCharArray() { return next().toCharArray(); } String[] nextStringArray(int n) { String[] s = new String[n]; for (int i = 0; i < n; i++) { s[i] = next(); } return s; } char[][] nextCharGrid(int n, int m) { char[][] a = new char[n][m]; for (int i = 0; i < n; i++) { a[i] = next().toCharArray(); } return a; } int[] nextIntArray(int n) { int[] a = new int[n]; for (int i = 0; i < n; i++) { a[i] = nextInt(); } return a; } int[] nextIntArray(int n, IntUnaryOperator op) { int[] a = new int[n]; for (int i = 0; i < n; i++) { a[i] = op.applyAsInt(nextInt()); } return a; } int[][] nextIntMatrix(int h, int w) { int[][] a = new int[h][w]; for (int i = 0; i < h; i++) { a[i] = nextIntArray(w); } return a; } long[] nextLongArray(int n) { long[] a = new long[n]; for (int i = 0; i < n; i++) { a[i] = nextLong(); } return a; } long[] nextLongArray(int n, LongUnaryOperator op) { long[] a = new long[n]; for (int i = 0; i < n; i++) { a[i] = op.applyAsLong(nextLong()); } return a; } long[][] nextLongMatrix(int h, int w) { long[][] a = new long[h][w]; for (int i = 0; i < h; i++) { a[i] = nextLongArray(w); } return a; } List<List<Integer>> nextEdges(int n, int m, boolean directed) { List<List<Integer>> res = new ArrayList<>(); for (int i = 0; i < n; i++) { res.add(new ArrayList<>()); } for (int i = 0; i < m; i++) { int u = nextInt() - 1; int v = nextInt() - 1; res.get(u).add(v); if (!directed) { res.get(v).add(u); } } return res; } } }