线段树入门:算法分析

算法分析

线段树采用了分而治之的策略,其点更新、区间更新、区间查询都可以在 时间内完成。树状数组和线段树都用于解决频繁修改和查询的问题,树状数组比线段树更节省空间、代码简单易懂,但是先单数用途更广、更加灵活,凡是可以使用树状数组的问题都可以用线段树来解决。

例题讲解

线段树的题目并不是一成不变的,对于不同的题目,你可能需要修改线段树的模板,来实现各种功能。

区域和检索-数组可修改

给你一个数组 nums ,请你完成两类查询。

  1. 其中一类查询要求 更新 数组 nums 下标对应的值
  2. 另一类查询要求返回数组 nums 中索引 left 和索引 right 之间(包含 )的nums元素的 ,其中 left <= right

实现 NumArray 类:

  • NumArray(int[] nums) 用整数数组 nums 初始化对象
  • void update(int index, int val)nums[index] 的值 更新val
  • int sumRange(int left, int right) 返回数组 nums 中索引 left 和索引 right 之间( 包含 )的 nums 元素的 (即,nums[left] + nums[left + 1], ..., nums[right]

示例 1:

输入"NumArray", "sumRange", "update", "sumRange"\[\[1, 3, 5], 0, 2, 1, 2, 0, 2] 输出null, 9, null, 8

解释

NumArray numArray = new NumArray(1, 3, 5);

numArray.sumRange(0, 2); // 返回 1 + 3 + 5 = 9

numArray.update(1, 2); // nums = 1,2,5

numArray.sumRange(0, 2); // 返回 1 + 2 + 5 = 8

解题思路

本题是经典的线段树模板题,只需要根据之前的线段树模板稍作修改即可 (将区间最值查询改为区间求和)。

参考代码

复制代码
class NumArray {
public:
struct sgntree{
    int l,r,val;
}T[120010];
vector<int>num;
    void build(int p,int l,int r){
        T[p].l=l,T[p].r=r;
        if(l==r){
            T[p].val=num[l-1];
            return;
        }
        int mid=(l+r)>>1;
        build(p<<1,l,mid);
        build(p<<1|1,mid+1,r);
        T[p].val=T[p<<1].val+T[p<<1|1].val;
    }

    NumArray(vector<int>& nums) {
        num=nums;
        int n=nums.size();
        build(1,1,n);
    }
    void Update(int p,int l,int val){
        if(T[p].l==T[p].r&&T[p].l==l){
            T[p].val=val;
            return;
        }
        int mid=(T[p].l+T[p].r)>>1;
        if(l<=mid)
        Update(p<<1,l,val);
        else
        Update(p<<1|1,l,val);
        T[p].val=T[p<<1].val+T[p<<1|1].val;
    }
    void update(int index, int val) {
        Update(1,index+1,val);
    }
    int query(int p,int l,int r){
        if(T[p].l>=l&&T[p].r<=r){
            return T[p].val;
        }
        int mid=(T[p].l+T[p].r)>>1;
        if(r<=mid)
        return query(p<<1,l,r);
        if(l>mid)
        return query(p<<1|1,l,r);
        return query(p<<1,l,mid)+query(p<<1|1,mid+1,r);
    }
    int sumRange(int left, int right) {
        return query(1,left+1,right+1);
    }
};

/**
 * Your NumArray object will be instantiated and called as such:
 * NumArray* obj = new NumArray(nums);
 * obj->update(index,val);
 * int param_2 = obj->sumRange(left,right);
 */
相关推荐
QiLinkOS4 小时前
第三视觉理解徐玉生与他的商业活动(30)
大数据·c++·人工智能·算法·开源协议
疯狂打码的少年4 小时前
【操作系统】页面置换算法(OPT/FIFO/LRU)
算法
小O的算法实验室4 小时前
2026年CIE,优化客货协同运输:综合地铁系统的列车容量动态分配
算法
Coder_Shenshen5 小时前
西门子S7CommPlus协议鉴权算法原理与流程详解
网络·后端·算法
硕风和炜6 小时前
【LeetCode: 2492. 两个城市间路径的最小分数 + DFS】
java·算法·leetcode·深度优先·dfs·bfs·并查集
我是一颗柠檬7 小时前
【Java项目技术亮点】加权轮询负载均衡算法
java·算法·负载均衡
灯厂码农7 小时前
C语言动态内存分配完全指南(malloc、calloc、realloc、free)
java·c语言·算法
凯瑟琳.奥古斯特8 小时前
K次取反最大化数组和解法(力扣1005)
开发语言·c++·算法·leetcode·职场和发展
Jerry9 小时前
LeetCode 203. 移除链表元素
算法
地平线开发者9 小时前
征程 6 | 工具链 QAT ObserverBase 源码解析
算法