Created
May 30, 2012 19:26
-
-
Save frp/2838428 to your computer and use it in GitHub Desktop.
Treap with rmq
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| class Treap | |
| class Node | |
| attr_accessor :key, :priority, :left, :right, :count, :minimum | |
| def initialize(key, priority) | |
| @key = key | |
| @priority = priority | |
| @count = 1 | |
| @minimum = key | |
| end | |
| def self.count_of(node) | |
| return node ? node.count : 0 | |
| end | |
| def update_count | |
| @count = Node.count_of(@left) + Node.count_of(@right) + 1 | |
| @minimum = @key | |
| if @left | |
| @minimum = [@minimum, @left.minimum].min | |
| end | |
| if @right | |
| @minimum = [@minimum, @right.minimum].min | |
| end | |
| end | |
| def kth(k) | |
| if k == Node.count_of(@left) | |
| return @key | |
| elsif k < Node.count_of(@left) | |
| return @left.kth(k) | |
| else | |
| return @right.kth(k - Node.count_of(@left) - 1) | |
| end | |
| end | |
| def self.merge(node1, node2) | |
| if not node1 or not node2 | |
| return node1 ? node1 : node2 | |
| elsif node1.priority > node2.priority | |
| node1.right = merge(node1.right, node2) | |
| node1.update_count | |
| return node1 | |
| else | |
| node2.left = merge(node1, node2.left) | |
| node2.update_count | |
| return node2 | |
| end | |
| end | |
| def self.split(node, count) | |
| if not node | |
| return [nil, nil] | |
| elsif Node.count_of(node.left) + 1 <= count | |
| spliting_right_result = split(node.right, | |
| count - Node.count_of(node.left) - 1 ) | |
| node.right = spliting_right_result[0] | |
| node.update_count | |
| return [node, spliting_right_result[1]] | |
| else | |
| spliting_left_result = split(node.left, count) | |
| node.left = spliting_left_result[1] | |
| node.update_count | |
| return [spliting_left_result[0], node] | |
| end | |
| end | |
| def self.traverse(node) | |
| if not node | |
| return [] | |
| end | |
| result = [] | |
| if node.left | |
| result += traverse(node.left) | |
| end | |
| result += [node.key] | |
| if node.right | |
| result += traverse(node.right) | |
| end | |
| return result | |
| end | |
| end | |
| attr_accessor :root | |
| def initialize | |
| @root = nil | |
| end | |
| def insert(key, position) | |
| left, right = Node.split(@root, position) | |
| new_node = Node.new(key, rand(2000000000)) | |
| @root = Node.merge(left, Node.merge(new_node, right)) | |
| end | |
| def remove(position) | |
| left_result_part, rest = Node.split(@root, position - 1) | |
| elements_to_remove, right_result_part = Node.split(rest, position) | |
| @root = Node.merge(left_result_part, right_result_part) | |
| end | |
| def traverse | |
| return Node.traverse(@root) | |
| end | |
| def kth(k) | |
| return @root.kth(k) | |
| end | |
| def rmq(left, right) | |
| part1, part2 = Node.split(@root, left) | |
| part2, part3 = Node.split(part2, right) | |
| minimum = part2.minimum | |
| @root = merge(part1, merge(part2, part3)) | |
| return minimum | |
| end | |
| end |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment