Gym 101911E "Painting the Fence"(线段树区间更新+双端队列)
题意:
庭院中有 n 个围栏,每个围栏上都被涂上了不同的颜色(数字表示);
有 m 条指令,每条指令给出一个整数 x ,你要做的就是将区间[ x第一次出现的位置 , x最后出现的位置 ]中的围栏
全部涂成 x ,经过 m 次操作后,输出每个围栏的涂色情况;
题解:
比赛的时,在读完题后,一瞬间,想到了线段树的区间更新,懒惰标记,but 我已经好久好久没写过线段树的代码了(嫌代码太长,逃);
所以,比赛时,就不了了之,去看其他题了;
今天,温习了一下线段树的用法,重新思考了本题的解题思路,感觉,线段树可以做出来;
然后,对着电脑撸了一个线段树版的代码,一发AC,哈哈,开森~~~~~~
手动艾特两个不相信线段树可以AC的童鞋,哈哈哈!
具体思路:
首先准备一个双端队列 q[ maxn ]; (maxn = 3e5+50 ,因为 ci ≤ 3*105 )
q[ i ] 中存储的是所有的被涂上颜色 i 的栅栏编号,按顺序从小到大存储;
对于每一个指令 x ,首先判断 q[x].size() 是否大于 1,如果大于 1,令 f = q[x].front() , e = q[x].end();
然后,判断下标 f 和 e 对应的栅栏的颜色是否为 x ,如果是,通过线段树的区间更新+懒惰标记将区间[f,e]染成 x,接着执行下一条指令;
如果 f 或 e 对应的栅栏颜色在之前被涂成了其他颜色,那么在 q[x] 中查找下一个满足条件的区间,如果找不到,不操作,执行下一条指令;
最后,调用线段树的查询操作,输出答案。
AC代码:
#include<iostream>
#include<cstdio>
#include<deque>
using namespace std;
#define ls(x) (x<<1)
#define rs(x) (x<<1|1)
const int maxn=3e5+; int n,m;
int a[maxn];
deque<int >q[maxn];
struct SegmentTree
{
int l,r;
int color;
int mid()
{
return l+((r-l)>>);
}
}segTree[*maxn]; void buildSegTree(int pos,int l,int r)
{
segTree[pos].l=l;
segTree[pos].r=r;
segTree[pos].color=;
if(l == r)
{
segTree[pos].color=a[l];
return ;
} int mid=l+((r-l)>>);
buildSegTree(ls(pos),l,mid);
buildSegTree(rs(pos),mid+,r);
} /**
向下更新,与区间更新懒惰标记不同的是
直接将pos的左右儿子的color赋值为x
*/
void pushDown(int pos)
{
int &x=segTree[pos].color;
if(x != )
{
segTree[ls(pos)].color=x;
segTree[rs(pos)].color=x;
x=;
}
}
/**
将区间[l,r]所包含的子区间的color值赋为x
出现的bug:起初直接将64行的赋值操作写成了
segTree[pos].color=a[l];
调试了好大会;
因为 a[l] 是改变前的涂色情况,如果l号栅栏在之前被染成了其他颜色
那么l号栅栏的颜色就不是a[l]了
*/
void Update(int pos,int l,int r,int x)
{
if(l <= segTree[pos].l && r >= segTree[pos].r)
{
segTree[pos].color=x;
return ;
}
pushDown(pos); int mid=segTree[pos].mid();
if(r <= mid)
Update(ls(pos),l,r,x);
else if(l > mid)
Update(rs(pos),l,r,x);
else
{
Update(ls(pos),l,mid,x);
Update(rs(pos),mid+,r,x);
}
}
int Query(int pos,int index)
{
if(segTree[pos].l == segTree[pos].r)
return segTree[pos].color; pushDown(pos); int mid=segTree[pos].mid();
if(index <= mid)
return Query(ls(pos),index);
else
return Query(rs(pos),index);
}
void Solve()
{
buildSegTree(,,n);
for(int i=;i <= m;++i)
{
int x;
scanf("%d",&x);
if(q[x].size() <= )
continue; /**
寻找可行的区间[f,e]
*/
int f=q[x].front();
int e=q[x].back();
while(Query(,f) != x && !q[x].empty())
{
q[x].pop_front();
if(!q[x].empty())
f=q[x].front();
}
while(Query(,e) != x && !q[x].empty())
{
q[x].pop_back();
if(!q[x].empty())
e=q[x].back();
}
//确保f != e
if(q[x].size() > )
Update(,f,e,x);
}
for(int i=;i <= n;++i)
printf("%d ",Query(,i));
printf("\n");
}
int main()
{
// freopen("C:/Users/hyacinthLJP/Desktop/stdin/contest","r",stdin);
scanf("%d",&n);
for(int i=;i <= n;++i)
{
scanf("%d",a+i);
q[a[i]].push_back(i);
}
scanf("%d",&m);
Solve(); return ;
}
Gym 101911E "Painting the Fence"(线段树区间更新+双端队列)的更多相关文章
- ZOJ 1610 Count the Color(线段树区间更新)
描述Painting some colored segments on a line, some previously painted segments may be covered by some ...
- HDU 1556 Color the ball(线段树区间更新)
Color the ball 我真的该认真的复习一下以前没懂的知识了,今天看了一下线段树,以前只会用模板,现在看懂了之后,发现还有这么多巧妙的地方,好厉害啊 所以就应该尽量搞懂 弄明白每个知识点 [题 ...
- hihoCoder 1080 : 更为复杂的买卖房屋姿势 线段树区间更新
#1080 : 更为复杂的买卖房屋姿势 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho都是游戏迷,“模拟都市”是他们非常喜欢的一个游戏,在这个游戏里面他们 ...
- HDU 5023 A Corrupt Mayor's Performance Art(线段树区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5023 解题报告:一面墙长度为n,有N个单元,每个单元编号从1到n,墙的初始的颜色是2,一共有30种颜色 ...
- HDU 4902 Nice boat 2014杭电多校训练赛第四场F题(线段树区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4902 解题报告:输入一个序列,然后有q次操作,操作有两种,第一种是把区间 (l,r) 变成x,第二种是 ...
- HDU 1698 线段树 区间更新求和
一开始这条链子全都是1 #include<stdio.h> #include<string.h> #include<algorithm> #include<m ...
- POJ-2528 Mayor's posters (线段树区间更新+离散化)
题目分析:线段树区间更新+离散化 代码如下: # include<iostream> # include<cstdio> # include<queue> # in ...
- ZOJ 1610 Count the Colors (线段树区间更新)
题目链接 题意 : 一根木棍,长8000,然后分别在不同的区间涂上不同的颜色,问你最后能够看到多少颜色,然后每个颜色有多少段,颜色大小从头到尾输出. 思路 :线段树区间更新一下,然后标记一下,最后从头 ...
- POJ 2528 Mayor's posters (线段树区间更新+离散化)
题目链接:http://poj.org/problem?id=2528 给你n块木板,每块木板有起始和终点,按顺序放置,问最终能看到几块木板. 很明显的线段树区间更新问题,每次放置木板就更新区间里的值 ...
随机推荐
- 在Hmtl页面中只让其中单独的一个div隐藏滚动条但是仍可滚动浏览下边的内容
<style> .box ::-webkit-scrollbar {width: 0px;} </style> <div class="box"> ...
- C#使用WebClient下载文件到本地目录
C#使用WebClient下载文件到本地目录. 1.配置本地目录路径 <appSettings> <!--文件下载目录--> <add key="Downloa ...
- GitHub大佬:供计算机学习鉴黄功能的图片数据库
ps:学无止境 想要构建一套鉴黄系统,必须有大量的真实图片供计算机进行学习,以便于区分开正常图片和黄色图片. 近期有位加拿大程序员在Github上传了图片列表,里面包含了大量图片地址可以供计算机进行学 ...
- HDU4625 JZPTREE 【树形DP】【第二类斯特林数】
题目大意: 对1到n求题目中描述的那个式子. 题目分析: 幂不好处理,转化为斯特林数. 根据$ n^k= \sum_ { i=0 }^k S(k,i)×i!×C(n,i) $. 我们可以将问题转化为对 ...
- Android短信大全
使用ListView实现点击条目跳转短信界面,并将内容传至短信页面: 代码如下:activity_main.xml: <?xml version="1.0" encoding ...
- GIL、进/线程池、同/异步、阻/非阻塞
1 GIL:全局解释器锁 GIL本质就是一把互斥锁,是夹在解释器身上的, 同一个进程内的所有线程都需要先抢到GIL锁,才能执行解释器代码 2.GIL的优缺点: 优点: 保证Cpython解释器内存管理 ...
- XML中添加换行符
网上说的说法大多是 可是我这里cuba框架用这个不是换行,下面这个才行 这也是转义字符, 为什么会是这样的?搞不懂,求大神告知! 另外 表示空格 表示回车
- Java实现一个双向链表的倒置功能
题目要求:Java实现一个双向链表的倒置功能(1->2->3 变成 3->2->1) 提交:代码.测试用例,希望可以写成一个Java小项目,可以看到单元测试部分 该题目的代码, ...
- IDEA调试技巧之条件断点
调试的时候,在循环里增加条件判断,可以极大的提高效率,心情也能愉悦.以下介绍下IDEA使用条件[Condition]断点的方法 1.编写一段样例代码 /** * @author jiashubing ...
- Java 异常处理流程
Java 异常处理流程 ============== End