传送门

题意:

  庭院中有 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"(线段树区间更新+双端队列)的更多相关文章

  1. ZOJ 1610 Count the Color(线段树区间更新)

    描述Painting some colored segments on a line, some previously painted segments may be covered by some ...

  2. HDU 1556 Color the ball(线段树区间更新)

    Color the ball 我真的该认真的复习一下以前没懂的知识了,今天看了一下线段树,以前只会用模板,现在看懂了之后,发现还有这么多巧妙的地方,好厉害啊 所以就应该尽量搞懂 弄明白每个知识点 [题 ...

  3. hihoCoder 1080 : 更为复杂的买卖房屋姿势 线段树区间更新

    #1080 : 更为复杂的买卖房屋姿势 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho都是游戏迷,“模拟都市”是他们非常喜欢的一个游戏,在这个游戏里面他们 ...

  4. HDU 5023 A Corrupt Mayor's Performance Art(线段树区间更新)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5023 解题报告:一面墙长度为n,有N个单元,每个单元编号从1到n,墙的初始的颜色是2,一共有30种颜色 ...

  5. HDU 4902 Nice boat 2014杭电多校训练赛第四场F题(线段树区间更新)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4902 解题报告:输入一个序列,然后有q次操作,操作有两种,第一种是把区间 (l,r) 变成x,第二种是 ...

  6. HDU 1698 线段树 区间更新求和

    一开始这条链子全都是1 #include<stdio.h> #include<string.h> #include<algorithm> #include<m ...

  7. POJ-2528 Mayor's posters (线段树区间更新+离散化)

    题目分析:线段树区间更新+离散化 代码如下: # include<iostream> # include<cstdio> # include<queue> # in ...

  8. ZOJ 1610 Count the Colors (线段树区间更新)

    题目链接 题意 : 一根木棍,长8000,然后分别在不同的区间涂上不同的颜色,问你最后能够看到多少颜色,然后每个颜色有多少段,颜色大小从头到尾输出. 思路 :线段树区间更新一下,然后标记一下,最后从头 ...

  9. POJ 2528 Mayor's posters (线段树区间更新+离散化)

    题目链接:http://poj.org/problem?id=2528 给你n块木板,每块木板有起始和终点,按顺序放置,问最终能看到几块木板. 很明显的线段树区间更新问题,每次放置木板就更新区间里的值 ...

随机推荐

  1. HashMap、HashTable、ConcurrentHashMap、HashSet区别 线程安全类

    HashMap专题:HashMap的实现原理--链表散列 HashTable专题:Hashtable数据存储结构-遍历规则,Hash类型的复杂度为啥都是O(1)-源码分析 Hash,Tree数据结构时 ...

  2. 使用synchronized 实现ReentrantLock(美团面试题目)

    刚看到这个题目的时候无从下手,因为觉得synchronized和lock在加锁的方式上有很大不同,比如,看看正常情况下synchronized时如何加锁的. 方式一: public synchroni ...

  3. pycharm 安装

    pycharm 1.模板 file->setting->Editor->file and code template->python script->右上方 #!/usr ...

  4. 十分钟了结MySQL information_schema

    information_schema数据库是MySQL系统自带的数据库,它提供了数据库元数据的访问方式.感觉information_schema就像是MySQL实例的一个百科全书,记录了数据库当中大部 ...

  5. PHPStorm 配置命名空间

    文件-设置-Directories 选中:application     点击顶部:Sources,右侧会出现 Source Floders 配置项 点击:p进行设置 输入app\

  6. Nginx 返回响应过滤响应内容

    陶辉94课 过滤模块 从下到上顺序 ngx_http_proxy_module 模块 Syntax: proxy_ignore_headers field ...; Default: — Contex ...

  7. BZOJ2281[Sdoi2011]黑白棋&BZOJ4550小奇的博弈——DP+nimk游戏

    题目描述 小A和小B又想到了一个新的游戏. 这个游戏是在一个1*n的棋盘上进行的,棋盘上有k个棋子,一半是黑色,一半是白色. 最左边是白色棋子,最右边是黑色棋子,相邻的棋子颜色不同. 小A可以移动白色 ...

  8. 一个服务器创建两个MySQL

    转载:http://www.2cto.com/database/201412/357863.html 将已安装的数据库文件夹复制到另一个目录下 打开复制目录下的my.ini文件修改 [client] ...

  9. Goldbach's Conjecture POJ - 2262 线性欧拉筛水题 哥德巴赫猜想

    题意 哥德巴赫猜想:任一大于2的数都可以分为两个质数之和 给一个n 分成两个质数之和 线行筛打表即可 可以拿一个数组当桶标记一下a[i]  i这个数是不是素数  在线性筛后面加个装桶循环即可 #inc ...

  10. opencv 图片缩放

    import cv2 as cv import numpy as np # 图片缩放 img = cv.imread('../images/moon.jpg', flags=1) # flags=1读 ...