区间修改与区间查询问题

模板:

int ans;

struct node{
int l,r,v;
node(){v=;}
}tree[LEN*];
int arr[LEN];
//建树
void build(int l,int r,int i){ //记录 [ l , r ]上的值,当前索引为 i
tree[i].l=l;
tree[i].r=r;
if(l==r){ //如果访问到了叶子节点
tree[i].v=arr[l]; //当前值就是arr[l,r]的值
return;
}
int m=(l+r)/; //二分
build(l,m,LS(i)); //左子树
build(m+,r,RS(i)); //右子树
tree[i].v=tree[LS(i)].v+tree[RS(i)].v; //递归建树出栈后,前驱结点的值为后继结点的和
}
//区间查询
void query(int i,int l,int r){ // i 为当前树的索引,初试化调用为 1 。查询 [ l , r ]区间上的和
if(tree[i].l>=l && tree[i].r<=r){ //如果查询区间包裹住了当前结点区间
ans+=tree[i].v; //直接加上这个结点的值
return;
}
if(l<=tree[LS(i)].r) //查询的 l 比左子树的右端点小
query(LS(i),l,r); //向左递归
if(r>=tree[RS(i)].l) //查询的 r 比右子树的左端点大
query(RS(i),l,r); //向右递归
}
//单点修改
void plus_i(int i,int p,int d){ //index(current) position delta
tree[i].v+=d; //当前结点的值更新
if(tree[i].l==tree[i].r) //访问到了叶子节点
return; //退出
if(p<=tree[LS(i)].r) //查询的 p 比左子树的右端点小
plus_i(LS(i), p,d);
if(p>=tree[RS(i)].l) //查询的 p 比右子树的左端点大
plus_i(RS(i), p,d);
}
//区间修改
void plus_s(int i,int l,int r,int d){
//取查询区间与结点区间的交集
int tl=max(l,tree[i].l);
int tr=min(r,tree[i].r);
if(tl<=tr) tree[i].v+=d*(tr-tl+); //如果这个交集合法,区间加
else return;
if(l<=tree[LS(i)].r) //查询的 l 比左子树的右端点小
plus_s(LS(i), l,r,d);
if(r>=tree[RS(i)].l) //查询的 r 比右子树的左端点大
plus_s(RS(i), l,r,d);
}

注:可以把上文的结构体拆写为3个数组

测试OJ:P3372 【模板】线段树 1

测试代码:(因为无lazy优化,只有70分。我理解不了lazy优化)

#include <stdio.h>
#include <memory.h>
#include <math.h>
#include <string>
#include <vector>
#include <set>
#include <stack>
#include <queue>
#include <algorithm>
#include <map> #define I scanf
#define OL puts
#define O printf
#define F(a,b,c) for(a=b;a<c;a++)
#define FF(a,b) for(a=0;a<b;a++)
#define FG(a,b) for(a=b-1;a>=0;a--)
#define LEN 100010
#define MAX 1<<30
#define V vector<int>
#define ll long long
#define LS(i) i<<1
#define RS(i) i<<1|1 using namespace std; int ans; struct node{
int l,r,v;
node(){v=;}
}tree[LEN*];
int arr[LEN];
//建树
void build(int l,int r,int i){ //记录 [ l , r ]上的值,当前索引为 i
tree[i].l=l;
tree[i].r=r;
if(l==r){ //如果访问到了叶子节点
tree[i].v=arr[l]; //当前值就是arr[l,r]的值
return;
}
int m=(l+r)/; //二分
build(l,m,LS(i)); //左子树
build(m+,r,RS(i)); //右子树
tree[i].v=tree[LS(i)].v+tree[RS(i)].v; //递归建树出栈后,前驱结点的值为后继结点的和
}
//区间查询
void query(int i,int l,int r){ // i 为当前树的索引,初试化调用为 1 。查询 [ l , r ]区间上的和
if(tree[i].l>=l && tree[i].r<=r){ //如果查询区间包裹住了当前结点区间
ans+=tree[i].v; //直接加上这个结点的值
return;
}
if(l<=tree[LS(i)].r) //查询的 l 比左子树的右端点小
query(LS(i),l,r); //向左递归
if(r>=tree[RS(i)].l) //查询的 r 比右子树的左端点大
query(RS(i),l,r); //向右递归
}
//单点修改
void plus_i(int i,int p,int d){ //index(current) position delta
tree[i].v+=d; //当前结点的值更新
if(tree[i].l==tree[i].r) //访问到了叶子节点
return; //退出
if(p<=tree[LS(i)].r) //查询的 p 比左子树的右端点小
plus_i(LS(i), p,d);
if(p>=tree[RS(i)].l) //查询的 p 比右子树的左端点大
plus_i(RS(i), p,d);
}
//区间修改
void plus_s(int i,int l,int r,int d){
//取查询区间与结点区间的交集
int tl=max(l,tree[i].l);
int tr=min(r,tree[i].r);
if(tl<=tr) tree[i].v+=d*(tr-tl+); //如果这个交集合法,区间加
else return;
if(l<=tree[LS(i)].r) //查询的 l 比左子树的右端点小
plus_s(LS(i), l,r,d);
if(r>=tree[RS(i)].l) //查询的 r 比右子树的左端点大
plus_s(RS(i), l,r,d);
} int main(){
// freopen("D:\\CbWorkspace\\ACM数据结构\\线段树\\模板1.txt","r",stdin);
int N,M,i,t,op,x,y,k;
I("%d%d",&N,&M);
for(i=;i<=N;i++){
I("%d",&arr[i]);
}
build(,N,);
while(M--){
I("%d%d%d",&op,&x,&y);
switch(op){
case :
I("%d",&k);
plus_s(,x,y,k);
break;
case :
ans=;
query(,x,y);
O("%d\n",ans);
break;
}
}
return ;
}

