【HDU - 4348】To the moon(主席树在线区间更新)
BUPT2017 wintertraining(15) #8G
题意
给一个数组a,有n个数,m次操作。\(N, M ≤ 10^5, |A i| ≤ 10^9, 1 ≤ l ≤ r ≤ N, |d| ≤ 10^4\)
操作有四种,
C l r d:更新区间[l,r],值都加上d,并且时间前进1
Q l r:查询[l,r]的区间和
H l r t:查询t时刻的区间和
B t:回到t时刻(一定是历史时刻),且t时刻之后的操作都作废了。
题解
这道题的关键是怎么更新区间和。
本来线段树更新区间和要push down也就是把懒惰标记下移,但是这样,整个修改区间都需要新建线段树的节点,空间不够。
所以考虑在查询的时候传参数下去,也就是一路加上大区间的lazy值,这样就不用push down了。
代码
#include <cstdio>
#include <cstring>
#define N 100005
#define M (N*25)
typedef long long ll;
namespace PST{
int T[N],lson[M],rson[M],tot,n;
ll sum[M], lz[M];
void init(int _n){
tot=0;
n=_n;
memset(sum,0,sizeof sum);
memset(lz,0,sizeof lz);
}
void push_up(int root,int l,int r){
sum[root]=sum[lson[root]]+sum[rson[root]]+lz[root]*(r-l+1);
}
int build(int l,int r){
int root=++tot;
if(l==r){
scanf("%lld",sum+tot);
return tot;
}
int mid=l+r>>1;
lson[root]=build(l,mid);
rson[root]=build(mid+1,r);
push_up(root,l,r);
return root;
}
int update(int root,int l,int r,int L,int R,int val){
int newroot=++tot;
lson[newroot]=lson[root];
rson[newroot]=rson[root];
lz[newroot]=lz[root];
if(L<=l&&r<=R){
sum[newroot]=sum[root]+(ll)val*(r-l+1);
lz[newroot]+=val;
return newroot;
}
int mid=l+r>>1;
if(L<=mid)
lson[newroot]=update(lson[root],l,mid,L,R,val);
if(R>mid)
rson[newroot]=update(rson[root],mid+1,r,L,R,val);
push_up(newroot,l,r);
return newroot;
}
ll query(int root,int l,int r,int L,int R,ll add){
if(L<=l&&r<=R){
return sum[root]+(add)*(r-l+1);
}
int mid=l+r>>1;
ll ans=0;
if(L<=mid)
ans+=query(lson[root],l,mid,L,R,lz[root]+add);
if(R>mid)
ans+=query(rson[root],mid+1,r,L,R,lz[root]+add);
return ans;
}
}
int main(){
int n,m;
while(~scanf("%d%d",&n,&m)){
int now;
PST::init(n);
PST::T[now=0]=PST::build(1,n);
for(int i=0;i<m;++i){
char op;
int l,r,t,d;
scanf(" %c",&op);
if(op=='Q'){
scanf("%d%d",&l,&r);
printf("%lld\n",PST::query(PST::T[now],1,n,l,r,PST::lz[PST::T[now]]));
}else if(op=='H'){
scanf("%d%d%d",&l,&r,&t);
printf("%lld\n",PST::query(PST::T[t],1,n,l,r,PST::lz[PST::T[t]]));
}else if(op=='B'){
scanf("%d", &now);
PST::tot=PST::T[now+1]-1;
}else if(op=='C'){
scanf("%d%d%d",&l,&r,&d);
int tmp=now;
PST::T[++now]=PST::update(PST::T[tmp],1,n,l,r,d);
}
}
}
}
【HDU - 4348】To the moon(主席树在线区间更新)的更多相关文章
- HDU 4348 To the moon 主席树 在线更新
http://acm.hdu.edu.cn/showproblem.php?pid=4348 以前做的主席树没有做过在线修改的题做一下(主席树这种东西正经用法难道不是在线修改吗),标记永久化比较方便. ...
- hdu 4348 To the moon (主席树)
版权声明:本文为博主原创文章,未经博主允许不得转载. hdu 4348 题意: 一个长度为n的数组,4种操作 : (1)C l r d:区间[l,r]中的数都加1,同时当前的时间戳加1 . (2)Q ...
- hdu 4348 To the moon (主席树区间更新)
传送门 题意: 一个长度为n的数组,4种操作 : (1)C l r d:区间[l,r]中的数都加1,同时当前的时间戳加1 . (2)Q l r:查询当前时间戳区间[l,r]中所有数的和 . (3)H ...
- HDU 4348 To the moon 主席树
题意: 给出一个长度为\(n(n \leq 10^5)\)的序列,最开始时间\(t=0\),支持下面几个操作: \(C \, l \, r \, d\):将区间\([l,r]\)每个数都加上\(d\) ...
- hdu 4348 To the moon 主席树区间更新
To the moon Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) Prob ...
- HDU 4417 Super Mario(主席树求区间内的区间查询+离散化)
Super Mario Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Tota ...
- HDU 2665 Kth number(主席树静态区间第K大)题解
题意:问你区间第k大是谁 思路:主席树就是可持久化线段树,他是由多个历史版本的权值线段树(不是普通线段树)组成的. 具体可以看q学姐的B站视频 代码: #include<cmath> #i ...
- hdu 3397 Sequence operation(线段树:区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3397 题意:给你一个长度为n的0,1序列,支持下列五种操作, 操作0(0 a b):将a到b这个区间的 ...
- HDU4348To the moon主席树,区间修改
题意: 一个长度为n的数组,4种操作 : (1)C l r d:区间[l,r]中的数都加1,同时当前的时间戳加1 . (2)Q l r:查询当前时间戳区间[l,r]中所有数的和 . (3)H l r ...
随机推荐
- (第十三周)Final阶段用户调查报告
项目名:食物链教学工具 组名:奋斗吧兄弟 组长:黄兴 组员:李俞寰.杜桥.栾骄阳.王东涵 用户调查报告 调查时间:2016年12月1日 21:00——2016年12月3日 12:00 项目分享链接 ...
- MySQL和Oracle的区别
由于SQL Server不常用,所以这里只针对MySQL数据库和Oracle数据库的区别 (1) 对事务的提交 MySQL默认是自动提交,而Oracle默认不自动提交,需要用户手动提交,需要在写 ...
- sql定时备份
老规矩,直接上代码: ) set @name='C:\Backup\MyStudy_'+ ),)+'.bak' BACKUP DATABASE[MyStudy]TO DISK=@name WITH N ...
- ElasticSearch 入门
http://www.oschina.net/translate/elasticsearch-getting-started?cmp ElasticSearch 简单入门 返回原文英文原文:Getti ...
- ES5与ES6的小差异
ES5与ES6的小差异 变量的定义 ES6与ES5的区别 ES5: <script> console.log(username); var username; var username = ...
- 接口工具之postman
在我们日常开发中,经常会对功能接口进行相应的测试.那么postman是一款不错的测试工具,因为平常使用的比较多,因此在这里简单记录一下,经常使用到的一些地方 简单的使用就不错介绍了, 基本流程: 新建 ...
- JS实用小函数 数据是否合法或存在 获取当前日期时间
1.判断数据是否合法或存在 //判断数据是否合法或存在 function isNotNull(data) { if(data === "" || data === undefine ...
- .Net批量插入数据
1. 一般我们普通数据插入是这样的: 现在我们写一个控制台程序用常规办法添加10000条数据. //以下是批量插入数据的办法 //连接字符串 string str = "Server=.;D ...
- python之路--初识函数
一 . 函数 什么是函数 f(x) = x + 1 y = x + 1 # 函数是对功能或者动作的封装 函数的语法 def 函数名(): 函数体 调用: 函数名() def play(): print ...
- 【转】解决Maxwell发送Kafka消息数据倾斜问题
最近用Maxwell解析MySQL的Binlog,发送到Kafka进行处理,测试的时候发现一个问题,就是Kafka的Offset严重倾斜,三个partition,其中一个的offset已经快200万了 ...