CodeVs1082 线段树练习 3

题解:一种特殊的树状数组写法

http://codevs.cn/problem/1082/

#include <bits/stdc++.h>
#define ll long long
const int MAXN=2e5+10;
using namespace std;
ll read(){
    ll x=0,f=1;char ch=getchar();
    while(!isdigit(ch)){if(ch==‘-‘)f=-1;ch=getchar();}
    while(isdigit(ch))x=x*10+ch-‘0‘,ch=getchar();
    return f*x;
}
ll a[MAXN],b[MAXN];
int get_id(int x){return x&(-x);}
int n;
void add1(int x,ll vul){
    if(x<=0)return ;
    for(int i=x;i<=n;i+=get_id(i))a[i]+=vul;
}
void add2(int x,ll vul){
    if(x<=0)return ;
    for(int i=x;i>0;i-=get_id(i))b[i]+=vul;
}
//ll ans;
ll Sum1(int x){
    if(x<=0)return 0;
    ll ans=0;
    for(int i=x;i>0;i-=get_id(i))ans+=a[i];
    return ans;
}
ll Sum2(int x){
    if(x<=0)return 0;
    ll ans=0;
    for(int i=x;i<=n;i+=get_id(i))ans+=b[i];
    return ans;
}
int main(){
    n=read();ll t;int p=n;n++;
    for(int i=1;i<=p;i++)t=read(),add1(i,1ll*i*t),add2(i,t),add1(i-1,1ll*(i-1)*-t),add2(i-1,t*-1);
  //  cout<<Sum1(1)<<" "<<Sum2(2)<<endl;
   // cout<<Sum1(3)+1ll*3*Sum2(3+1)-Sum1(2-1)-1ll*(2-1)*Sum2(2)<<endl;
    int q;q=read();
    int op,l,r;ll vul;
    for(int i=1;i<=q;i++){
	op=read();l=read();r=read();
	if(op==1){
	    vul=read();add1(r,1ll*r*vul);add2(r,vul);
	    add1(l-1,1ll*(l-1)*-vul);add2(l-1,vul*-1);
	}
	else{
	    printf("%lld\n",Sum1(r)+1ll*r*Sum2(r+1)-Sum1(l-1)-1ll*(l-1)*Sum2(l));
	}
    }
    return 0;
}

原文地址:https://www.cnblogs.com/wang9897/p/9348899.html

时间: 2024-10-13 23:55:25

CodeVs1082 线段树练习 3的相关文章

codevs1082 线段树练习3

线段树 蒟蒻年轻的时候照着hzwer神犇的写的,勿喷= = #include<cstdio> #include<bits/stdc++.h> #include<cstring> #include<cstdlib> #include<algorithm> #define R0(i,n) for(int i=0;i<n;++i) #define R1(i,n) for(int i=1;i<=n;++i) #define CLR(x,c)

【洛谷P3372】【模板】线段树 1

题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.求出某区间每一个数的和 输入输出格式 输入格式: 第一行包含两个整数N.M,分别表示该数列数字的个数和操作的总个数. 第二行包含N个用空格分隔的整数,其中第i个数字表示数列第i项的初始值. 接下来M行每行包含3或4个整数,表示一个操作,具体如下: 操作1: 格式:1 x y k 含义:将区间[x,y]内每个数加上k 操作2: 格式:2 x y 含义:输出区间[x,y]内每个数的和 输出格式: 输出包含若干行整

【codevs1082】线段树练习 3

题目描述 Description 给你N个数,有两种操作: 1:给区间[a,b]的所有数增加X 2:询问区间[a,b]的数的和. 输入描述 Input Description 第一行一个正整数n,接下来n行n个整数, 再接下来一个正整数Q,每行表示操作的个数, 如果第一个数是1,后接3个正整数, 表示在区间[a,b]内每个数增加X,如果是2, 表示操作2询问区间[a,b]的和是多少. pascal选手请不要使用readln读入 输出描述 Output Description 对于每个询问输出一行

线段树基础

关于线段树的原理学习,可以参看杨弋大牛的论文<线段树>以及刘汝佳老师的<算法竞赛入门经典(训练指南)>,代码风格学习hzwer或者notonlysuccess均可. 一.单点更新 最基础的线段树 题目:codevs1080 链接:http://codevs.cn/problem/1080/ 分析:最简单的线段树,单点更新,区间求和 1 #include<iostream> 2 #include<cstdio> 3 using namespace std; 4

线段树-进阶

上一次我们写的线段树已经可以解决区间查询.单点修改了!可喜可贺 那如果现在我们需要区间修改.区间查询呢? 一般有两种思路:lazytag和标记永久化,lazytag的使用面好像更广一些. 一.lazytag 比如我们现在要修改一个区间,我们可以像查询一样分成若干段,然后分别修改每一段. 那么问题来了,每一段要怎么修改?如果直接修改sum,询问就乱套了.如果暴力修改,最坏复杂度O(n),那还要线段树干嘛(╯‵□′)╯掀桌 那我们这么想,我们修改就不修改整个区间了,而是在区间上维护一个标记. 那我们

[poj2104]可持久化线段树入门题(主席树)

解题关键:离线求区间第k小,主席树的经典裸题: 对主席树的理解:主席树维护的是一段序列中某个数字出现的次数,所以需要预先离散化,最好使用vector的erase和unique函数,很方便:如果求整段序列的第k小,我们会想到离散化二分和线段树的做法, 而主席树只是保存了序列的前缀和,排序之后,对序列的前缀分别做线段树,具有差分的性质,因此可以求任意区间的第k小,如果主席树维护索引,只需要求出某个数字在主席树中的位置,即为sort之后v中的索引:若要求第k大,建树时反向排序即可 1 #include

【BZOJ4942】[Noi2017]整数 线段树+DFS(卡过)

[BZOJ4942][Noi2017]整数 题目描述去uoj 题解:如果只有加法,那么直接暴力即可...(因为1的数量最多nlogn个) 先考虑加法,比较显然的做法就是将A二进制分解成log位,然后依次更新这log位,如果最高位依然有进位,那么找到最高位后面的第一个0,将中间的所有1变成0,那个0变成1.这个显然要用到线段树,但是复杂度是nlog2n的,肯定过不去. 于是我在考场上yy了一下,这log位是连续的,我们每次都要花费log的时间去修改一个岂不是很浪费?我们可以先在线段树上找到这段区间

bzoj1798: [Ahoi2009]Seq 维护序列seq 线段树

题目传送门 这道题就是线段树 先传乘法标记再传加法 #include<cstdio> #include<cstring> #include<algorithm> #define LL long long using namespace std; const int M=400010; LL read(){ LL ans=0,f=1,c=getchar(); while(c<'0'||c>'9'){if(c=='-') f=-1; c=getchar();}

Vijos P1066 弱弱的战壕【多解,线段树,暴力,树状数组】

弱弱的战壕 描述 永恒和mx正在玩一个即时战略游戏,名字嘛~~~~~~恕本人记性不好,忘了-_-b. mx在他的基地附近建立了n个战壕,每个战壕都是一个独立的作战单位,射程可以达到无限(“mx不赢定了?!?”永恒[email protected][email protected]). 但是,战壕有一个弱点,就是只能攻击它的左下方,说白了就是横纵坐标都不大于它的点(mx:“我的战壕为什么这么菜”ToT).这样,永恒就可以从别的地方进攻摧毁战壕,从而消灭mx的部队. 战壕都有一个保护范围,同它的攻击