首页 > 其他分享 >线段树

线段树

时间:2023-12-14 22:44:55浏览次数:34  
标签:index int 线段 length data public

线段树

什么是线段树

线段树(英语:Segment tree)是一种二叉树形数据结构,1977年由Jon Louis Bentley发明[1],用以存储区间线段,并且允许快速查询结构内包含某一点的所有区间。

一个包含n个区间的线段树,空间复杂度为O(n),查询的时间复杂度则为O(logn+k)},其中k是匹配条件的区间数量。

此数据结构亦可推广到高维度。

摘自《维基百科》

为什么要使用线段树

线段树的时间复杂度分析:

线段树基础表示

如果区间有n个元素,用数组表示线段树,需要多少结点?

0层:1

1层:2

2层:4

3层:8

...

h-1层:2^(h-1)

对于满二叉树:

h层,一共有2h-1个结点(约为2h)

最后一层(h-1),有2^(h-1)个结点

最后一层的结点数大致等于前面所有层的结点数之和

由此,可得出结论,我们线段树不考虑添加元素,即区间固定,使用4*n的静态空间即可。

public class SegmentTree<E> {
    private E[] tree;
    private E[] data;
    public SegmentTree(E[] arr){
        data=(E[])new Object[arr.length];
        for(int i=0;i<arr.length;i++){
            data[i]=arr[i];
        }
        tree=(E[])new Object[4*arr.length];
    }

    public int getSize(){
        return data.length;
    }

    public E get(int index){
        if(index<0 || index>=data.length){
            throw new IllegalArgumentException("Index is illegal");
        }
        return data[index];
    }

    //返回完全二叉树的数组表示,一个索引所表示的元素的左孩子结点的索引
    public int leftChild(int index){
        return 2*index+1;
    }

    //返回完全二叉树的数组表示,一个索引所表示的元素的右孩子结点的索引
    public int rightChild(int index){
        return 2*index+2;
    }
}

创建线段树

public class SegmentTree<E> {
    private E[] tree; //存储线段树中数据
    private E[] data;
    private Merger<E> merger;

    public SegmentTree(E[] arr, Merger<E> merger){
        this.merger=merger;
        data=(E[])new Object[arr.length];
        for(int i=0;i<arr.length;i++){
            data[i]=arr[i];
        }
        tree=(E[])new Object[4*arr.length];
        buildSegmentTree(0,0,data.length-1);
    }

    //在treeIndex根节点的位置创建区间在[l,r]的线段树
    private 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)/2;
        //创建左子树的线段树
        buildSegmentTree(leftTreeIndex,l,mid);
        //创建右子树的线段树
        buildSegmentTree(rightTreeIndex,mid+1,r);
        //当左右子树创建完成之后,综合左右子树的结果,
        //如何去综合是由业务逻辑去决定的。
        tree[treeIndex]=merger.meger(tree[leftTreeIndex],tree[rightTreeIndex]);
    }

    public int getSize(){
        return data.length;
    }

    public E  get(int index){
        if(index<0 || index>=data.length){
            throw new IllegalArgumentException("Index is illegal");
        }
        return data[index];
    }

    //在完全二叉树的数组表示中,获取左孩子节点的索引
    private int leftChild(int index){
        return 2*index+1;
    }

    //在完全二叉树的数组表示中,获取右孩子节点的索引
    private int rightChild(int index){
        return 2*index+2;
    }

    @Override
    public String toString() {
        StringBuilder res=new StringBuilder();
        res.append("[");
        for(int i=0;i<tree.length;i++){
            if(tree[i]!=null){
                res.append(tree[i]);
            }else{
                res.append("null");
            }
            if(i!=tree.length-1){
                res.append(", ");
            }
        }
        res.append("]");
        return res.toString();
    }
}
  • 线段树的合并器接口

    融合器,根据业务逻辑来融合左右子树的内容

public interface Merger<E> {
    E meger(E a,E b);
}

线段树的查询