区间最大最小值查询问题

OJ 链接:P1198 [JSOI2008]最大数

AC代码:

#include <stdio.h>
#include <memory.h>
#include <math.h>
#include <string>
#include <vector>
#include <set>
#include <stack>
#include <queue>
#include <algorithm>
#include <map> #define I scanf
#define OL puts
#define O printf
#define F(a,b,c) for(a=b;a<c;a++)
#define FF(a,b) for(a=0;a<b;a++)
#define FG(a,b) for(a=b-1;a>=0;a--)
#define LEN 400010
#define MAX 1<<30
#define V vector<int>
#define ll long long
#define LS(i) i<<1
#define RS(i) i<<1|1 using namespace std; int ans; struct node{
int l,r,v;
node(){v=;}
}tree[LEN*]; //建树
void build(int l,int r,int i){ //记录 [ l , r ]上的值,当前索引为 i
tree[i].l=l;
tree[i].r=r;
tree[i].v=-MAX;
if(l==r){ //如果访问到了叶子节点
return;
}
int m=(l+r)/; //二分
build(l,m,LS(i)); //左子树
build(m+,r,RS(i)); //右子树
}
//区间查询
void query(int i,int l,int r){ // i 为当前树的索引,初试化调用为 1 。查询 [ l , r ]区间上的和
if(tree[i].l>=l && tree[i].r<=r){ //如果查询区间包裹住了当前结点区间
ans=max(ans,tree[i].v); //取最大值
return;
}
if(l<=tree[LS(i)].r) //查询的 l 比左子树的右端点小
query(LS(i),l,r); //向左递归
if(r>=tree[RS(i)].l) //查询的 r 比右子树的左端点大
query(RS(i),l,r); //向右递归
}
//单点修改
void plus_i(int i,int p,int d){ //index(current) position delta
tree[i].v=max(tree[i].v,d); //取最大值
if(tree[i].l==tree[i].r) //访问到了叶子节点
return; //退出
if(p<=tree[LS(i)].r) //查询的 p 比左子树的右端点小
plus_i(LS(i), p,d);
if(p>=tree[RS(i)].l) //查询的 p 比右子树的左端点大
plus_i(RS(i), p,d);
} int main(){
// freopen("D:\\CbWorkspace\\ACM数据结构\\线段树\\最大数.txt","r",stdin);
int N,D,t=,num,pos=;
build(,LEN,);
char buf[];
I("%d%d",&N,&D);
while(N--){
I("%s%d",buf,&num);
if(buf[]=='A'){
num+=t;
num%=D;
plus_i(,pos++,num);
}else{
if(num==){
O("%d\n",t=);
continue;
}
ans=-MAX;
query(,pos-num,pos-);
O("%d\n",t=ans);
}
}
return ;
}

