hihocoder-平衡树·SBT
http://hihocoder.com/problemset/problem/1337
#1337 : 平衡树·SBT
描述
小Ho:小Hi,之前你不是讲过Splay和Treap么,那么还有没有更简单的平衡树呢?
小Hi:但是Splay和Treap不是已经很简单了么?
小Ho:是这样没错啦,但是Splay和Treap和原来的二叉搜索树相比都有很大的改动,我有点记不住。
小Hi:这样啊,那我不妨再给你讲解一个新的平衡树算法好了。和二叉搜索树相比,它只需要修改insert函数,就可以做到高度的平衡。
小Ho:好,我就喜欢这样的!
输入
第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的更多相关文章
- tyvj 普通平衡树 SBT or splay
普通平衡树 From admin 背景 Background 此为平衡树系列第一道:普通平衡树 描述 Description 您需要写一种数据结构(可参考题目标题),来维护一些数,其中 ...
- 【bzoj3435】[Wc2014]紫荆花之恋 替罪点分树套SBT
题目描述 强强和萌萌是一对好朋友.有一天他们在外面闲逛,突然看到前方有一棵紫荆树.这已经是紫荆花飞舞的季节了,无数的花瓣以肉眼可见的速度从紫荆树上长了出来.仔细看看的话,这个大树实际上是一个带权树.每 ...
- Hihocoder 1337 (splay)
Problem 平衡树 SBT 题目大意 维护一个序列,支持两种操作. 操作一:插入一个数. 操作二:询问第k小的数. 解题分析 ~~刷刷水题,再熟悉一下splay的基本操作. ps:哇咔咔,有连续四 ...
- 三大平衡树(Treap + Splay + SBT)总结+模板[转]
Treap树 核心是 利用随机数的二叉排序树的各种操作复杂度平均为O(lgn) Treap模板: #include <cstdio> #include <cstring> #i ...
- 子树大小平衡树(Size Balanced Tree,SBT)操作模板及杂谈
基础知识(包括但不限于:二叉查找树是啥,SBT又是啥反正又不能吃,平衡树怎么旋转,等等)在这里就不(lan)予(de)赘(duo)述(xie)了. 先贴代码(数组模拟): int seed; int ...
- 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 ...
- 平衡树初阶——AVL平衡二叉查找树+三大平衡树(Treap + Splay + SBT)模板【超详解】
平衡树初阶——AVL平衡二叉查找树 一.什么是二叉树 1. 什么是树. 计算机科学里面的树本质是一个树状图.树首先是一个有向无环图,由根节点指向子结点.但是不严格的说,我们也研究无向树.所谓无向树就是 ...
- Hihocoder 1325 平衡树·Treap(平衡树,Treap)
Hihocoder 1325 平衡树·Treap(平衡树,Treap) Description 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说的是哪两个啊? 小Ho:就是二叉 ...
- HihoCoder 1325 平衡树·Treap
HihoCoder 1325 平衡树·Treap 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho:小Hi,我发现我们以前讲过的两个数据结构特别相似. 小Hi:你说 ...
随机推荐
- PL/SQL Developer连接Oracle 11g在Win8 64位系统下乱码
PL/SQL Developer在64位系统上连接Oracle,需要安装32位Oracle client客户端,使用后,发现操作数据库出现乱码的情况.经过查找资料,解决此问题,方法如下: 需要在系统的 ...
- js错误
1ajax提交 从数据库里取文件typeerror: 'stepup' called on an object that does not implement interface HTMLInputE ...
- ftp协议详解
客户端与服务器之间,需要多条连接才能完成应用的协议,属于复杂协议.如FTP,PPTP,H.323和SIP均属于复杂协议. 这里主要介绍ftp协议的工作原理.首先,ftp通信协议有两种工作模式,被动模式 ...
- 解决对含有第三方jar包的项目打包出现java.lang.NoClassDefFoundError问题
用eclipse普通的打包方式,对含有第三方jar包的项目进行打包.调用方法后一只出现java.lang.NoClassDefFoundError问题. 从网上搜寻,很多都是在MANIFEST.MF文 ...
- SourceTree 免登录跳过初始设置 - 转
转自:http://www.cnblogs.com/xiofee/p/sourcetree_pass_initialization_setup.html 在SourceTree的配置目录新建(或修改) ...
- winform 移动窗体,和窗体阴影(引用)
无边框窗体移动://窗体移动API [DllImport("user32.dll")] public static extern bool ReleaseCapture(); [D ...
- grep命令
grep (缩写来自Globally search a Regular Expression and Print)是一种强大的文本搜索工具,它能使用正则表达式搜索文本,并把匹配的行打印出来. grep ...
- Creating Dialogs
#ifndef DIALOG_H #define DIALOG_H #include <QtWidgets> //#include <QDialog> //#include & ...
- T-SQL编程 —— 用户自定义函数(标量函数)
用户自定义函数 在使用SQL server的时候,除了其内置的函数之外,还允许用户根据需要自己定义函数.根据用户定义函数返回值的类型,可以将用户定义的函数分为三个类别: 返回值为可更新表的函数 如果用 ...
- js小知识归纳(待续)
1.javascript 中 if (window != top) top.location.href = location.href;的意思 如果当前窗口不是最上层窗口(比如是在Iframe中),那 ...