hdu 1540 线段树
这题的意思是现在有一些村庄成一条直线排列,现在有三个操作,D:摧毁一个指定的村庄,Q:询问与指定村庄相连的村庄个数,
就是这个村庄向左和向右数村庄数量,遇到尽头或损坏的村庄为止,这个就是与这个村庄相连的村庄数量,当然,如果指定村庄已经被摧毁,则数量为0。R:把最后摧毁的那一个村庄恢复。在这里D操作和R操作都可以看成对线段树的单点修改,只是执行的操作不同,
最主要的是如何找和指定村庄相连的村庄数量,刚刚做的时候我很纠结,看了别人博客后... 我们可以定义一个max1和一个min1,表示这个区间里的最大值和最小值(max1初始化为-1,min1初始化为无穷大),这样一开始就是没有左右边界,如果某个村庄被摧毁,则把他的max1和min1都变成他的节点值,(自己体会一下下)
如果我们想找a村庄的的相连村庄数目,惯性思维是想一次性就把左右两个边界求出来,后来才发现我们其实可以用两次区间查询来分别找到a的左边界和右边界,query1(1,a,1)和query2(a,n,1)就是找区间1到a的的最大值和区间a到n的最小值,然后存一下这两个边界,最后相减。
具体可以看看代码:
#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<stack>
using namespace std;
#define inf 0x3f3f3f
struct node{
int l,r,min1,max1;
}str[*+];
stack<int>s;
int n,m,k,t,a,b;
int start,end1;
char ss[];
void build(int l,int r,int k)
{
str[k].l=l;
str[k].r=r;
str[k].max1=-;
str[k].min1=inf;
if(l==r)
{
return;
}
int mid=(l+r)/;
build(l,mid,k*);
build(mid+,r,k*+);
str[k].max1=max(str[k*].max1,str[k*+].max1);
str[k].min1=min(str[k*].min1,str[k*+].min1);
}
void change_point1(int k)//摧毁a点
{
if(str[k].l==str[k].r&&str[k].l==a)
{
str[k].max1=str[k].l;
str[k].min1=str[k].l;
return;
}
int mid=(str[k].l+str[k].r)/;
if(a<=mid)
change_point1(k*);
else
change_point1(k*+);
str[k].max1=max(str[k*].max1,str[k*+].max1);
str[k].min1=min(str[k*].min1,str[k*+].min1);
}
void change_point2(int k)//恢复a点
{
if(str[k].l==str[k].r&&str[k].l==a)
{
str[k].max1=-;
str[k].min1=inf;
return;
}
int mid=(str[k].l+str[k].r)/;
if(a<=mid)
change_point2(k*);
else
change_point2(k*+);
str[k].max1=max(str[k*].max1,str[k*+].max1);
str[k].min1=min(str[k*].min1,str[k*+].min1);
}
void query1(int l,int r,int k)//找a点的左边界
{
if(str[k].l>=l&&str[k].r<=r)
{
start=max(str[k].max1,start);
return;
}
int mid=(str[k].l+str[k].r)/;
if(l<=mid)
query1(l,r,k*);
if(r>mid)
query1(l,r,k*+);
}
void query2(int l,int r,int k)//找a点的右边界
{
if(str[k].l>=l&&str[k].r<=r)
{
end1=min(end1,str[k].min1);
return;
}
int mid=(str[k].l+str[k].r)/;
if(l<=mid)
query2(l,r,k*);
if(r>mid)
query2(l,r,k*+);
}
int main()
{
while(scanf("%d%d",&n,&m)!=EOF)
{
build(,n,);
while(!s.empty())
s.pop();
for(int i=;i<m;i++)
{
scanf("%s",ss);
if(ss[]=='D')
{
scanf("%d",&a);
change_point1();
s.push(a);
}
else if(ss[]=='Q')
{
scanf("%d",&a);
start=-;
end1=inf;
query1(,a,);
query2(a,n,);
if(start==-)//特判一下
start=;
if(end1==inf)
end1=n+;
int ans=end1-start-; if(start==end1)
ans=;
printf("%d\n",ans);
}
else if(ss[]=='R')
{
if(!s.empty())
{
a=s.top();
change_point2();
s.pop();
}
}
}
}
return ;
}
hdu 1540 线段树的更多相关文章
- HDU - 1540 线段树的合并
这个题题意我大概解释一下,就是一开始一条直线,上面的点全是联通的,有三种操作 1.操作D把从左往右第x个村庄摧毁,然后断开两边的联通. 2.询问Q节点相联通的最长长度 3.把最后破坏的村庄重建. 这个 ...
- HDU 1540<线段树,区间并>
题目连接 参考 题意: 维护各个点的连续的最大连续长度. 思路: 主要是维护一个区间的三个变量ll,f[i].l为起点向右的最大连续 长度,rl:f[i].r为起点向左的最大连续长度,ml:[l,r] ...
- I - Tunnel Warfare HDU - 1540 线段树最大连续区间
题意 :一段区间 操作1 切断点 操作2 恢复最近切断的一个点 操作3 单点查询该点所在最大连续区间 思路: 主要是push_up : 设区间x 为母区间 x<<1 ,x< ...
- hdu 5877 线段树(2016 ACM/ICPC Asia Regional Dalian Online)
Weak Pair Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 262144/262144 K (Java/Others)Total ...
- hdu 3974 线段树 将树弄到区间上
Assign the task Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- hdu 3436 线段树 一顿操作
Queue-jumpers Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) To ...
- hdu 3397 线段树双标记
Sequence operation Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
- hdu 4578 线段树(标记处理)
Transformation Time Limit: 15000/8000 MS (Java/Others) Memory Limit: 65535/65536 K (Java/Others) ...
- hdu 4533 线段树(问题转化+)
威威猫系列故事——晒被子 Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 65535/32768 K (Java/Others) Tot ...
随机推荐
- Kali 局域网 DNS 劫持
<一> 所需工具 1: Kali-linux-2017 2: ettercap 0.8.2 3: web 服务器, 这里以 node 为例 <二> 原理 1: DNS劫持 ...
- JS 相等判断 / 类型判断
相等判断 JavaScript提供三种不同的值比较操作: 严格相等 ("triple equals" 或 "identity"),使用 === , 宽松相等 ( ...
- ISO7816之管脚定义
卡座的管脚定义 如果使用示波器或者逻辑分析仪来观察 连接C3.C5.C7 小技巧当C3为3.57MHZ时候,可以使用波特率为9600的串口来监听.
- 深度学习原理与框架-Tensorflow卷积神经网络-神经网络mnist分类
使用tensorflow构造神经网络用来进行mnist数据集的分类 相比与上一节讲到的逻辑回归,神经网络比逻辑回归多了隐藏层,同时在每一个线性变化后添加了relu作为激活函数, 神经网络使用的损失值为 ...
- 轻量级Java持久化框架,Hibernate完美助手,Minidao 1.6.2版本发布
Minidao 1.6.2 版本发布,轻量级Java持久化框架(Hibernate完美助手) Minidao产生初衷? 采用Hibernate的J2EE项目都有一个痛病,针对复杂业务SQL,hiber ...
- Django - session 会话跟踪技术
1.session简介 |session 英 /'seʃ(ə)n/ 美 /'sɛʃən/ 基于cookies开发,将值存到服务端 写session 读session Session是服务器端技术,利用 ...
- Java——如何创建文件夹及文件,删除文件,文件夹
package com.zz; import java.io.File; import java.io.IOException; /** * Java创建文件夹 */ public class Cre ...
- class 方法
实例对象调用class方法时返回这个实例对象的isa指针,也就是对应的类对象: 类对象调用class方法时返回这个类对象本身. (注:如果想一直获得一个类的类对象,也就是isa指针,可以调用runti ...
- Masonry 动画
比如想做一个最简单的位移动画: 关键点在,改完约束后,调用下面这段代码,父view调用 layoutIfNeeded [UIView animateWithDuration:0.5 animation ...
- java学习书单
1 程序员必读书单 1.0 https://blog.csdn.net/onlylove_longshao/article/details/52337865 2 程序员读书雷达 ht ...