前言
线段树(区间树)是什么呢?有了二叉树、二分搜索树,线段树又是干什么的呢?最经典的线段树问题:区间染色;正如它的名字而言,主要解决区间的问题
一、线段树说明
1、什么是线段树?
线段树首先是二叉树,并且是平衡二叉树(它是一 棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树),并且具有二分性质。
如下图,就是一颗线段树:
假如,用数组表示线段树,如果区间有n个元素,数组表示需要有多少节点?
2、4n节点推导过程
要进行一下,如果对推导过程不感兴趣的,可以直接记住结论,需要4n个节点,推导过程如下图: PS:依旧是全博客园最丑图,当感觉有进步啊!是不是推荐一下,鼓励一下啊
说明:感觉用尽了洪荒之力,才推导出来了。感觉高考之后再也不会用到等比公式了,但又用到了,还是缘分未尽啊,哈哈哈!最后,都放弃了,一直推导不出来,忘却了最后一层的null,假设是满二叉树,按最大值进行估算,所以4n是完全够大的!
二、为什么要使用线段树
线段树主要解决一些区间问题的,如下:
1、区间染色
有一面墙,长度为n,每次选择一段墙进行染色,m次操作之后,我们可以看见多少种颜色?
2、区间查询
查询区间[i,j]的最大值、最小值,或者区间数字和;实质:基于区间的统计查询。
例如:2018年注册用户中消费最高的用户?消费最低的用户?学习最长时间的用户?
三、代码实现
1、创建线段树
二叉树具有天然递归性质,所以用递归相对简单,用迭代也是可以的,我才用递归实现,代码如下:
template<class T>
class SegmentTree {
private:
T *tree;
T *data;
int size;
std::function<T(T, T)> function; int leftChild(int index) { //左孩子下标;例如用数组存储,根节点是下标0,则左孩子为1,右孩子为2
return index * + ;
} int rightChild(int index) { //右孩子下标
return index * + ;
} void buildSegmentTree(int treeIndex, int l, int r) {
if (l == r) {
tree[treeIndex] = data[l];
return;
}
int leftTreeIndex = leftChild(treeIndex);
int rightTreeIndex = rightChild(treeIndex);
int mid = l + (r - l) / ; //中间值求法,防止整型溢出 buildSegmentTree(leftTreeIndex, l, mid); //构建左子树
buildSegmentTree(rightTreeIndex, mid + , r); //构建右子树
tree[treeIndex] = function(tree[leftTreeIndex], tree[rightTreeIndex]);
}
public:
SegmentTree(T arr[], int n, std::function<T(T, T)> function) { //构造函数,构建一棵树
this->function = function;
data = new T[n];
for (int i = ; i < n; ++i) {
data[i] = arr[i];
}
tree = new T[n * ]; //分配4n节点
size = n;
buildSegmentTree(, , size - );
}
};
2、线段树查询
线段树具有二分查找性质,所以二分查找那种思路就可以了,代码如下:
T query(int treeIndex, int l, int r, int queryL, int queryR) {
if (l == queryL && r == queryR) {
return tree[treeIndex];
} int mid = l + (r - l) / ;
int leftTreeIndex = leftChild(treeIndex);
int rightTreeIndex = rightChild(treeIndex); if (queryL >= mid + ) {
return query(rightTreeIndex, mid + , r, queryL, queryR);
} else if (queryR <= mid) {
return query(leftTreeIndex, l, mid, queryL, queryR);
} T leftResult = query(leftTreeIndex, l, mid, queryL, mid);
T rightResult = query(rightTreeIndex, mid + , r, mid + , queryR);
return function(leftResult, rightResult);
} T query(int queryL, int queryR) {
assert(queryL >= && queryL < size && queryR >= && queryR < size && queryL <= queryR);
return query(, , size - , queryL, queryR);
}
3、整体代码
SegmentTree.h如下:
#ifndef SEGMENT_TREE_SEGMENTTREE_H
#define SEGMENT_TREE_SEGMENTTREE_H #include <cassert>
#include <functional> template<class T>
class SegmentTree {
private:
T *tree;
T *data;
int size;
std::function<T(T, T)> function; int leftChild(int index) {
return index * + ;
} int rightChild(int index) {
return index * + ;
} void buildSegmentTree(int treeIndex, int l, int r) {
if (l == r) {
tree[treeIndex] = data[l];
return;
}
int leftTreeIndex = leftChild(treeIndex);
int rightTreeIndex = rightChild(treeIndex);
int mid = l + (r - l) / ; buildSegmentTree(leftTreeIndex, l, mid);
buildSegmentTree(rightTreeIndex, mid + , r);
tree[treeIndex] = function(tree[leftTreeIndex], tree[rightTreeIndex]);
} T query(int treeIndex, int l, int r, int queryL, int queryR) {
if (l == queryL && r == queryR) {
return tree[treeIndex];
} int mid = l + (r - l) / ;
int leftTreeIndex = leftChild(treeIndex);
int rightTreeIndex = rightChild(treeIndex); if (queryL >= mid + ) {
return query(rightTreeIndex, mid + , r, queryL, queryR);
} else if (queryR <= mid) {
return query(leftTreeIndex, l, mid, queryL, queryR);
} T leftResult = query(leftTreeIndex, l, mid, queryL, mid);
T rightResult = query(rightTreeIndex, mid + , r, mid + , queryR);
return function(leftResult, rightResult);
} public:
SegmentTree(T arr[], int n, std::function<T(T, T)> function) {
this->function = function;
data = new T[n];
for (int i = ; i < n; ++i) {
data[i] = arr[i];
}
tree = new T[n * ];
size = n;
buildSegmentTree(, , size - );
} int getSize() {
return size;
} T get(int index) {
assert(index >= && index < size);
return data[index];
} T query(int queryL, int queryR) {
assert(queryL >= && queryL < size && queryR >= && queryR < size && queryL <= queryR);
return query(, , size - , queryL, queryR);
} void print() {
std::cout << "[";
for (int i = ; i < size * ; ++i) {
if (tree[i] != NULL) {
std::cout << tree[i];
} else {
std::cout << "";
}
if (i != size * - ) {
std::cout << ", ";
}
}
std::cout << "]" << std::endl;
}
}; #endif //SEGMENT_TREE_SEGMENTTREE_H
main.cpp如下:
#include <iostream>
#include "SegmentTree.h" int main() {
int nums[] = {-, , , -, , -};
SegmentTree<int> *segmentTree = new SegmentTree<int>(nums, sizeof(nums) / sizeof(int), [](int a, int b) -> int {
return a + b;
});
std::cout << segmentTree->query(,) << std::endl;
segmentTree->print();
return ;
}
4、演示
运行结果,如下:
5、时间复杂度分析
更新 O(logn)
查询 O(logn)
线段树(区间树)之区间染色和4n推导过程的更多相关文章
-
poj-----(2528)Mayor&#39;s posters(线段树区间更新及区间统计+离散化)
Mayor's posters Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 43507 Accepted: 12693 ...
-
POJ 3468 A Simple Problem with Integers(线段树 成段增减+区间求和)
A Simple Problem with Integers [题目链接]A Simple Problem with Integers [题目类型]线段树 成段增减+区间求和 &题解: 线段树 ...
-
BZOJ 3110 ZJOI 2013 K大数查询 树套树(权值线段树套区间线段树)
题目大意:有一些位置.这些位置上能够放若干个数字. 如今有两种操作. 1.在区间l到r上加入一个数字x 2.求出l到r上的第k大的数字是什么 思路:这样的题一看就是树套树,关键是怎么套,怎么写.(话说 ...
-
poj 2892---Tunnel Warfare(线段树单点更新、区间合并)
题目链接 Description During the War of Resistance Against Japan, tunnel warfare was carried out extensiv ...
-
hdu 3974 线段树 将树弄到区间上
Assign the task Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
-
POJ 3468 A Simple Problem with Integers(线段树模板之区间增减更新 区间求和查询)
A Simple Problem with Integers Time Limit: 5000MS Memory Limit: 131072K Total Submissions: 140120 ...
-
约会安排---hdu4553(线段树,麻烦的区间覆盖)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4553 算是poj3667的加强版,建立两颗线段树,一个是DS区间,另一个是NS区间.那么根据题意, ...
-
【BZOJ】1798: [Ahoi2009]Seq 维护序列seq 线段树多标记(区间加+区间乘)
[题意]给定序列,支持区间加和区间乘,查询区间和取模.n<=10^5. [算法]线段树 [题解]线段树多重标记要考虑标记与标记之间的相互影响. 对于sum*b+a,+c直接加上即可. *c后就是 ...
-
HDU 1698 【线段树,区间修改 + 维护区间和】
题目链接 HDU 1698 Problem Description: In the game of DotA, Pudge’s meat hook is actually the most horri ...
随机推荐
-
java跳出多重嵌套循环
public class BreaklFor { public static void main(String args[]){ OK: //设置一个标记 使用带此标记的break语句跳出多重循环体 ...
-
好玩的SQL
1. 做一个3*3的加法表 SQL), (); A||'+'||B||'='||(A+B) ------------------------------------------------------ ...
-
Yii2初谈
Yii2发布有两个月时间了,一直没有去仔细关注过. 今天在回顾PSR标准时,稍稍扫了一眼Yii2.它的命名风格还是一如既往的与Zend那种既首字母大写又还要下划线连接的很二的命名风格格格不入.其实我看 ...
-
[ITSEC]信息安全&#183;Web安全培训第一期客户端安全之UBB系列
缩略图: 引文: 所谓UBB代码,是指论坛中的替代HTML代码的安全代码.ubb发帖编辑器 这种代码使用正则表达式来进行匹配,不同的论坛所使用的UBB代码很可能不同,不能一概而论.UBB代码的出现,使 ...
-
asp.net下ajax.ajaxMethod使用方法
使用AjaxMethod可以在客户端异步调用服务端方法,简单地说就是在JS里调用后台.cs文件里的方法,做一些JS无法做到的操作,如查询数据库. 使用AjaxMethod要满足一下几点: 1.如果 ...
-
kafka知识点
一.为什么需要消息系统 1.解耦: 允许你独立的扩展或修改两边的处理过程,只要确保它们遵守同样的接口约束. 2.冗余: 消息队列把数据进行持久化直到它们已经被完全处理,通过这一方式规避了数据丢失风险. ...
-
vue 中跨组件的表单验证
使用的是element写的,里面提供了表单验证. 子组件是这样的 <template> <div> <el-form :model="value" r ...
-
HDU 3081 Marriage Match II (二分图,并查集)
HDU 3081 Marriage Match II (二分图,并查集) Description Presumably, you all have known the question of stab ...
-
Linux通过端口转发来访问内网服务(端口转发访问阿里云Redis数据库等服务)
# 安装rinetd wget http://www.boutell.com/rinetd/http/rinetd.tar.gz&&tar -xvf rinetd.tar.gz& ...
-
android开发步步为营之67:使用android开源项目android-async-http异步下载文件
android-async-http项目地址 https://github.com/loopj/android-async-http.android-async-http顾名思义是异步的http请求, ...