区间更新求和

主要用来练习splay树区间更新问题

//splay树的题解

// File Name: 3468-splay.cpp
// Author: Zlbing
// Created Time: 2013年08月09日 星期五 16时30分32秒 #include<iostream>
#include<string>
#include<algorithm>
#include<cstdlib>
#include<cstdio>
#include<set>
#include<map>
#include<vector>
#include<cstring>
#include<stack>
#include<cmath>
#include<queue>
using namespace std;
#define CL(x,v); memset(x,v,sizeof(x));
#define INF 0x3f3f3f3f
#define LL long long
#define REP(i,r,n) for(int i=r;i<=n;i++)
#define RREP(i,n,r) for(int i=n;i>=r;i--) #define L ch[x][0]
#define R ch[x][1]
#define KT (ch[ch[rt][1]][0])
const int MAXN=2e5+;
struct SplayTree{
int ch[MAXN][];
int pre[MAXN],sz[MAXN],val[MAXN];
int rt,top;
void Rotate(int x,int f)
{
int y=pre[x];
down(y);down(x);
ch[y][!f]=ch[x][f];
pre[ch[x][f]]=y;
pre[x]=pre[y];
if(pre[x])
ch[pre[y]][ch[pre[y]][]==y]=x;
ch[x][f]=y;
pre[y]=x;
up(y);
}
void Splay(int x,int goal)
{
down(x);
while(pre[x]!=goal)
{
down(pre[pre[x]]);
down(pre[x]);
down(x);
if(pre[pre[x]]==goal)
Rotate(x,ch[pre[x]][]==x);
else
{
int y=pre[x],z=pre[y];
int f=(ch[z][]==y);
if(ch[y][f]==x)
Rotate(x,!f),Rotate(x,f);
else Rotate(y,f),Rotate(x,f);
}
}
up(x);
if(goal==)rt=x;
}
void RTO(int k,int goal)
{
int x=rt;
down(x);
while(sz[L]+!=k)
{
if(k<sz[L]+)x=L;
else
{
k-=(sz[L]+);
x=R;
}
down(x);
}
Splay(x,goal);
}
void vist(int x)
{
if(x)
{
printf("结点%2d : 左儿子 %2d 右儿子 %2d val: %2d sum=%lld\n",x,ch[x][],ch[x][],val[x],sum[x]);
vist(L);
vist(R);
}
}
void debug()
{
puts(""); vist(rt); puts("");
}
void up(int x)
{
sz[x]=+sz[L]+sz[R];
sum[x]=val[x]+sum[L]+sum[R];
}
void down(int x)
{
if(add[x])
{
val[L]+=add[x];
val[R]+=add[x];
add[L]+=add[x];
add[R]+=add[x];
sum[L]+=(LL)add[x]*sz[L];
sum[R]+=(LL)add[x]*sz[R];
add[x]=;
}
}
void Newnode(int &x,int c,int f)
{
x=++top;
L=R=; sz[x]=; pre[x]=f;
val[x]=sum[x]=c;
add[x]=;
}
void build(int &x,int l,int r,int f)
{
if(l>r)return;
int m=(l+r)>>;
Newnode(x,num[m],f);
build(L,l,m-,x);
build(R,m+,r,x);
up(x);
}
void init(int n)
{
ch[][]=ch[][]=pre[]=;
sz[]=rt=top=; add[]=sum[]=;
Newnode(rt,-,);
Newnode(ch[rt][],-,rt);
sz[rt]=;
for(int i=;i<=n;i++)
scanf("%d",&num[i]); build(KT,,n,ch[rt][]);
up(ch[rt][]);up(rt);
}
void update()
{
int l,r,c;
scanf("%d%d%d",&l,&r,&c);
RTO(l,);
RTO(r+,rt);
add[KT]+=c;
val[KT]+=c;
sum[KT]+=(LL)c*sz[KT];
}
void query()
{
int l,r;
scanf("%d%d",&l,&r);
RTO(l,);
RTO(r+,rt);
printf("%lld\n",sum[KT]);
}
LL sum[MAXN];
int add[MAXN];
int num[MAXN];
}spt;
int main()
{
int m,n;
char op[];
while(~scanf("%d%d",&n,&m))
{
spt.init(n);
while(m--)
{
scanf("%s",op);
if(op[]=='Q')spt.query();
else spt.update();
}
}
return ;
}

