HDU 4267 A Simple Problem with Integers(2012年长春网络赛A 多颗线段树+单点查询)
以前似乎做过类似的不过当时完全不会。现在看到就有点思路了,开始还有洋洋得意得觉得自己有不小的进步了,结果思路错了。。。改了很久后测试数据过了还果断爆空间。。。
给你一串数字A,然后是两种操作:
"1 l r k c":意思是当 l=<i<=r 对(i-a)%k = =0 的每个 Ai 都增加 c (1=<k<=10)
"2 i" :意思是求出 Ai
一看就是区间更新和单点查询,其实可以用树状数组做,可是觉得线段树好弄一点,结果成功入坑。。。
我们可以发现k特别的小,而对于每一位的k,都有k个不同的余数,所以可以从这儿入手。可以看出对于每一个k,难点在于区间更新的时候并不是一定严格+1的连续区间,但是一定是+k连续区间,所以:
k=1,建一棵从1开始每次+1的树
k=2,建一棵从1开始每次+2的树 建一颗从2开始每次+2的树
k=3,建一棵从1开始每次+3的树 建一颗从2开始每次+3的树 建一棵从3开始每次+3的树
......
建立55棵线段树
但是如果就直接建立55颗线段树,再建55个对应的更新树,则会爆空间。不过我们可以看是单点查询,根本不需要用父节点记录孩子节点的和,建立线段树仅仅是为了区间更新。所以就可以直接模拟更新树,每个节点记录是此区间每个位置需要增加的值,求值的时候下更新到叶子节点就可以了。注意因为输入的l不是一定对应每棵树的l位置(不是每次都+1),所以我们要处理l,还有就是[l,r]之间我们仅仅更新一些点,右端点要处理好。最后查询的时候要查询10棵树
本以为对线段树有些心得了,可是对于有一点变化的东西都不能灵活运用,还需努力了
#include<set>
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<vector>
#include<string>
#include<cstdio>
#include<cstring>
#include<stdlib.h>
#include<iostream>
#include<algorithm>
using namespace std;
#define eps 1E-8
/*注意可能会有输出-0.000*/
#define Sgn(x) (x<-eps? -1 :x<eps? 0:1)//x为两个浮点数差的比较,注意返回整型
#define Cvs(x) (x > 0.0 ? x+eps : x-eps)//浮点数转化
#define zero(x) (((x)>0?(x):-(x))<eps)//判断是否等于0
#define mul(a,b) (a<<b)
#define dir(a,b) (a>>b)
typedef long long ll;
typedef unsigned long long ull;
const int Inf=<<;
const double Pi=acos(-1.0);
const int Max=<<;
int segtr[Max][];//建立55棵线段树 因为是单点查询,所以每次查询到孩子节点,父节点就只需要记录孩子增加了多少,所以线段树节点就模拟更新的树就好
int per[Max],pos[][];//记录初始值 记录节点在树的位置
void Create(int sta,int enn,int now)
{
memset(segtr[now],,sizeof(segtr[now]));
if(sta==enn)
{
scanf("%d",&per[sta]);
return;
}
int mid=dir(sta+enn,);
int next=mul(now,);
Create(sta,mid,next);
Create(mid+,enn,next|);
return;
}
void Downow(int now,int next,int k)//区间更新的关键
{
if(segtr[now][k])//相当于区间更新
{
segtr[next][k]+=segtr[now][k];
segtr[next|][k]+=segtr[now][k];
segtr[now][k]=;
}
return;
}
void Update(int sta,int enn,int now,int x,int y,int k,int add)
{
if(sta>=x&&enn<=y)
{
segtr[now][k]+=add;
return;
}
int mid=dir(sta+enn,);
int next=mul(now,);
Downow(now,next,k);//只需要下更新
if(mid>=x)
Update(sta,mid,next,x,y,k,add);
if(mid<y)
Update(mid+,enn,next|,x,y,k,add);
return;
}
int Query(int sta,int enn,int now,int x,int k)
{
if(sta==enn)
{
return segtr[now][k];
}
int mid=dir(sta+enn,);
int next=mul(now,);
Downow(now,next,k);//只需要下更新
if(mid>=x)
return Query(sta,mid,next,x,k);
else
return Query(mid+,enn,next|,x,k);
}
int main()
{
int n,q,coun=;
for(int i=;i<;i++)
for(int j=;j<=i;j++)
pos[i][j]=coun++;
while(~scanf("%d",&n))
{
Create(,n,);
scanf("%d",&q);
int typ,lef,rig,k;
int add;
while(q--)
{
scanf("%d",&typ);
if(typ==)
{
add=;
scanf("%d",&lef);
for(int i=; i<; i++)//查询时需要查询10棵树
add+=Query(,n,,(lef+i)/(i+),pos[i][(lef-)%(i+)]);//相同大小在每棵树的位置不一样,注意
add+=per[lef];
printf("%d\n",add);
}
else
{
scanf("%d %d %d %d",&lef,&rig,&k,&add);//只是更新一棵树就好
Update(,n,,(lef+k-)/k,(rig-lef)/k+(lef+k-)/k,pos[k-][(lef-)%k],add);//注意更新的只是输入的左右区间内的一部分
}
}
}
return ;
}
HDU 4267 A Simple Problem with Integers(2012年长春网络赛A 多颗线段树+单点查询)的更多相关文章
- HDU 4267 A Simple Problem with Integers
A Simple Problem with Integers Time Limit: 5000/1500 MS (Java/Others) Memory Limit: 32768/32768 K ...
- HDU 4267 A Simple Problem with Integers(树状数组区间更新)
A Simple Problem with Integers Time Limit: 5000/1500 MS (Java/Others) Memory Limit: 32768/32768 K ...
- 【树状数组区间修改单点查询+分组】HDU 4267 A Simple Problem with Integers
http://acm.hdu.edu.cn/showproblem.php?pid=4267 [思路] 树状数组的区间修改:在区间[a, b]内更新+x就在a的位置+x. 然后在b+1的位置-x 树状 ...
- HDU 4267 A Simple Problem with Integers --树状数组
题意:给一个序列,操作1:给区间[a,b]中(i-a)%k==0的位置 i 的值都加上val 操作2:查询 i 位置的值 解法:树状数组记录更新值. 由 (i-a)%k == 0 得知 i%k == ...
- 2012年长春网络赛(hdu命题)
为迎接9月14号hdu命题的长春网络赛 ACM弱校的弱菜,苦逼的在机房(感谢有你)呻吟几声: 1.对于本次网络赛,本校一共6名正式队员,训练靠的是完全的自主学习意识 2.对于网络赛的群殴模式,想竞争现 ...
- HDOJ 4267 A Simple Problem with Integers (线段树)
题目: Problem Description Let A1, A2, ... , AN be N elements. You need to deal with two kinds of opera ...
- HDU 4747 Mex (2013杭州网络赛1010题,线段树)
Mex Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others)Total Submis ...
- HDU 3974 Assign the task(DFS序+线段树单点查询,区间修改)
描述There is a company that has N employees(numbered from 1 to N),every employee in the company has a ...
- 【HDOJ】4267 A Simple Problem with Integers
树状数组.Easy. /* 4267 */ #include <iostream> #include <string> #include <map> #includ ...
随机推荐
- OpenCV 3.1
http://www.tuicool.com/articles/FRfMni2 http://docs.opencv.org/3.1.0/d7/d9f/tutorial_linux_install.h ...
- 运用Arc Hydro提取河网
Arc hydro 插件需要 spatial analyst 支持: 解决方法:Tools菜单>>Extensions...,勾选Spatial Analyst 1.设置存储路径 ApUt ...
- EasyNetQ操作RabbitMQ(高级消息队列)
RabbitMQ是实现了高级消息队列协议(AMQP)的开源消息代理软件(亦称面向消息的中间件).写消息队列的时候用RabbitMQ比较好,但是写的时候需要自己封装下,自己的封装,就需要对RabbitM ...
- freemarker 展示数据列表并传值给后台
select id="initiatorId" name="initiatorId"> <#if initiato ...
- struts2 拦截器,使用spring注入
ActionContext actionContext = invocation.getInvocationContext();ServletContext context = (ServletCon ...
- NPTL LinuxThreads
Linux 线程模型的比较:LinuxThreads 和 NPTL 进行移植的开发人员需要了解的关键区别摘要 Vikram Shukla 2006 年 8 月 28 日发布 WeiboGoogle+用 ...
- ios错误ignoring file xxx missing required architecture x86_64 in file
错误ignoring file xxx missing required architecture x86_64 in file 解决方法: 1.在Project target里“Architectu ...
- MongoDB可视化工具 Studio 3T
告别终端使用可视化工具Studio 3T对MongoDB进行数据库的操作. 简单的使用步骤介绍 1.启动MongoDB服务器(方法见MongoDB介绍与安装中的介绍) 2.连接MongoDB服务器 ...
- 自定义列表数据自动循环向下滚动view(类似于通知通报消息)
首先申明,这个自定义View不是本人写的,大神写的,本人仅限学习一级研究使用 直接上代码吧!后面我再解释一遍 package com.egojit.android.gcoa.views; import ...
- uitableview滚动到最后一行
本文转载至 http://mrjeye.iteye.com/blog/1278521 - (void)scrollTableToFoot:(BOOL)animated { NSInteger s = ...