//查询区间[queryL,queryR]的值
public E query(int queryL, int queryR){
    if(queryL<0 || queryL>=data.length
            || queryR<0 || queryR>=data.length || queryL>queryR){
        throw new IllegalArgumentException("Index is illegal");
    }
    return query(0,0,data.length-1,queryL,queryR);
}

//以treeIndex为根的线段树[l...r]的范围搜索[queryL,queryR]区间
private E query(int treeIndex,int l,int r,int queryL,int queryR){
    //搜索到目标区间
    if(l==queryL && r==queryR){
        return tree[treeIndex];
    }
    int mid=l+(r-l)/2;
    int leftTreeIndex=leftChild(treeIndex);
    int rightTreeIndex=rightChild(treeIndex);
    if(queryL>=mid+1){
        //仅在右部分
        return query(rightTreeIndex,mid+1,r,queryL,queryR);
    }else if(queryR<=mid){
        //仅在左部分
        return query(leftTreeIndex,l,mid,queryL,queryR);
    }
    //剩下的情况是:有部分落在左区间,另一部分落在右区间
    
    //在左子树中寻找
    E leftResult=query(leftTreeIndex,l,mid,queryL,mid);
    //在右子树中寻找
    E rightResult=query(rightTreeIndex,mid+1,r,mid+1,queryR);
    return merger.merge(leftResult,rightResult);
}

线段树的更新

//更新index位置的值
public void set(int index,E e){
    if(index<0 || index>data.length){
        throw new IllegalArgumentException("Index is illegal");
    }
    set(0,0,data.length-1,index,e);
}

//更新以treeIndex为根的线段树[l...r]的值为e
private void set(int treeIndex,int l,int r,int index,E e){
    //搜索到目标,直接更新
    if(l==r){
        tree[treeIndex]=e;
        return;
    }
    
    int mid=l+(r-l)/2;
    int leftTreeIndex=leftChild(treeIndex);
    int rightTreeIndex=rightChild(treeIndex);
    if(index>=mid+1){
        //继续到右子树更新
        set(rightTreeIndex,mid+1,r,index,e);
    }else if(index<=mid){
        //继续到左子树更新
        set(leftTreeIndex,l,mid,index,e);
    }
    //更新祖辈节点
    tree[treeIndex]=merger.merge(tree[leftTreeIndex],tree[rightTreeIndex]);
}

LeetCode中有关线段树的问题

LeetCode 303题 区域和检索 - 数组不可变

public class NumArray {
    private SegmentTree<Integer> segmentTree;

    public NumArray(int[] nums) {
        if(nums.length>0){
            Integer[] data=new Integer[nums.length];
            for(int i=0;i<nums.length;i++){
                data[i]=nums[i];
            }
            segmentTree=new SegmentTree<Integer>(data,(a,b)-> a+b);
        }
    }

    public int sumRange(int i, int j) {
        if(segmentTree==null){
            throw new IllegalArgumentException("Segment Tree is null");
        }
        return segmentTree.query(i,j);
    }
}

https://www.mianshi.onlinehttps://www.i9code.cn
LeetCode 307题 区域和检索 - 数组可修改

private SegmentTree<Integer> segmentTree;

public NumArray2(int[] nums) {
    if(nums.length>0){
        Integer[] data=new Integer[nums.length];
        for(int i=0;i<nums.length;i++){
            data[i]=nums[i];
        }
        segmentTree=new SegmentTree<Integer>(data,(a,b)-> a+b);
    }
}

public void update(int i, int val) {
    if(segmentTree==null){
        throw new IllegalArgumentException("Segment Tree is null");
    }
    segmentTree.set(i,val);
}

public int sumRange(int i, int j) {
    if(segmentTree==null){
        throw new IllegalArgumentException("Segment Tree is null");
    }
    return segmentTree.query(i,j);
}

本文由博客一文多发平台 OpenWrite 发布!

标签:index,int,线段,length,data,public
From: https://www.cnblogs.com/i9code/p/17902341.html

