HDU 4831 Scenic Popularity (段树)
Scenic Popularity
假设全部景点区和歇息区都是X轴直线上的一系列顶点,所相应的坐标Xi 保证唯一。每一个景点区有个初始的热度值,而一个歇息区(坐标为Xi)的热度值等于离它距离近期的景点区Xj的热度值(距离定义为|Xi-Xj|)。假设此歇息区与两个景点区的距离一样。则歇息区的热度值选择两个景点区中的热度值最大值。假设两个热度值都一样,则任意选择当中一个。
度度熊在出门之前会常常去查看百度地图,每次查看前会有某些景点区的热度值已发生改变,从而也会导致周围的歇息区的热度值发生改变,然后度度熊想知道当前热度值<=Rk的顶点(包含景点区和歇息区)有多少个
每一个Case的第一行是N(0<N<=10000),表示景点区和歇息区的总数。
接着会有N行数据,每一列首先是顶点的X坐标Xi (0< Xi <=1e8),第二列是一个整数Hi(0=<Hi <=100000),假设Hi 不为0。则表示当前顶点为风景区且初始的热度值为Hi,否则表示当前顶点为歇息区。这N行数据会依照坐标Xi递增的方式依次给出。
接着的一行数据是操作的次数K(K<=100),最后会有K行数据,每一行的第一列要么是’U’或者’Q’,’U’表示当前操作为更改热度操作。’Q’表示当前操作为查询操作。假设是更改操作,接着会有两列数据,各自是热度值要改变的风景区的下标Lk(0<=Lk<N)以及改变后的热度值Vk(0< Vk<=100000);假设是查询操作,第二列是要查询的热度范围Rk(0< Rk<=100000)
1
4
10 0
20 3
30 0
40 2
3
Q 3
U 3 4
Q 3
Case #1:
4
2
题目大意:
T组測试数据,每组首先一个n表示 旅游区和歇息区 共同拥有 n个。
接下来是n行介绍,每行两个数字,第一个表示位置。第二个表示热度,假设热度为0表示歇息区。否则为景区。歇息区的热度与近期景区的热度同样,假设(左边和右边)两个景区距离同样。歇息区的热度取热度高的。
经接着m,表示m组询问,Q x 。表示 热度小于等于x的有多少个。
U 表示改变某个景区的热度。
解题思路:
对于Q操作的询问,非常明显是维护的一个前缀和,因此想到用线段树维。可是U操作非常复杂。要维护差量计算。
解题代码:
#include <iostream>
#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <algorithm>
using namespace std; const int maxn=110000;
int pos[maxn],value[maxn],lson[maxn],rson[maxn],n; struct node{
int l,r,sum;
}a[maxn*4]; void build(int l,int r,int k){
a[k].l=l;
a[k].r=r;
a[k].sum=0;
if(l<r){
int mid=(l+r)/2;
build(l,mid,2*k);
build(mid+1,r,2*k+1);
}
} void insert(int l,int r,int k,int c){
//if(k==1) cout<<"insert:"<<l<<" "<<r<<" "<<c<<" "<<endl;
if(l<=a[k].l && a[k].r<=r){
a[k].sum+=c;
}else{
int mid=(a[k].l+a[k].r)/2;
if(r<=mid) insert(l,r,2*k,c);
else if(l>=mid+1) insert(l,r,2*k+1,c);
else{
insert(l,mid,2*k,c);
insert(mid+1,r,2*k+1,c);
}
a[k].sum=a[2*k].sum+a[2*k+1].sum;
}
} int query(int l,int r,int k){
if(l<=a[k].l && a[k].r<=r){
return a[k].sum;
}else{
int mid=(a[k].l+a[k].r)/2;
if(r<=mid) return query(l,r,2*k);
else if(l>=mid+1) return query(l,r,2*k+1);
else return query(l,mid,2*k)+query(mid+1,r,2*k+1);
}
} void findlson(int k){
if(k<=0) lson[k]=-1;
else{
if(value[k-1]==0) lson[k]=lson[k-1];
else lson[k]=k-1;
}
} void findrson(int k){
if(k>=n-1) rson[k]=-1;
else{
if(value[k+1]==0) rson[k]=rson[k+1];
else rson[k]=k+1;
}
} void input(){
build(0,110000,1);
scanf("%d",&n);
for(int i=0;i<n;i++){
scanf("%d%d",&pos[i],&value[i]);
lson[i]=-1;
rson[i]=-1;
}
for(int i=0;i<n;i++){
if(value[i]==0) findlson(i);
}
for(int i=n-1;i>=0;i--){
if(value[i]==0) findrson(i);
}
for(int i=0;i<n;i++){
if(value[i]==0){
if( lson[i]==-1 && rson[i]==-1 ) insert(0,0,1,1);
if( lson[i]!=-1 && rson[i]==-1 ) insert(value[lson[i]],value[lson[i]],1,1);
if( lson[i]==-1 && rson[i]!=-1 ) insert(value[rson[i]],value[rson[i]],1,1);
if( lson[i]!=-1 && rson[i]!=-1 ){
int l0=lson[i],r0=rson[i];
if(pos[i]-pos[l0] <pos[r0]-pos[i] ) insert(value[l0],value[l0],1,1);
else if(pos[i]-pos[l0] > pos[r0]-pos[i] ) insert(value[r0],value[r0],1,1);
else{
insert(max(value[r0],value[l0]),max(value[r0],value[l0]),1,1);
}
}
}else{
insert(value[i],value[i],1,1);
}
}
} void solve(){
int m;
scanf("%d",&m);
while(m-- >0){
char ch;
cin>>ch;
if(ch=='Q'){
int c;
scanf("%d",&c);
cout<<query(0,c,1)<<endl;
}else{
int k,c;
scanf("%d%d",&k,&c);
insert(c,c,1,1);
insert(value[k],value[k],1,-1);
if(value[k]!=c){
for(int i=k-1;i>=0;i--){
if(value[i]!=0) continue;
if(rson[i]!=k) break;
if(lson[i]==-1){
insert(c,c,1,1);
insert(value[k],value[k],1,-1);
}else{
int l0=lson[i],r0=rson[i];
if(pos[i]-pos[l0] > pos[r0]-pos[i] ){
insert(c,c,1,1);
insert(value[k],value[k],1,-1);
}
if(pos[i]-pos[l0] == pos[r0]-pos[i] ){
insert( max(value[r0],value[l0]) , max(value[r0],value[l0]) ,1,-1);
insert( max(c,value[l0]) , max(c,value[l0]) ,1,1);
}
}
}
for(int i=k+1;i<n;i++){
if(value[i]!=0) continue;
if(lson[i]!=k) break;
if(rson[i]==-1){
insert(c,c,1,1);
insert(value[k],value[k],1,-1);
}else{
int l0=lson[i],r0=rson[i];
if(pos[i]-pos[l0] < pos[r0]-pos[i] ){
insert(c,c,1,1);
insert(value[k],value[k],1,-1);
}
if(pos[i]-pos[l0] == pos[r0]-pos[i] ){
insert( max(value[r0],value[l0]) , max(value[r0],value[l0]) ,1,-1);
insert( max(c,value[r0]) , max(c,value[r0]) ,1,1);
}
}
}
}
value[k]=c;
}
}
} int main(){
int t;
scanf("%d",&t);
for(int i=1;i<=t;i++){
input();
printf("Case #%d:\n",i);
solve();
}
return 0;
}
版权声明:本文博客原创文章。博客,未经同意,不得转载。
HDU 4831 Scenic Popularity (段树)的更多相关文章
- hdu 4831 Scenic Popularity(模拟)
pid=4831" style="font-weight:normal">题目链接:hdu 4831 Scenic Popularity 题目大意:略. 解题思路: ...
- HDU 4831 Scenic Popularity
Scenic Popularity Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others ...
- 2014百度之星初赛第二场hdu 4831 Scenic Popularity
Scenic Popularity Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others ...
- HDU 6315.Naive Operations-线段树(两棵树合并)(区间单点更新、区间最值、区间求和)+思维 (2018 Multi-University Training Contest 2 1007)
6315.Naive Operations 题意很好理解,但是因为区间求和求的是向下取整的a[i]/b[i],所以直接分数更新区间是不对的,所以反过来直接当a[i]==b[i]的时候,线段树对应的位置 ...
- HDU ACM 4578 Transformation->段树-间隔的变化
分析:复杂的经营分部树. 只有一个查询操作,这是要求[l,r]的数量之间p钍总和.并不是所有的查询所有节点,会议TLE.最好的是查询部件[a.b].所有这个区间值我们是平等的,即能返回(b-a+1)* ...
- hdu 4107当卡段树
其核心思想是记录最大的节点值和最低值,假设max<p要么min>=p时间,在节点只变化add值,不要子树遍历:否则,就往子树递归. #include<iostream> #in ...
- HDU 1394 Minimum Inversion Number (数据结构-段树)
Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java ...
- HDU 6356.Glad You Came-线段树(区间更新+剪枝) (2018 Multi-University Training Contest 5 1007)
6356.Glad You Came 题意就是给你一个随机生成函数,然后从随机函数里确定查询的左右区间以及要更新的val值.然后最后求一下异或和就可以了. 线段树,区间最大值和最小值维护一下,因为数据 ...
- HDU 5649.DZY Loves Sorting-线段树+二分-当前第k个位置的数
DZY Loves Sorting Time Limit: 12000/6000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Oth ...
随机推荐
- 三——第二部分——第二篇论文 计划建设SQL Server镜像
本文接着前面的章节:SQL Server镜像简单介绍 本文出处:http://blog.csdn.net/dba_huangzj/article/details/27203053 俗话说:工欲善其事必 ...
- 直接选择排序----java实现
直接选择排序思路: 从待排序数据中选择第一个假定为最小的下标,然后他后面的与他循环比较,得到真的最小值下标,然后最小值前的那一区段依次后移,并把最小值赋值给第一个元素.第二次时,假定第二个为最小,然后 ...
- mysqlbackup 还原特定的表
mysqlbackup使用TTS恢复指定表. ************************************************************* 4.恢复特定表 ******* ...
- 【iOS发展-61】更换plist经过资源,执行iOS一旦数据仍显示在模拟器的外观,如何解决?
(1)案例介绍 --我们首先导入plist文件做项目,模拟的观看效果. --删除plist,更换一个新的plist,CMD+R模拟执行,或者找到该程序界面上显示最后一个数据. (2)原因 是由于第一次 ...
- FPGA 时序问题
近期 做一个项目------4个 1080p(1920 x 1080) 合成 一个 4K(3840 x 2160,297M)的接口板.当 1080p 进去, 1080p出来的时候,视频正常 播放出来. ...
- 迅雷云加速开放平台c#demo
迅雷云加速开放平台c#demo.很多人很遇到下载文件的问题.这个例子是调用迅雷云加速开放平台的dll,进行下载,速度很快,下载过程中可以获取到很全的下载信息,比如下载速度,进度,完成状态等. 例子中带 ...
- SQL Server管理员专用连接的使用
原文:SQL Server管理员专用连接的使用 作为一名DBA,经常会处理一些比较棘手的服务无响应问题,鉴于事态的严重性,多数DBA可能直接用“重启”大法,以便尽快的恢复生产环境的正常运转,但是多数情 ...
- Html5用Canvas制作画图板
需求: 绘制多边形 可填充颜色 可设置文字 可移动,可删除 鼠标按住后,抬起之前线段应该尾随鼠标当前位置 可与后台方便的进行数据交互,保存到后台,或将数据从后台取到前台显示对应的图形 思考: 第一想到 ...
- 【转】Android 4.3源码下载及问题解决
[html] view plaincopy 1 2 3 4 5 6 7 8 9 10 11 jianguoliao@jianguoliao-Lenovo-IdeaPad-Y470:~$ cat /et ...
- android大概是通过logcat拦截Log
我们必须在系统的环境变量先增加adb 路径: 在原有环境的后面增加;E:\Android\android-sdk-r16\platform-tools(;是不能缺少的) 然后我们在cmd中输入adb, ...