結果
問題 | No.649 ここでちょっとQK! |
ユーザー | tamura2004 |
提出日時 | 2022-03-27 08:46:54 |
言語 | Crystal (1.14.0) |
結果 |
AC
|
実行時間 | 324 ms / 3,000 ms |
コード長 | 6,976 bytes |
コンパイル時間 | 15,570 ms |
コンパイル使用メモリ | 295,860 KB |
実行使用メモリ | 24,764 KB |
最終ジャッジ日時 | 2024-11-06 03:52:10 |
合計ジャッジ時間 | 21,308 ms |
ジャッジサーバーID (参考情報) |
judge1 / judge3 |
(要ログイン)
テストケース
テストケース表示入力 | 結果 | 実行時間 実行使用メモリ |
---|---|---|
testcase_00 | AC | 2 ms
6,820 KB |
testcase_01 | AC | 2 ms
6,816 KB |
testcase_02 | AC | 2 ms
6,820 KB |
testcase_03 | AC | 306 ms
6,816 KB |
testcase_04 | AC | 176 ms
24,728 KB |
testcase_05 | AC | 170 ms
24,764 KB |
testcase_06 | AC | 178 ms
24,752 KB |
testcase_07 | AC | 2 ms
6,816 KB |
testcase_08 | AC | 2 ms
6,820 KB |
testcase_09 | AC | 2 ms
6,820 KB |
testcase_10 | AC | 2 ms
6,816 KB |
testcase_11 | AC | 2 ms
6,820 KB |
testcase_12 | AC | 136 ms
12,544 KB |
testcase_13 | AC | 127 ms
12,544 KB |
testcase_14 | AC | 134 ms
12,672 KB |
testcase_15 | AC | 135 ms
12,672 KB |
testcase_16 | AC | 136 ms
12,544 KB |
testcase_17 | AC | 154 ms
12,672 KB |
testcase_18 | AC | 170 ms
15,972 KB |
testcase_19 | AC | 194 ms
15,956 KB |
testcase_20 | AC | 204 ms
15,952 KB |
testcase_21 | AC | 228 ms
15,872 KB |
testcase_22 | AC | 243 ms
16,260 KB |
testcase_23 | AC | 258 ms
19,524 KB |
testcase_24 | AC | 277 ms
20,284 KB |
testcase_25 | AC | 299 ms
20,372 KB |
testcase_26 | AC | 324 ms
20,276 KB |
testcase_27 | AC | 3 ms
6,820 KB |
testcase_28 | AC | 3 ms
6,820 KB |
testcase_29 | AC | 3 ms
6,816 KB |
testcase_30 | AC | 127 ms
12,800 KB |
testcase_31 | AC | 130 ms
12,544 KB |
testcase_32 | AC | 2 ms
6,816 KB |
testcase_33 | AC | 2 ms
6,816 KB |
testcase_34 | AC | 2 ms
6,816 KB |
testcase_35 | AC | 2 ms
6,816 KB |
ソースコード
# require "crystal/avl_tree" # AVL木によるOrderedSet class AVLTree(T) getter root : Node(T)? def initialize(@root : Node(T)? = nil) end def insert(v : T) @root = root.try &.insert(v) || Node(T).new(v) end def <<(v : T) insert(v) end def delete(v : T) @root = root.try &.delete(v) end # v以下(未満)の最大値 # # ``` # tr = [1, 3, 5, 7].to_ordered_set # tr.lower(4) # => 3 # tr.lower(5) # => 5 # tr.lower(5, eq: false) # => 3 # ``` def lower(v, eq = true) @root.try &.lower(v, eq) end def lower_index(v, eq = true) @root.try &.lower_index(v, eq) end # v以下(未満)の要素数 def lower_count(v, eq = true) @root.try(&.lower_index(v, eq)).try(&.succ) || 0 end # v以上(超える)の最小値 def upper(v, eq = true) @root.try &.upper(v, eq) end def upper_index(v, eq = true) @root.try &.upper_index(v, eq) end # v以上(を超える)要素数 def upper_count(v, eq = true) @root.try(&.upper_index(v, eq)).try{ |v| (@root.try(&.size) || 0) - v} || 0 end # 最小の値を持つノード def min_node @root.try &.min_node end # 最小の値 def min @root.try &.min end # 最大の値を持つノード def max_node @root.try &.max_node end # 最大の値 def max @root.try &.max end def pop @root, node = root.try &.pop || {nil.as(Node(T)?), nil.as(Node(T)?)} node end # 小さいほうからk番目のノードの値(0-origin) def at(k) @root.try &.at(k) end def [](k) at(k) end def size @root.try &.size || 0 end def clear @root = nil end def inspect @root.inspect end def debug @root.try &.debug(0) end class Node(T) getter val : T getter balance : Int32 getter height : Int32 getter size : Int32 property left : Node(T)? property right : Node(T)? def initialize(@val) @balance = 0 @size = 1 @height = 1 @left = nil @right = nil end # 挿入 def insert(v) # return self if v == val if v <= val @left = left.try &.insert(v) || Node(T).new(v) else @right = right.try &.insert(v) || Node(T).new(v) end update re_balance end # 削除 def delete(v) if v == val left.try do |l| @left, node = l.pop node.left, node.right = left, right node.update.re_balance end || right.try do |r| right end || nil elsif v < val @left = left.try &.delete(v) update.re_balance else @right = right.try &.delete(v) update.re_balance end end # v以下(未満)の最大値 def lower(v, eq = true) if val < v || (eq && v == val) right.try(&.lower(v, eq)).try { |v| Math.max(v, val) } || val else left.try &.lower(v, eq) end end # v以下(未満)の最大のインデックス def lower_index(v, eq = true) pos = left_size if val < v || (eq && v == val) right.try(&.lower_index(v, eq)).try(&.+(pos + 1)) || pos else left.try(&.lower_index(v, eq)) end end # v以上(超える)の最小値 def upper(v, eq = true) if v < val || (eq && v == val) left.try(&.upper(v, eq)).try { |v| Math.min(v, val) } || val else right.try &.upper(v, eq) end end # v以上(超える)の最小のインデックス def upper_index(v, eq = true) pos = left_size if val > v || (eq && v == val) left.try(&.upper_index(v, eq)) || pos else right.try(&.upper_index(v, eq)).try(&.+(pos + 1)) end end # 最大値を削除して返す def pop right.try do |r| @right, node = r.pop {update, node} end || {left, self} end # 最小のノード def min_node left.try &.min_node || self end # 最小の値 def min min_node.try &.val end # 最大のノード def max_node right.try &.max_node || self end # 最大の値 def max max_node.try &.val end # 小さいほうからk番目のノードの値(0-origin) def at(k) ord = left.try &.size || 0 if k == ord val elsif k < ord left.try &.at(k) else right.try &.at(k - ord - 1) end end # ノードの状態を更新 def update @height = Math.max(left_height, right_height) + 1 @balance = left_height - right_height @size = left_size + right_size + 1 self end # 回転によりバランスを保つ def re_balance case balance when 2.. if left_balance < 0 @left = left.try &.rotate_left.update end rotate_right.update when ..-2 if right_balance > 0 @right = right.try &.rotate_right.update end rotate_left.update else self end end # 左回転 def rotate_left right.try do |root| root.tap do @right, root.left = root.left, self root.update update end end || self end # 右回転 def rotate_right left.try do |root| root.tap do @left, root.right = root.right, self root.update update end end || self end @[AlwaysInline] def [](k) at(k) end @[AlwaysInline] private def left_size left.try &.size || 0 end @[AlwaysInline] private def right_size right.try &.size || 0 end @[AlwaysInline] private def left_height left.try &.height || 0 end @[AlwaysInline] private def right_height right.try &.height || 0 end @[AlwaysInline] private def left_balance left.try &.balance || 0 end @[AlwaysInline] private def right_balance right.try &.balance || 0 end def inspect # "(#{val} #{left.inspect} #{right.inspect})".gsub(/nil/, ".") "(#{left.inspect} #{val} #{right.inspect})".gsub(/nil/, ".") # "(#{left.inspect} #{[val, size, height, balance]} #{right.inspect})".gsub(/nil/, ".") end def debug(indent = 0) left.try &.debug(indent + 2) puts " " * indent + "#{val}[#{size},#{height},#{balance}]" right.try &.debug(indent + 2) end end end module Indexable(T) def to_ordered_set AVLTree(T).new.tap do |tr| each do |v| tr << v end end end end alias OrderedSet = AVLTree q, k = gets.to_s.split.map(&.to_i64) st = OrderedSet(Int64).new q.times do line = gets.to_s case line when /^1/ _, val = line.split.map(&.to_i64) st << val when /^2/ if v = st.at(k-1) puts v st.delete(v) else puts -1 end end end