http://hihocoder.com/problemset/problem/1337

#1337 : 平衡树·SBT

时间限制:10000ms
单点时限:1000ms
内存限制:256MB

描述

小Ho:小Hi,之前你不是讲过Splay和Treap么,那么还有没有更简单的平衡树呢?

小Hi:但是Splay和Treap不是已经很简单了么?

小Ho:是这样没错啦,但是Splay和Treap和原来的二叉搜索树相比都有很大的改动,我有点记不住。

小Hi:这样啊,那我不妨再给你讲解一个新的平衡树算法好了。和二叉搜索树相比,它只需要修改insert函数,就可以做到高度的平衡。

小Ho:好,我就喜欢这样的!

提示:Size Balanced Tree

输入

第1行:1个正整数n,表示操作数量,10≤n≤100,000

第2..n+1行:每行1个字母c和1个整数k:

若c为'I',表示插入一个数字k到树中,-1,000,000,000≤k≤1,000,000,000

若c为'Q',表示询问树中第k小数字,保证1≤k≤树的节点数量

输出

若干行:每行1个整数,表示针对询问的回答,保证一定有合法的解

样例输入
5
I 3
I 2
Q 1
I 5
Q 2
样例输出
2
3

---恢复内容结束---

动态查询Ktop系列

1.对于固定的Ktop系列,可以使用 优先队列,最小堆,Treap,BST,SBT

2.动态的Ktop Treap,BST,SBT 效率: BST<Treap<SBT

解法一 使用二叉搜索树: 此方法是直接建立起二叉树,对于树不做调整,这会造成树变得很长!

遇到的坑: Java swap不能像C++那样,C++可以传地址,传值,传应用但是Java并不是,Java只能传值,并且传递参数的时候,使用的是深copy,也就是参数的对象和本尊不是同一个对象地址,而仅仅是和它拥有相同数值的不同对象。所以swap不能像C++那样.

 import java.util.Scanner;

 /**
* author: 龚细军
* class-aim:
*/ class Node {
public Integer key;
public long size;
public Node left;
public Node right; public Node() {
size = 0;
key = null;
left = right = null;
}
} /*二叉排序树,此题不需要调解平衡*/
class BSTree {
private static final int DEFAULT_INITIAL_CAPACITY = 1; public static int query(Node node, int kMin) {
Long flag = node.left.size - kMin + 1;
if (flag == 0) return node.key;
if (flag < 0) return query(node.right, (int) abs(flag));
return query(node.left, kMin);
} private static long abs(Long flag) {
return flag > 0 ? flag : -1 * flag;
} public static void insert(Node node, int data) {
if (node.size > 0) {
node.size++;
insert(data > node.key ? node.right : node.left, data);
} else {
node.key = data;
node.size = DEFAULT_INITIAL_CAPACITY;
node.left = new Node();
node.right = new Node();
}
} } public class Main { public static void main(String args[]) {
int num, val;
String cmd;
Scanner scanner = new Scanner(System.in);
while (scanner.hasNext()) {
num = scanner.nextInt();
Node root = new Node();
while (num-- > 0) {
cmd = scanner.next();
val = scanner.nextInt();
if (cmd.equals("I")) BSTree.insert(root, val);
else {
System.out.println(BSTree.query(root, val));
}
}
}
} }

解法二: SBT树 平衡树,在解法一基础上进行优化,也就每次对其不满足这样条件的进行调整:

node.left.size >= max(node.right.right.size,node.right.left.size);

node.right.size >= max(node.left.right.size , node.left.left.size);

进行平衡调整.这样就可以使其查询数据降到O(lgn)

 import java.util.Scanner;

 /**
* author: 龚细军
* class-aim:
*/ class Node {
public int key, size;
public Node left, right; public Node() {
this.size = 0;
this.left = null;
this.right = null;
} public void clone(Node node) {
this.key = node.key;
this.size = node.size;
this.left = node.left;
this.right = node.right;
}
} public class Main {
private static final int DEFAULT_INITIAL_CAPACITY = 1; public static int getSize(Node node) {
return node == null ? 0 : node.size;
} public static int compare(Node a, int key) {
return a.key - key;
} public static void update(Node node) {
if (node == null) return;
node.size = getSize(node.left) + getSize(node.right) + 1;
} public static void rightRotate(Node master, Node node) {
Node newNode = new Node();
newNode.clone(master);
newNode.left = node.right;
node.right = newNode;
update(newNode);
update(node);
master.clone(node);
} public static void leftRotate(Node master, Node node) {
Node tmpNode = new Node();
tmpNode.clone(master);
tmpNode.right = node.left;
node.left = tmpNode;
update(tmpNode);
update(node);
master.clone(node);
} public static void insert(Node master, int key) { if (master.size == 0) {
master.left = new Node();
master.right = new Node();
master.size = DEFAULT_INITIAL_CAPACITY;
master.key = key;
} else if (compare(master, key) > 0) {
insert(master.left, key);
if (getSize(master.left.left) > getSize(master.right)) {
//右旋转
rightRotate(master, master.left);
}
} else {
insert(master.right, key);
if (getSize(master.right.right) > getSize(master.left)) {
//左旋转
leftRotate(master, master.right);
}
} update(master); } private static int abs(int flag) {
return flag > 0 ? flag : -1 * flag;
} public static int query(Node node, int kMin) { int flag = node.left.size - kMin + 1;
if (flag == 0) return node.key;
if (flag < 0) return query(node.right, abs(flag));
return query(node.left, kMin);
} public static void main(String args[]) {
int num, val;
String cmd;
Scanner scanner = new Scanner(System.in);
while (scanner.hasNext()) {
num = scanner.nextInt();
Node root = new Node();
while (num-- > 0) {
cmd = scanner.next();
val = scanner.nextInt();
if (cmd.equals("I"))
Main.insert(root, val);
else
System.out.println(Main.query(root, val));
}
}
}
}

