数据结构-PHP 线段树的实现
转:
数据结构-PHP 线段树的实现
1.线段树介绍
线段树是基于区间的统计查询,线段树是一种 二叉搜索树
,它将一个区间划分成一些单元区间,每个单元区间对应线段树中的一个叶结点。使用线段树可以快速的查找某一个节点在若干条线段中出现的次数,时间复杂度为O(logN)
,线段树是一颗 平衡二叉树
。
2.线段树示意图
如下图所示,数组 E
中,假设区间 0-9
一共 10 个元素,每个儿子节点区间元素的个数都是父亲节点元素个数的一半,若出现 奇数
的情况,则右儿子
元素区间比 左儿子
元素区间多一个:
Tips:如图所示的中节点中区间指的是数组
E
的索引值。
3.线段树需要空间分析
假设我们把 线段树
看做是一颗 满二叉树
,并且不考虑添加元素的情况(即区间固定),对于区间有 n
个元素的数组若 n=2^k(k是正整数)
则需要 2n
的空间,最差的情况是若 n=2^k+1
则需要 4n
的空间,如下图所示,最下面一层没有元素的节点使用 null
填充:
Tips: 若索引是从
i=0
开始的,左儿子left(i) = 2*i+1
,右儿子right(i) = 2*i+2
,parent(i) = (i-1)/2 取整
;
对于满二叉树来说,需要的节点数如下:
若当 n=2^k+1
需要的的空间数:
Tips:对于区间有
n
个元素的数组若n=2^k(k是正整数)
则需要2n
的空间,最差的情况是若n=2^k+1
则需要4n
的空间就足够了。
4.定义 SegmentTree 线段树类
其中定义了 leftSon($i)
方法,表示求某个节点左儿子节点索引值
的方法,rightSon($i)
表示求某个节点右儿子节点 索引值
的方法:
data[$i] = $arr[$i];
}
//若是静态语言需要开 4n 空间来表示 $this->tree
}
public function getSize() {
return count($this->data);
}
public function get(int $index) {
if ($index < 0 || $index >= count($this->data)) {
echo "索引错误";
exit;
}
return $this->data[$index];
}
/**
* 获取某个节点儿子节点索引,若索引是从 i=0 开始的,左儿子 left(i) = 2*i+1
* @param $i
* @return int
*/
private function leftSon($i): int {
return $i * 2 + 1;
}
/**
* 获取某个节点右儿子节点索引,若索引是从 i=0 开始的,右儿子 left(i) = 2*i+2
* @param $i
* @return int
*/
private function rightSon($i): int {
return $i * 2 + 2;
}
}
5.创建线段树
接下来使用递归
思想去 创建线段树
,下面给出递归函数 PHP
代码:
if ($left == $right) {
$this->tree[$i] = $this->data[$left]; //处理递归到叶子节点时 并赋值最原始的 $data 对应的索引值
} else {
$leftSon = $this->leftSon($i); //左儿子索引
$rightSon = $this->rightSon($i); //右儿子索引
$mid = $left + ceil(($right - $left) / 2);//求区间中值
$this->buildSegmentTree($leftSon, $left, $mid - 1); //递归左儿子树
$this->buildSegmentTree($rightSon, $mid, $right); //递归右儿子树
$this->tree[$i] = $this->merge->operate($this->tree[$leftSon], $this->tree[$rightSon]); //这里是根据业务来定节点需要存储的元素
}
Tips:其中节点元素存储的值需要根据业务来定,如上面代码表示的是每个节点存储的是
区间求和
的值,很显然这种方式不灵活,用户在实例化该类的时候可以传入一个merge
对象用于元素操作的。
6.节点元素计算规则
上述SegmentTree
类中可以在 __construct()
方法中传入一个 $merge
对象,$merge
中可以定义一个 operate()
方法计算得出节点元素值,如下:
merge = $merge;
for ($i = 0; $i < count($arr); $i++) {
$this->data[$i] = $arr[$i];
}
//若是静态语言需要开 4n 空间来表示 $this->tree
//递归创建线段树
$this->buildSegmentTree(0, 0, count($this->data) - 1);
}
private function buildSegmentTree(int $i, int $left, int $right) {
if ($left == $right) {
$this->tree[$i] = $this->data[$left]; //处理递归到叶子节点时 并赋值最原始的 $data 对应的索引值
} else {
$leftSon = $this->leftSon($i); //左儿子索引
$rightSon = $this->rightSon($i); //右儿子索引
$mid = $left + ceil(($right - $left) / 2);//求区间中值
$this->buildSegmentTree($leftSon, $left, $mid - 1); //递归左儿子树
$this->buildSegmentTree($rightSon, $mid, $right); //递归右儿子树
$this->tree[$i] = $this->merge->operate($this->tree[$leftSon], $this->tree[$rightSon]); //这里是根据业务来定节点需要存储的元素
}
}
public function getSize() {
return count($this->data);
}
public function get(int $index) {
if ($index < 0 || $index >= count($this->data)) {
echo "索引错误";
exit;
}
return $this->data[$index];
}
/**
* 获取某个节点儿子节点索引,若索引是从 i=0 开始的,左儿子 left(i) = 2*i+1
* @param $i
* @return int
*/
private function leftSon($i): int {
return $i * 2 + 1;
}
/**
* 获取某个节点右儿子节点索引,若索引是从 i=0 开始的,右儿子 left(i) = 2*i+2
* @param $i
* @return int
*/
private function rightSon($i): int {
return $i * 2 + 2;
}
}
6.1 Merge 类定义
如下定义就可以很灵活的处理每个节点的计算规则:
class Merge{
public funcrion operate($left,$right){
//这里可以定义需要操作的规则
return $left+$right; //如求平均值,这里可以 return ($left+$right)/2;
}
}
7. 求和演示
若是各个线段区间存储的是区间求和,则 Merge
类中的 operate()
方法返回是两个元素的和
,代码如下:
输出如下:
此时线段树的节点元素值示意图如下:
8. 线段树的区间查询
这里以查询
[2-6]
区间为例,若要查询区间 [2-6]
的求和需要根据区间来寻找需要求的值,示意图如下:
PHP 代码使用递归思想实现如下:
public function query($qleft, $qright) {
if ($qleft < 0 || $qright >= count($this->data) || $qright < $qleft) {
echo "索引范围错误";
exit;
}
return $this->recursionQuery(0, 0, count($this->data) - 1, $qleft, $qright);
}
/**
* 递归查询区间
* @param $left 当前节点区间左端值
* @param $right 当前节点区间右端值
* @param $qleft 需要查询的区间左端值
* @param $qright 需要查询的区间右端值
*/
private function recursionQuery($i, $left, $right, $qleft, $qright) {
$mid = $left + ceil(($right - $left) / 2);//求区间中值向上取整
//先处理满足区间条件的情况
if ($qleft == $left && $qright == $right) { //查询左右端和当前节点左右端重合
return $this->tree[$i];
} elseif ($qright < $mid) { //查询左右端在中值左边,那么结果区间在左儿子树
return $this->recursionQuery($this->leftSon($i), $left, $mid - 1, $qleft, $qright);
} elseif ($qleft >= $mid) { //查询左右端在中值右边,那么结果区间在右儿子树
return $this->recursionQuery($this->rightSon($i), $mid, $right, $qleft, $qright);
} else { //中值在查询左右端中间 将区间分成两边,结果在左右儿子树上都有
$leftSon = $this->recursionQuery($this->leftSon($i), $left, $mid - 1, $qleft, $mid - 1);
$righttSon = $this->recursionQuery($this->rightSon($i), $mid, $right, $mid, $qright);
return $this->merge->operate($leftSon, $righttSon);
}
}
输出如下:
9.完整 PHP 代码
9.1 SegmentTree 类
merge = $merge;
for ($i = 0; $i < count($arr); $i++) {
$this->data[$i] = $arr[$i];
}
//若是静态语言需要开 4n 空间来表示 $this->tree
//递归创建线段树
$this->buildSegmentTree(0, 0, count($this->data) - 1);
}
public function query($qleft, $qright) {
if ($qleft < 0 || $qright >= count($this->data) || $qright < $qleft) {
echo "索引范围错误";
exit;
}
return $this->recursionQuery(0, 0, count($this->data) - 1, $qleft, $qright);
}
/**
* 递归查询区间
* @param $left 当前节点区间左端值
* @param $right 当前节点区间右端值
* @param $qleft 需要查询的区间左端值
* @param $qright 需要查询的区间右端值
*/
private function recursionQuery($i, $left, $right, $qleft, $qright) {
$mid = $left + ceil(($right - $left) / 2);//求区间中值向上取整
//先处理满足区间条件的情况
if ($qleft == $left && $qright == $right) { //查询左右端和当前节点左右端重合
return $this->tree[$i];
} elseif ($qright < $mid) { //查询左右端在中值左边,那么结果区间在左儿子树
return $this->recursionQuery($this->leftSon($i), $left, $mid - 1, $qleft, $qright);
} elseif ($qleft >= $mid) { //查询左右端在中值右边,那么结果区间在右儿子树
return $this->recursionQuery($this->rightSon($i), $mid, $right, $qleft, $qright);
} else { //中值在查询左右端中间 将区间分成两边,结果在左右儿子树上都有
$leftSon = $this->recursionQuery($this->leftSon($i), $left, $mid - 1, $qleft, $mid - 1);
$righttSon = $this->recursionQuery($this->rightSon($i), $mid, $right, $mid, $qright);
return $this->merge->operate($leftSon, $righttSon);
}
}
private function buildSegmentTree(int $i, int $left, int $right) {
if ($left == $right) {
$this->tree[$i] = $this->data[$left]; //处理递归到叶子节点时 并赋值最原始的 $data 对应的索引值
} else {
$leftSon = $this->leftSon($i); //左儿子索引
$rightSon = $this->rightSon($i); //右儿子索引
$mid = $left + ceil(($right - $left) / 2);//求区间中值
$this->buildSegmentTree($leftSon, $left, $mid - 1); //递归左儿子树
$this->buildSegmentTree($rightSon, $mid, $right); //递归右儿子树
$this->tree[$i] = $this->merge->operate($this->tree[$leftSon], $this->tree[$rightSon]); //这里是根据业务来定节点需要存储的元素
}
}
public function getSize() {
return count($this->data);
}
public function get(int $index) {
if ($index < 0 || $index >= count($this->data)) {
echo "索引错误";
exit;
}
return $this->data[$index];
}
/**
* 获取某个节点儿子节点索引,若索引是从 i=0 开始的,左儿子 left(i) = 2*i+1
* @param $i
* @return int
*/
private function leftSon($i): int {
return $i * 2 + 1;
}
/**
* 获取某个节点右儿子节点索引,若索引是从 i=0 开始的,右儿子 left(i) = 2*i+2
* @param $i
* @return int
*/
private function rightSon($i): int {
return $i * 2 + 2;
}
}
9.2 输出演示代码
query(2,6);
代码仓库 :https://gitee.com/love-for-po...
扫码关注爱因诗贤
转:
数据结构-PHP 线段树的实现
数据结构-PHP 线段树的实现的更多相关文章
- 【Foreign】数据结构C [线段树]
数据结构C Time Limit: 20 Sec Memory Limit: 512 MB Description Input Output Sample Input Sample Output H ...
- 【数据结构】线段树(Segment Tree)
假设我们现在拿到了一个非常大的数组,对于这个数组里面的数字要反复不断地做两个操作. 1.(query)随机在这个数组中选一个区间,求出这个区间所有数的和. 2.(update)不断地随机修改这个数组中 ...
- 数据结构1 线段树查询一个区间的O(log N) 复杂度的证明
线段树属于二叉树, 其核心特征就是支持区间加法,这样就可以把任意待查询的区间$[L, R]$分解到线段树的节点上去,再把这些节点的信息合并起来从而得到区间$[L,R]$的信息. 下面证明在线段树上查询 ...
- 数据结构(线段树):Educational Codeforces Round 6 620E. New Year Tree
E. New Year Tree time limit per test 3 seconds memory limit per test 256 megabytes input standard in ...
- 数据结构(线段树):BZOJ 1568 [JSOI2008]Blue Mary开公司
1568: [JSOI2008]Blue Mary开公司 Time Limit: 15 Sec Memory Limit: 162 MBSubmit: 602 Solved: 214[Submit ...
- 牛客练习赛28 B数据结构(线段树)
链接:https://www.nowcoder.com/acm/contest/200/B来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言5242 ...
- 2018.10.12 NOIP模拟 数据结构(线段树)
传送门 sb线段树题居然还卡常. 修改操作直接更新区间最小值和区间标记下传即可. 询问加起来最多5e65e65e6个数. 因此直接询问5e65e65e6次最小值就行了. 代码
- 【uoj#228】基础数据结构练习题 线段树+均摊分析
题目描述 给出一个长度为 $n$ 的序列,支持 $m$ 次操作,操作有三种:区间加.区间开根.区间求和. $n,m,a_i\le 100000$ . 题解 线段树+均摊分析 对于原来的两个数 $a$ ...
- 数据结构习题 线段树&树状数组
说明:这是去年写了一半的东西,一直存在草稿箱里,今天整理东西的时候才发现,还是把它发表出来吧.. 以下所有题目来自Lrj的<训练指南> LA 2191 单点修改,区间和 Fenwick直 ...
随机推荐
- P2764 最小路径覆盖问题 (最小点覆盖=顶点数-最大匹配)
题意:最小路径覆盖 题解:对于一个有向图,最小点覆盖 = 顶点数 - 最大匹配 这里的最大匹配指的是将原图中每一个点拆成入点.出点, 每条边连接起点的出点和终点的入点 源点S连接每个点的出点,汇点T连 ...
- C. Table Decorations
time limit per test 1 second memory limit per test 256 megabytes input standard input output standar ...
- linux下安装python3.7.2
1.到python的官网去下载python3.7.2安装包,必须是Linux版本的 2.在/usr/tmp下下载python安装包 wget https://www.python.org/ftp/py ...
- [Python] Pandas 中 Series 和 DataFrame 的用法笔记
目录 1. Series对象 自定义元素的行标签 使用Series对象定义基于字典创建数据结构 2. DataFrame对象 自定义行标签和列标签 使用DataFrame对象可以基于字典创建数据结构 ...
- 链接脚本再探和VMA与LMA
链接脚本简单描述 连接脚本的描述都是以节(section)的单位的,网上也有很多描述链接脚本语法的好文章,再不济还有官方的说明文档可以用来学习,其实主要就是对编译构建的整个过程有了深入的理解后就能对链 ...
- Linux 驱动框架---platform驱动框架
Linux系统的驱动框架主要就是三个主要部分组成,驱动.总线.设备.现在常见的嵌入式SOC已经不是单纯的CPU的概念了,它们都会在片上集成很多外设电路,这些外设都挂接在SOC内部的总线上,不同与IIC ...
- Sentry & React
Sentry & React https://docs.sentry.io/platforms/javascript/guides/react/ https://docs.sentry.io/ ...
- JavaScript getter and setter All In One
JavaScript getter and setter All In One getter & setter JavaScript Object Accessors JavaScript A ...
- H5 广告落地页
H5 广告落地页 Landing Page 用于承接通过付费搜索渠道点击进入的用户,所以叫落地页 什么是登陆页面? 在数字营销中,登录页面是专门为营销或广告活动创建的独立网页. 访问者单击电子邮件中的 ...
- 微前端 & 微前端实践 & 微前端教程
微前端 & 微前端实践 & 微前端教程 微前端 micro frontends https://micro-frontends.org/ https://github.com/neul ...