[C++]线段树 区间查询 单点修改
线段树 区间查询 单点修改
算法思想
这个算法是用于数组的查询和修改
可以高效的进行查询修改
但是会增加内存的使用
本质上是一种 空间换时间 的算法
这个算法把一串数组无限二分
直到分的只剩下一个数据

将每一段看成一个节点
这样就组成了一个树形结构
故名 线段树
代码实现
实现这个代码一共分三个步骤:
建树 查询 修改
这里先把变量含义解释一遍:
#define maxn 1000010
#define mid ((l+r)>>1)
#define li i<<1
#define ri 1+(i<<1)
/*
mid 线段中间节点的小标
li i线段的左子树
ri i线段的右子树
*/
int n,val[maxn];
/*
n 数组的长度
val 数组的值
*/
struct Node{
int l,r,sum;
}tree[maxn];
/*
tree 即这个树形结构
tree[i].l i线段的左端
tree[i].r i线段的右端
tree[i].sum i线段的所有节点的权值和
*/
建树
void build(int i,int l,int r){
tree[i].l = l;
tree[i].r = r;
if(l == r){
tree[i].sum = val[l];
return ;
}
build(li,l,mid);
build(ri,mid+1,r);
tree[i].sum = tree[li].sum + tree[ri].sum;
return ;
}
欲建树 先分步
我们只要处理好每个节点的三个值(l,r,sum)
这棵树自然也就建好了
l,r直接赋值即可
如果 if(l == r)
则说明这个节点已经无法再二分了
那么就把 \(val\) 的值直接赋给 \(sum\)
并且要记得 return ;
若 \(l != r\)
那就继续二分建子树
然后再把两个子树的值加起来即为自己的 \(sum\)
查询
int search(int i,int l,int r){
if(l <= tree[i].l && tree[i].r <= r)
return tree[i].sum;
if(tree[i].r < l || r < tree[i].l)
return 0;
int ans = 0;
if(tree[li].r >= l) ans += search(li,l,r);
if(tree[ri].l <= r) ans += search(ri,l,r);
return ans;
}
这步的主要思想是能大块就返回大块的值
不能再二分给儿子线段处理
由于已经把数组分得很细
因此不存在查询边界在线段中却无法二分的情况