hihocoder-平衡树·SBT的更多相关文章

  1. tyvj 普通平衡树 SBT or splay

    普通平衡树 From admin     背景 Background 此为平衡树系列第一道:普通平衡树     描述 Description 您需要写一种数据结构(可参考题目标题),来维护一些数,其中 ...

  2. 【bzoj3435】[Wc2014]紫荆花之恋 替罪点分树套SBT

    题目描述 强强和萌萌是一对好朋友.有一天他们在外面闲逛,突然看到前方有一棵紫荆树.这已经是紫荆花飞舞的季节了,无数的花瓣以肉眼可见的速度从紫荆树上长了出来.仔细看看的话,这个大树实际上是一个带权树.每 ...

  3. Hihocoder 1337 (splay)

    Problem 平衡树 SBT 题目大意 维护一个序列,支持两种操作. 操作一:插入一个数. 操作二:询问第k小的数. 解题分析 ~~刷刷水题,再熟悉一下splay的基本操作. ps:哇咔咔,有连续四 ...

  4. 三大平衡树(Treap + Splay + SBT)总结+模板[转]

    Treap树 核心是 利用随机数的二叉排序树的各种操作复杂度平均为O(lgn) Treap模板: #include <cstdio> #include <cstring> #i ...

  5. 子树大小平衡树(Size Balanced Tree,SBT)操作模板及杂谈

    基础知识(包括但不限于:二叉查找树是啥,SBT又是啥反正又不能吃,平衡树怎么旋转,等等)在这里就不(lan)予(de)赘(duo)述(xie)了. 先贴代码(数组模拟): int seed; int ...

  6. HDU 4006 The kth great number 优先队列、平衡树模板题(SBT)

    The kth great number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65768/65768 K (Java/Oth ...

  7. 平衡树初阶——AVL平衡二叉查找树+三大平衡树(Treap + Splay + SBT)模板【超详解】

    平衡树初阶——AVL平衡二叉查找树 一.什么是二叉树 1. 什么是树. 计算机科学里面的树本质是一个树状图.树首先是一个有向无环图,由根节点指向子结点.但是不严格的说,我们也研究无向树.所谓无向树就是 ...

  8. Hihocoder 1325 平衡树·Treap(平衡树,Treap)

    Hihocoder 1325 平衡树·Treap(平衡树,Treap) Description 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说的是哪两个啊? 小Ho:就是二叉 ...

  9. HihoCoder 1325 平衡树·Treap

    HihoCoder 1325 平衡树·Treap 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说 ...

随机推荐

  1. dev GridView常用属性,事件

    一.属性 1.GridControl属性 //允许拖拽行 gridControl1.AllowDrop = true; 2.GridView属性 //不可编辑 gridView1.OptionsBeh ...

  2. wireshark常用过滤条件

    抓取指定IP地址的数据流: 如果你的抓包环境下有很多主机正在通讯,可以考虑使用所观察主机的IP地址来进行过滤.以下为IP地址抓包过滤示例: host 10.3.1.1:抓取发到/来自10.3.1.1的 ...

  3. Android下添加新的自定义键值和按键处理流程

            Android下添加新的自定义键值和按键处理流程     说出来不怕大家笑话,我写这篇博客的原因在于前几天去一个小公司面试Android系统工程师,然后在面试的时候对方的技术总监问了我 ...

  4. phpMyAdmin的配置

    好久没写东西了,上来记录一下今天学的一点小东西吧~ 之前搞php开发的时候,一直用的是SQLyog来操作mysql数据库的,但是今天发现sqlyog功能不是很完善,主要是我想修改数据库名,但是sqly ...

  5. Hibernate的关联映射

    单向N-1关联 <many-to-one> 单向N-1关系,比如多个人对应同一个住址,只需要从人实体端找到对应的住址实体,无须关系某个地址的全部住户.程序在N的一端增加一个属性,该属性引用 ...

  6. python 学习笔记十六 django深入学习一 路由系统,模板,admin,数据库操作

    django 请求流程图 django 路由系统 在django中我们可以通过定义urls,让不同的url路由到不同的处理函数 from . import views urlpatterns = [ ...

  7. 【jQuery】【转】jQuery中的trigger和triggerHandler区别

    trigger(event, [data]) 在每一个匹配的元素上触发某类事件. 这个函数也会导致浏览器同名的默认行为的执行.比如,如果用trigger()触发一个'submit',则同样会导致浏览器 ...

  8. Element is not currently interactable and may not be manipulated

    Element is not currently interactable and may not be manipulated:元素当前不可交互,并且可能无法操作. 解决方法: 调用该方法,智能等待 ...

  9. word域1

    Description: 域是WORD中的一种特殊命令,它由花括号.域名(域代码)及选项开关构成.域代码类似于公式,域选项并关是特殊指令,在域中可触发特定的操作.在用WORD处理文档时若能巧妙应用域, ...

  10. inline-boock的默认间距

    1 2 3 4 div{width:900px;} div li{ display:inline-block; width:300px;} <ul>     <li></ ...