相关文章

  • 【UVCAD】- 多段线教程,及与线段的区别
    【UVCAD】手机二维CAD建模,不止是看图UVCAD专注于二维(2D)的移动计算机辅助绘图(CAD)。UVCAD具有触摸优化的直观界面和工具。使用UVCAD,您可以在触摸屏上用手指或铅笔进行真正的2D绘图、2D建模和2D设计。对于需要易于使用的工具来更快、更精确地创建图纸的设计师和绘图员来说,UVC......
  • 刷题 ST表、单调栈、线段树->区间最值
    2023.12.13cf1904D2解题思路首先,a[i]大于b[i]时肯定不行,等于就满足了,直接过掉其次,要想使得a[i]等于b[i],就要在a[i]左右找最近的j使得a[j]=b[i](最近的最优,可证)k是i和j中间的一个数,想要满足题意,要满足以下两个条件(a[j]=b[i])1.ak<aj,即max(区间[i,j]中的ak)2.bk>bi,即min(区间......
  • 【线段树入门】P3353 在你窗外闪耀的星星(区间求和)
    这题正解是前缀和,我用线段树练练手><11//笔记-自用22//#pragmaGCCoptimize("Ofast")33//#pragmaGCCoptimize("unroll-loops")44#define_CRT_SECURE_NO_WARNINGS55#defineAll(a)a.begin(),a.end()66#defineINF214748364777......
  • 【线段树入门】 P1198 最大数(区间最大值+无懒标记+末尾插入)
    1//笔记-自用2//#pragmaGCCoptimize("Ofast")3//#pragmaGCCoptimize("unroll-loops")4#define_CRT_SECURE_NO_WARNINGS5#defineAll(a)a.begin(),a.end()6#defineINF21474836477#include<bits/stdc++.h>8#include<nu......
  • 【线段树入门】P3373 线段树 2(区间乘加)
    //笔记-自用//#pragmaGCCoptimize("Ofast")//#pragmaGCCoptimize("unroll-loops")#define_CRT_SECURE_NO_WARNINGS#defineAll(a)a.begin(),a.end()#defineINF2147483647#include<bits/stdc++.h>#include<numeric>usingnamespac......
  • 2020年初一初二集训队(线段树) 基本操作
    其他线段树详解与实现-知乎⁤(zhihu.com)线段树-OIWiki(oi-wiki.org) 线段树学习笔记-xujindong的博客-洛谷博客(luogu.com.cn)  简介线段树(segmenttree)是一种二叉搜索树,也是平衡二叉树,它的每一个结点对应一个区间[L,R],叶子结点对应的区间只有一个......
  • 线段树模板区间加(含懒标记)
    constintN=1e5+10;intn,m;inta[N];structTree{ intl,r; llsum,add; }tr[4*N];voidbuild(intu,intl,intr){ //l=tr[u].l;r=tr[u].r; //注释掉的部分是典型的错误,对于build过程中我们是初始化编号对应区间节点,上面赋值逻辑反了 tr[u]={l,r}; if(l==r)t......
  • 线段树
    首先是建树我们先构建整棵树的框架 structnode{intl,r;stringdata;}g[N*4];//不一定非要构建结构体,看题目需求,如果不涉及左右范围的话就可以直接构造数组 //n表示的是树上每个结点的数值,比如说第一个结点为1,那莫第一个结点的左子树为2,右子树为3//l(是......
  • 【数据结构】线段树 (二) 学习笔记
    线段树(二)点击查看:线段树(一)学习笔记本文介绍权值线段树与动态开点线段树,(可能后面还会加线段树合并等等)。权值线段树线段树的动态开点线段树合并推荐题目&&参考资料&&拓展阅读《算法竞赛进阶指南》0x43线段树P3870 [TJOI2009]开关P1438 无聊的数列P1253 扶苏的问......
  • 李超线段树
    问题:洛谷P4097在平面直角坐标系维护两个操作:1.加入一条线段。2.求目前平面直角坐标系中截一条直线\(x=k\)中与线段交点\(y\)最大的是那一条线段。解决:李超线段树模板。首先建一个以\(x\)为区间的线段树。和普通线段树的主要区别是在对懒标记的处理上,这里是是没有单独的下......