线段树入门:算法分析

算法分析

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

例题讲解

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

区域和检索-数组可修改

给你一个数组 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);
 */
相关推荐
水木流年追梦5 小时前
大模型入门-DPO 直接偏好优化
人工智能·学习·算法·机器学习·正则表达式
悠仁さん6 小时前
数据结构 树 二叉树 堆 (堆的模拟实现篇)
数据结构
汉克老师6 小时前
GESP6级C++考试语法知识(二十七、广度优先搜索(二、二维BFS))
c++·算法·图论·宽度优先·广度优先搜索·gesp6级·gesp六级
此生决int6 小时前
算法从入门到精通——位运算
数据结构·c++·算法·蓝桥杯
春栀怡铃声6 小时前
【C++修仙录02】筑基篇:vector 使用
开发语言·c++·算法
Loli_Wolf6 小时前
AI 原生研发闭环:从提需到线上监测,再自动回到提需
人工智能·深度学习·算法·microsoft·ai·ai编程·harness
计算机安禾6 小时前
【算法分析与设计】第4篇:分治策略的理论框架与经典案例
数据结构·算法·排序算法
Kiling_07046 小时前
面向对象和集合编程题 ( 二 )
java·开发语言·数据结构·算法
过期动态6 小时前
【LeetCode 热题 100】两数之和— 暴力法与哈希表法详解
java·数据结构·算法·leetcode·散列表