线段树模板(无lazy优化)的更多相关文章

  1. UESTC - 1057 秋实大哥与花 线段树模板题

    http://acm.uestc.edu.cn/#/problem/show/1057 题意:给你n个数,q次操作,每次在l,r上加上x并输出此区间的sum 题解:线段树模板, #define _CR ...

  2. POJ3468:A Simple Problem with Integers(线段树模板)

    A Simple Problem with Integers Time Limit: 5000MS   Memory Limit: 131072K Total Submissions: 149972 ...

  3. 洛谷P3372线段树模板1——线段树

    题目:https://www.luogu.org/problemnew/show/P3372 线段树模板. 代码如下: #include<iostream> #include<cst ...

  4. [AHOI 2009] 维护序列(线段树模板题)

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec  Memory Limit: 64 MB Description 老师交给小可可一个维护数列的任务,现在小 ...

  5. hdu1754 I hate it线段树模板 区间最值查询

    题目链接:这道题是线段树,树状数组最基础的问题 两种分类方式:按照更新对象和查询对象 单点更新,区间查询; 区间更新,单点查询; 按照整体维护的对象: 维护前缀和; 维护区间最值. 线段树模板代码 # ...

  6. P3373 线段树模板

    好,这是一个线段树模板. #include <cstdio> using namespace std; ; long long int sum[N],tag1[N],tag2[N],mo; ...

  7. 线段树模板hdu 1754:I Hate It

    I Hate It Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total S ...

  8. POJ 3468 A Simple Problem with Integers(线段树模板之区间增减更新 区间求和查询)

    A Simple Problem with Integers Time Limit: 5000MS   Memory Limit: 131072K Total Submissions: 140120 ...

  9. 【BZOJ4311】向量(线段树分治,斜率优化)

    [BZOJ4311]向量(线段树分治,斜率优化) 题面 BZOJ 题解 先考虑对于给定的向量集,如何求解和当前向量的最大内积. 设当前向量\((x,y)\),有两个不同的向量\((u1,v1),(u2 ...

随机推荐

  1. JavaScript 数据结构与算法之美 - 非线性表中的树、堆是干嘛用的 ?其数据结构是怎样的 ?

    1. 前言 想学好前端,先练好内功,内功不行,就算招式练的再花哨,终究成不了高手. 非线性表(树.堆),可以说是前端程序员的内功,要知其然,知其所以然. 笔者写的 JavaScript 数据结构与算法 ...

  2. Ubuntu 14.04 安装mysql

    Ubuntu 14.04 没有mysql5.7的源,需要连接外部资源下载安装. wget http://dev.mysql.com/get/mysql-apt-config_0.8.1-1_all.d ...

  3. Kafka 入门1

    Kafka 简介 作为一个消息中间件,Kafka 以高扩展性.高吞吐量等特点,在互联网项目中被广泛采用. 不清楚 Kafka 的同学,可以先看看这篇文章: http://blog.csdn.net/s ...

  4. 小玩意儿之Gitlab 代码提交日志同步到禅道项目管理系统

    以前都是使用禅道官方推荐的服务器本地扫描的方式,但其实不太方便,需要跟着项目的变化,不断的在配置文件维护项目相应仓库的配置. 然后现在Web Hooks越来越普遍的情况下,想尝试一种新的方式.看了禅道 ...

  5. H5 背景图片自适应屏幕问题解决办法

    今天也是偶然遇到这个问题的,平成很少用到关于css的内容,就在网上找了很多个方法,最后总结以下几种很实用的方法 方法1: <style> body{ background:url(beij ...

  6. 队列和 BFS —— 栈和 DFS

    队列和 BFS: 广度优先搜索(BFS)的一个常见应用是找出从根结点到目标结点的最短路径. 示例 这里我们提供一个示例来说明如何使用 BFS 来找出根结点 A 和目标结点 G 之间的最短路径. 洞悉 ...

  7. MySQL数据库中查询表的所有列名

    MySQL数据库中: 查询某个数据库中某个表的所有列名 SELECT COLUMN_NAME FROM information_schema.COLUMNS WHERE TABLE_SCHEMA = ...

  8. 【杂文】CSP2019蒟蒻AFO(假)记

    [杂文]CSP2019蒟蒻AFO(假)记 [初赛前 N 天] 时间:2019-10-15 今晚 \(2012\) 的初赛题做到心态爆炸,选择考计算机基础知识一脸懵逼,填空和后面一道大模拟直接跳过,最后 ...

  9. pyqt5学习

    详细设计追函数报告生成 界面大致如下: 部分UI代码: #!/usr/bin/env python3.7 # -*- coding:utf-8 -*- # Author: Lancer 2019-09 ...

  10. jenkins安装后提示localhost 拒绝了我们的连接请求。

    我是用msi文件安装的windows本地 ,安装文件看另外安装的博文. 此问题解决不是第一次安装方案 ,而是第一次安装完,使用也正常,关电脑再次访问的时候提示找不到 ,是因为本地服务没有启动  ,wi ...