线段树的题解

// File Name: /home/neuacm/ACM/POJ/3468.cpp
// Author: Zlbing
// Created Time: 2013年08月09日 星期五 14时35分11秒 #include<iostream>
#include<string>
#include<algorithm>
#include<cstdlib>
#include<cstdio>
#include<set>
#include<map>
#include<vector>
#include<cstring>
#include<stack>
#include<cmath>
#include<queue>
using namespace std;
#define CL(x,v); memset(x,v,sizeof(x));
#define INF 0x3f3f3f3f
#define LL long long
#define REP(i,r,n) for(int i=r;i<=n;i++)
#define RREP(i,n,r) for(int i=n;i>=r;i--)
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
const int MAXN=1e5+;
LL col[MAXN<<];
LL sum[MAXN<<];
void pushup(int rt)
{
sum[rt]=sum[rt<<]+sum[rt<<|];
}
void pushdown(int rt,int l,int r)
{
if(col[rt])
{
col[rt<<]+=col[rt];
col[rt<<|]+=col[rt];
int m=(l+r)>>;
sum[rt<<]+=(m-l+)*col[rt];
sum[rt<<|]+=(r-m)*col[rt];
col[rt]=;
}
}
void build(int l,int r,int rt)
{
col[rt]=;
if(l==r)
{
//scanf("%I64d",&sum[rt]);
cin>>sum[rt];
return ;
}
int m=(l+r)>>;
build(lson);
build(rson);
pushup(rt);
}
LL query(int L,int R,int l,int r,int rt)
{
if(L<=l&&R>=r)
{
return sum[rt];
}
pushdown(rt,l,r);
int m=(l+r)>>;
LL ret=;
if(L<=m)ret+=query(L,R,lson);
if(R>m)ret+=query(L,R,rson);
return ret;
}
void update(int L,int R,int p,int l,int r,int rt)
{
if(L<=l&&R>=r)
{
col[rt]+=p;
sum[rt]+=(r-l+)*p;
return;
}
pushdown(rt,l,r);
int m=(l+r)>>;
if(L<=m)update(L,R,p,lson);
if(R>m)update(L,R,p,rson);
pushup(rt);
}
void debug(int l,int r,int rt)
{
if(l==r)
{
printf("l==%d r==%d rt=%d sum[rt]=%lld col[rt]=%lld\n",l,r,rt,sum[rt],col[rt]);
return ;
}
printf("l==%d r==%d rt=%d sum[rt]=%lld col[rt]=%lld\n",l,r,rt,sum[rt],col[rt]);
int m=(l+r)>>;
debug(lson);
debug(rson);
}
int main()
{
int n,m;
//std::ios::sync_with_stdio(false);
while(~scanf("%d%d",&n,&m))
{
build(,n,);
char ch[];
int a,b,c;
REP(i,,m)
{
scanf("%s",ch);
if(ch[]=='Q')
{
scanf("%d%d",&a,&b);
LL ans=query(a,b,,n,);
//debug(1,n,1);
cout<<ans<<endl;
}
else if(ch[]=='C')
{
scanf("%d%d%d",&a,&b,&c);
update(a,b,c,,n,);
//debug(1,n,1);
}
}
}
return ;
}

