結果
問題 | No.649 ここでちょっとQK! |
ユーザー | titan23 |
提出日時 | 2022-10-20 01:18:55 |
言語 | PyPy3 (7.3.15) |
結果 |
WA
|
実行時間 | - |
コード長 | 8,461 bytes |
コンパイル時間 | 249 ms |
コンパイル使用メモリ | 82,304 KB |
実行使用メモリ | 130,184 KB |
最終ジャッジ日時 | 2024-06-30 00:42:29 |
合計ジャッジ時間 | 13,435 ms |
ジャッジサーバーID (参考情報) |
judge2 / judge1 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | WA | - |
testcase_01 | AC | 43 ms
54,528 KB |
testcase_02 | WA | - |
testcase_03 | AC | 160 ms
96,996 KB |
testcase_04 | AC | 1,084 ms
128,736 KB |
testcase_05 | AC | 1,004 ms
130,184 KB |
testcase_06 | WA | - |
testcase_07 | AC | 41 ms
54,784 KB |
testcase_08 | AC | 41 ms
54,528 KB |
testcase_09 | AC | 41 ms
54,400 KB |
testcase_10 | WA | - |
testcase_11 | WA | - |
testcase_12 | AC | 369 ms
96,296 KB |
testcase_13 | AC | 340 ms
95,488 KB |
testcase_14 | AC | 337 ms
95,616 KB |
testcase_15 | AC | 340 ms
95,532 KB |
testcase_16 | AC | 317 ms
95,936 KB |
testcase_17 | AC | 380 ms
98,176 KB |
testcase_18 | AC | 427 ms
99,968 KB |
testcase_19 | AC | 420 ms
102,528 KB |
testcase_20 | AC | 472 ms
103,552 KB |
testcase_21 | AC | 466 ms
104,976 KB |
testcase_22 | AC | 546 ms
108,908 KB |
testcase_23 | AC | 501 ms
108,556 KB |
testcase_24 | AC | 579 ms
111,924 KB |
testcase_25 | AC | 599 ms
112,556 KB |
testcase_26 | AC | 654 ms
115,036 KB |
testcase_27 | WA | - |
testcase_28 | WA | - |
testcase_29 | WA | - |
testcase_30 | WA | - |
testcase_31 | WA | - |
testcase_32 | WA | - |
testcase_33 | WA | - |
testcase_34 | AC | 41 ms
54,656 KB |
testcase_35 | AC | 40 ms
54,272 KB |
ソースコード
import math alpha = 0.75 beta = math.log2(1/alpha) class ScapegoatTreeNode: # __slots__ = ('key', 'left', 'right', 'size') def __init__(self, key): self.key = key self.left = None self.right = None self.size = 1 def __str__(self): if self.left is None and self.right is None: return f'key:{self.key, self.size}\n' return f'key:{self.key, self.size},\n left:{self.left},\n right:{self.right}\n' class ScapegoatTreeSet: # __slots__ = ('node', '__iter', '__len') '''Make a new AVLTree. / O(NlogN)''' def __init__(self, a=[]) -> None: if a: aa = sorted(a) a = [aa[0]] for i in range(1, len(aa)): if aa[i] != a[-1]: a.append(aa[i]) self.node = self.__build(a) def __build(self, a) -> None: def sort(l, r): if l >= r: return None, 0 mid = (l + r) >> 1 root = ScapegoatTreeNode(a[mid]) root.left, sl = sort(l, mid) root.right, sr = sort(mid+1, r) root.size = sl + sr + 1 return root, sl+sr+1 return sort(0, len(a))[0] def __rebuild(self, node: ScapegoatTreeNode) -> ScapegoatTreeNode: a = [] def get(node): if node.left is not None: get(node.left) a.append(node) if node.right is not None: get(node.right) def sort(l, r): if l >= r: return None, 0 mid = (l + r) >> 1 root = a[mid] root.left, sl = sort(l, mid) root.right, sr = sort(mid+1, r) root.size = sl + sr + 1 return root, sl+sr+1 get(node) return sort(0, len(a))[0] '''add a key. / O(logN)''' def add(self, key) -> bool: if self.node is None: self.node = ScapegoatTreeNode(key) return True node = self.node path, di = [], 0 while node is not None: if key < node.key: path.append(node) di = di<<1 | 1 node = node.left elif key > node.key: path.append(node) di = di<<1 node = node.right else: return False if di & 1: path[-1].left = ScapegoatTreeNode(key) else: path[-1].right = ScapegoatTreeNode(key) if len(path)*beta > math.log(self.__len__()): node_size = 1 while path: pnode = path.pop() di >>= 1 pnode_size = pnode.size + 1 if alpha * pnode_size < node_size: break node_size = pnode_size new_node = self.__rebuild(pnode) if not path: self.node = new_node return True if di & 1: path[-1].left = new_node else: path[-1].right = new_node for p in path: p.size += 1 return True '''Discard key. / O(logN)''' def discard(self, key) -> bool: di = 1 node = self.node path = [] while node is not None: if key < node.key: path.append(node) di = 1 node = node.left elif key > node.key: path.append(node) di = -1 node = node.right else: break else: return False if node.left is not None and node.right is not None: path.append(node) di = 1 lmax = node.left while lmax.right is not None: path.append(lmax) di = -1 lmax = lmax.right node.key = lmax.key node = lmax cnode = node.right if node.left is None else node.left if path: if di == 1: path[-1].left = cnode else: path[-1].right = cnode else: self.node = cnode for p in path: p.size -= 1 return True '''Find the largest element <= key, or None if it doesn't exist. / O(logN)''' def le(self, key): res, node = None, self.node while node is not None: if key < node.key: node = node.left elif key > node.key: res, node = node.key, node.right else: res = key break return res '''Find the largest element < key, or None if it doesn't exist. / O(logN)''' def lt(self, key): res, node = None, self.node while node is not None: if key < node.key: node = node.left elif key > node.key: res, node = node.key, node.right else: break return res '''Find the smallest element >= key, or None if it doesn't exist. / O(logN)''' def ge(self, key): res, node = None, self.node while node is not None: if key < node.key: res, node = node.key, node.left elif key > node.key: node = node.right else: res = key break return res '''Find the smallest element > key, or None if it doesn't exist. / O(logN)''' def gt(self, key): res, node = None, self.node while node is not None: if key < node.key: res, node = node.key, node.left elif key > node.key: node = node.right else: break return res '''Count the number of elements < key. / O(logN)''' def index(self, key) -> int: indx = 0 node = self.node while node: if key < node.key: node = node.left elif key > node.key: indx += 1 if node.left is None else node.left.size + 1 node = node.right else: indx += 0 if node.left is None else node.left.size break return indx '''Count the number of elements <= key. / O(logN)''' def index_right(self, key) -> int: indx = 0 node = self.node while node: if key < node.key: node = node.left elif key > node.key: indx += 1 if node.left is None else node.left.size + 1 node = node.right else: indx += 1 if node.left is None else node.left.size + 1 break return indx '''Return and Remove max element or a[p]. / O(logN)''' def pop(self, p=-1): if p < 0: p += self.__len__() now = 0 node = self.node path = [] di = 1 while node is not None: t = now if node.left is None else now + node.left.size if t < p: path.append(node) now = t + 1 di = -1 node = node.right elif t > p: path.append((node)) di = 1 node = node.left else: break else: raise IndexError(f'k={p}, len={self.__len__()}') res = node.key if node.left is not None and node.right is not None: path.append(node) di = 1 lmax = node.left while lmax.right is not None: path.append(lmax) di = -1 lmax = lmax.right node.key = lmax.key node = lmax cnode = node.right if node.left is None else node.left if path: if di == 1: path[-1].left = cnode else: path[-1].right = cnode else: self.node = cnode for p in path: p.size -= 1 return res '''Return and Remove min element. / O(logN)''' def popleft(self): return self.pop(0) def __kth_elm(self, k): if k < 0: k += self.__len__() now, node = 0, self.node while node is not None: t = now if node.left is None else now + node.left.size if t < k: now, node = t + 1, node.right elif t > k: node = node.left else: return node.key raise IndexError(f'k={k}, len={self.__len__()}') def __contains__(self, x): node = self.node while node: if x < node.key: node = node.left elif x > node.key: node = node.right else: return True return False def __getitem__(self, x): return self.__kth_elm(x) def __iter__(self): self.__iter = 0 return self def __next__(self): if self.__iter == self.__len__(): raise StopIteration res = self.__getitem__(self.__iter) self.__iter += 1 return res def __reversed__(self): for i in range(self.__len__()): yield self.__getitem__(-i-1) def __len__(self): return 0 if self.node is None else self.node.size def __bool__(self): return self.node is not None def __repr__(self): return '{' + ', '.join(map(str, self)) + '}' def __str__(self): return '{' + ', '.join(map(str, self)) + '}' # ----------------------- # import sys input = lambda: sys.stdin.readline().rstrip() q, k = map(int, input().split()) avl = ScapegoatTreeSet() query = [list(map(int, input().split())) for _ in range(q)] for qu in query: if qu[0] == 1: avl.add(qu[1]) else: if len(avl) < k: print(-1) else: print(avl.pop(k-1))