- \(l <= tree[i].l\) && \(tree[i].r <= r\)
这说明线段已经完全包裹在区间内(就和第二根绿色线段一样)
直接返回这个线段的值即可
- \(tree[i].r < l\) \(||\) \(r < tree[i].l\)
这说明线段完全不在取值区间内
那就返回0
- \(tree[li].r >= l\)
这说明有区间一部分在左子线段上
那就二分进行搜索
然后返回搜好的值
- \(tree[ri].l <= r\)
和上面同理
有区间一部分在右子线段上
修改
void add(int i,int dis,int k){
if(tree[i].l == tree[i].r){
tree[i].sum += k;
return ;
}
if(dis <= tree[li].r)
add(li,dis,k);
else
add(ri,dis,k);
tree[i].sum = tree[li].sum + tree[ri].sum;
return ;
}
修改我自身感觉和建树有点相像
就是改变一个节点的值然后再将涉及到这个节点的线段重新建树
- \(tree[i].l == tree[i].r\)
这代表已经找到了这个节点
那就把这个点的值修改掉
- \(dis <= tree[li].r\)
如果在线段里
那就继续找
- \(tree[i].sum = tree[li].sum + tree[ri].sum;\)
更新线段的值
线段数 区间修改 单点查询
Code
#include<bits/stdc++.h>
#define maxn 1000010
#define mid ((l+r)>>1)
#define li i<<1
#define ri 1+(i<<1)
using namespace std;
int n,val[maxn];
struct Node{
int l,r,sum;
}tree[maxn];
void Read(){
cin >> n;
for(int i = 1;i <= n;i++)cin >> val[i];
}
void build(int i,int l,int r){
tree[i].l = l;
tree[i].r = r;
if(l == r){
tree[i].sum = val[l];
return ;
}
build(li,l,mid);
build(ri,mid+1,r);
tree[i].sum = tree[li].sum + tree[ri].sum;
return ;
}
int search(int i,int l,int r){
if(l <= tree[i].l && tree[i].r <= r)
return tree[i].sum;
if(tree[i].r < l || r < tree[i].l)
return 0;
int ans = 0;
if(tree[li].r >= l) ans += search(li,l,r);
if(tree[ri].l <= r) ans += search(ri,l,r);
return ans;
}
void add(int i,int dis,int k){
if(tree[i].l == tree[i].r){
tree[i].sum += k;
return ;
}
if(dis <= tree[li].r)
add(li,dis,k);
else
add(ri,dis,k);
tree[i].sum = tree[li].sum + tree[ri].sum;
return ;
}
void interaction(){
while(1){
int tot;
cin >> tot;
if(tot == 1){
int l,r;
cin >> l >> r;
cout << search(1,l,r) << endl;
} else if(tot == 2){
int dis,k;
cin >> dis >> k;
add(1,dis,k);
} else if(tot == 3){
return ;
}
}
}
int main(){
cout << "query section" << endl << "change point" << endl;
Read();
build(1,1,n);
cout << "query 1" << endl << "change 2" << endl << "break 3" << endl;
interaction();
return 0;
}
[C++]线段树 区间查询 单点修改的更多相关文章
- hdu 1754 线段树(Max+单点修改)
I Hate It Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- hdu 1166 线段树(sum+单点修改)
敌兵布阵 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submi ...
- I Hate It:线段树:单点修改+区间查询
I Hate It Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- poj3171 Cleaning Shifts【线段树(单点修改区间查询)】【DP】
Cleaning Shifts Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 4422 Accepted: 1482 D ...
- HZAU 1207 Candies(线段树区间查询 区间修改)
[题目链接]http://acm.hzau.edu.cn/problem.php?id=1207 [题意]给你一个字符串,然后两种操作:1,将区间L,R更新为A或者B,2,询问区间L,R最长的连续的B ...
- 【线段树(单点修改,区间求和)】HDU1166 - 敌军布阵
hdu1166 敌兵布阵,单点修改,区间求和. [ATTENTION]MAXN要开成节点数的4倍,开得不够会提示TLE. #include<iostream> #include<cs ...
- HDU - 1166 - 敌兵布阵 线段树的单点修改,区间求和
#include<iostream> #include<stdio.h> #include<string.h> #include<algorithm> ...
- 线段树 区间查询区间修改 poj 3468
#include<cstdio> #include<iostream> #include<algorithm> #include<string.h> u ...
- hdu1754线段树的单点更新区间查询
I Hate It Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- HDU 1754 I Hate It(线段树区间查询,单点更新)
描述 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少. 这让很多学生很反感.不管你喜不喜欢,现在需要你做的是,就是按照老师的要求,写一个程序,模拟老师的询问.当然,老 ...
随机推荐
- 【转载】Linux虚拟化KVM-Qemu分析(十)之virtio驱动
原文信息 作者:LoyenWang 出处:https://www.cnblogs.com/LoyenWang/ 公众号:LoyenWang 版权:本文版权归作者和博客园共有 转载:欢迎转载,但未经作者 ...
- 2023ccpc大学生程序设计竞赛-zzh
比赛开始没有开到签到题,看了一会别的题才开始跟榜.A题我写的,不过没有考虑到S长度为1的情况,wa了一次.然后lhy和zx把F题做了出来.接着他俩去看H,我去看B.他俩把H过了,B我想出了一个n*根n ...
- 手写raft(一) 实现leader选举
1. 一致性算法介绍 1.1 一致性同步与Paxos算法 对可靠性有很高要求的系统,通常都会额外部署1至多个机器为备用副本组成主备集群,避免出现单点故障. 有状态的系统需要主节点与备用副本间以某种方式 ...
- Centos7下创建Oracle用户
Centos7下创建Oracle用户 Oracle中,一个用户其实就类似于一个数据库,本次就来创建一个新用户 登录 将系统用户切换到oracle用户下 su - oracle -- 启动sqlplus ...
- Vue详解----一篇带你从头领悟到尾,享受飞升的感觉
脚手架文件结构 """ ├── node_modules ├── public │ ├── favicon.ico: 页签图标 │ └── index.html: 主页面 ...
- Swiper.vue?56a2:132 Uncaught DOMException: Failed to execute 'insertBefore' on 'Node': The node before which the new node is to be inserted is not a child of this node.
错误代码 解决方案 删除div标签.修改后,如下所示:
- gpg 解密-禁用交互式密码输入
背景描述 gpg 解密默认弹出如下窗口,请用户输入密码,但在脚本自动化时遇到了问题 lqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqqk x ...
- 「Go笔记-02」变量、数据类型、数据类型间转换、进制转换...看这一篇就Go了
前言 一个程序就是一个世界,不论是使用哪种高级程序语言编写程序, 变量都是其程序的基本组成单位, 变量 在 go 中 变量是用于存储数据的命名空间(内存位置),它可以表示一个值,这个值在程序执行过程中 ...
- Go语言-Slice详解
Go语言中的slice表示一个具有相同类型元素的可变长序列,语言本身提供了两个操作方法: 创建:make([]T,len,cap) 追加: append(slice, T ...) 同时slice支持 ...
- LeetCode952三部曲之一:解题思路和初级解法(137ms,超39%)
欢迎访问我的GitHub 这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos 题目描述 难度:困难 编程语言:Java 给定一个由不 ...