POJ-3468-A Simple Problem with Integers(区间更新,求和)-splay或线段树的更多相关文章

  1. POJ.3468 A Simple Problem with Integers(线段树 区间更新 区间查询)

    POJ.3468 A Simple Problem with Integers(线段树 区间更新 区间查询) 题意分析 注意一下懒惰标记,数据部分和更新时的数字都要是long long ,别的没什么大 ...

  2. poj 3468 A Simple Problem with Integers 【线段树-成段更新】

    题目:id=3468" target="_blank">poj 3468 A Simple Problem with Integers 题意:给出n个数.两种操作 ...

  3. 线段树(成段更新) POJ 3468 A Simple Problem with Integers

    题目传送门 /* 线段树-成段更新:裸题,成段增减,区间求和 注意:开long long:) */ #include <cstdio> #include <iostream> ...

  4. POJ 3468 A Simple Problem with Integers(线段树功能:区间加减区间求和)

    题目链接:http://poj.org/problem?id=3468 A Simple Problem with Integers Time Limit: 5000MS   Memory Limit ...

  5. poj 3468 A Simple Problem with Integers(线段树+区间更新+区间求和)

    题目链接:id=3468http://">http://poj.org/problem? id=3468 A Simple Problem with Integers Time Lim ...

  6. POJ 3468 A Simple Problem with Integers(线段树&区间更新)题解

    Description You have N integers, A1, A2, ... , AN. You need to deal with two kinds of operations. On ...

  7. POJ 3468 A Simple Problem with Integers(分块入门)

    题目链接:http://poj.org/problem?id=3468 A Simple Problem with Integers Time Limit: 5000MS   Memory Limit ...

  8. poj 3468 A Simple Problem with Integers 线段树区间加,区间查询和

    A Simple Problem with Integers Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://poj.org/problem?i ...

  9. poj 3468 A Simple Problem with Integers 线段树区间加,区间查询和(模板)

    A Simple Problem with Integers Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://poj.org/problem?i ...

  10. poj 3468:A Simple Problem with Integers(线段树,区间修改求和)

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

随机推荐

  1. windows 与Linux 互传文件

    下载putty,将putty的安装路径添加到Windows的环境变量中:     我的电脑->属性->高级->环境变量->系统变量,双击其中的Path,在分号后添加putty的 ...

  2. oracle 字符串切割成结果集方法

    oracle字符串切割几种方式 方法一: SELECT COLUMN_VALUE FROM TABLE(SYS.ODCIVARCHAR2LIST('1','2','3','4','5')); 方法二: ...

  3. Android Cursor类的概念和用法

    http://www.2cto.com/kf/201109/103163.html 关于 Cursor 在你理解和使用 Android Cursor 的时候你必须先知道关于 Cursor 的几件事情: ...

  4. c-八进制 转 十进制

    概述 其实x进制转十进制的算法都差不多,不过如果是针对于字符形式,他们却有点不同.使用指针和数组的形式计算,又不同.这里演示将字符型的数组形式的八进制转成十进制: #include <stdio ...

  5. c#读取文件

    你平时是怎么读取文件的?使用流读取.是的没错,C#给我们提供了非常强大的类库(又一次吹捧了.NET一番), 里面封装了几乎所有我们可以想到的和我们没有想到的类,流是读取文件的一般手段,那么你真的会用它 ...

  6. memcached和mongodb 在windows下安装

    要在新机器上安装memcached和mongodb服务,折腾了一天,终于把这两个服务在windows下跑起来了. memcached主要参考http://www.rootop.org/pages/27 ...

  7. runnable和thread的区别

    一是写一个类继承自Thread类,然后重写里面的run方法,用start方法启动线程二是写一个类实现Runnable接口,实现里面的run方法,用new Thread(Runnable target) ...

  8. 了解负载均衡 会话保持 session同步(转)

    一,什么负载均衡 一个新网站是不要做负载均衡的,因为访问量不大,流量也不大,所以没有必要搞这些东西.但是随着网站访问量和流量的快速增长,单台服务器受自身硬件条件的限制,很难承受这么大的访问量.在这种情 ...

  9. 虚拟机VHD格式解析到NTFS文件系统解析

    本来的需求是XEN下的镜像取证,但这篇仅包括他支持的一种格式,就是VHD,此项目从头开始大概用了两周时间,中间遇到了很多让人头大的问题,光是思考的笔记就写了十几页纸,不过实际上并没有那么难,主要是很久 ...

  10. 20款jquery下拉导航菜单特效代码分享

    20款jquery下拉导航菜单特效代码分享 jquery仿京东商城左侧分类导航下拉菜单代码 jQuery企业网站下拉导航菜单代码 jQuery css3黑色的多级导航菜单下拉列表代码 jquery响应 ...