Codeforces Round #504 D. Array Restoration
Codeforces Round #504 D. Array Restoration
题目描述:有一个长度为\(n\)的序列\(a\),有\(q\)次操作,第\(i\)次选择一个区间,将区间里的数全部改为\(i\),序列\(a\)的每个位置至少被改一次。得到最终的序列,然后将序列里的某些位置变成\(0\),输出一种可能的置零之前的最终序列,或无解。
solution
求出每种数字最长的染色区间,按这个区间染色,记下没出现的数字。染色后如果存在\(0\)联通块,则用没出现的数字从大到小染色(一个联通块一种颜色),不够则无解,最后判断最大的数字\(q\)是否用上了,如果没有,则在原序列中找一个\(0\)变成\(q\)即可,若原序列没有\(0\),则无解。
时间复杂度:\(O(nlogn)\)
#include <bits/stdc++.h>
using namespace std;
const int maxn=int(2e5)+100;
struct mes
{
int x, y, v;
};
int n, m;
int a[maxn], ans[maxn];
int BIT[maxn];
int tree[maxn*4], mark[maxn*4];
bool vis[maxn];
vector<int> zero;
pair<int, int> block[maxn];
mes dat;
void read()
{
scanf("%d%d", &n, &m);
for (int i=1; i<=n; ++i) scanf("%d", &a[i]);
}
void insert(int cur, int v)
{
for (; cur<=m; cur+=(-cur)&cur) BIT[cur]=max(BIT[cur], v);
}
int ask(int cur)
{
int w=0;
for (; cur; cur-=(-cur)&cur) w=max(w, BIT[cur]);
return w;
}
void down(int cur)
{
tree[cur<<1]=tree[cur<<1 | 1]=mark[cur];
mark[cur<<1]=mark[cur<<1 | 1]=mark[cur];
mark[cur]=0;
}
void updata(int cur, int L, int R)
{
if (dat.x>R || dat.y<L) return;
if (dat.x<=L && R<=dat.y)
{
tree[cur]=dat.v;
mark[cur]=dat.v;
return;
}
int mid=(L+R)>>1;
if (mark[cur]) down(cur);
updata(cur<<1, L, mid);
updata(cur<<1 | 1, mid+1, R);
}
int get(int cur, int L, int R)
{
if (L==R) return tree[cur];
int mid=(L+R)>>1;
if (mark[cur]) down(cur);
if (dat.x<=mid) return get(cur<<1, L, mid);
else return get(cur<<1 | 1, mid+1, R);
}
void solve()
{
for (int i=1; i<=n; ++i)
{
if (a[i]==0) continue;
if (!vis[a[i]]) block[a[i]].first=ask(a[i]-1);
vis[a[i]]=true;
insert(a[i], i);
}
for (int i=1; i<=m; ++i) BIT[i]=0, vis[i]=false;
for (int i=n; i; --i)
{
if (a[i]==0) continue;
if (!vis[a[i]]) block[a[i]].second=n-ask(a[i]-1)+1;
vis[a[i]]=true;
insert(a[i], n-i+1);
}
for (int i=1; i<=m; ++i)
if (!vis[i]) zero.push_back(i);
else
{
dat.x=block[i].first+1;
dat.y=block[i].second-1;
dat.v=i;
updata(1, 1, n);
}
for (int i=1; i<=n; ++i)
{
dat.x=dat.y=i;
ans[i]=get(1, 1, n);
}
for (int i=1; i<=n; ++i)
if (a[i]!=0 && a[i]!=ans[i])
{
printf("NO\n");
return;
}
for (int i=1; i<=n; )
{
if (ans[i]!=0) { ++i; continue; }
if (zero.size()==0)
{
puts("NO");
return;
}
int j=i;
while (j<=n && a[j]==0) ++j;
for (int k=i; k<j; ++k) ans[k]=zero.back();
zero.pop_back();
i=j;
}
if (!zero.empty() && zero.back()==m)
{
int idx=0;
for (int i=1; i<=n; ++i)
if (a[i]==0)
{
idx=i;
break;
}
if (idx==0)
{
puts("NO");
return;
}
ans[idx]=m;
}
puts("YES");
for (int i=1; i<=n; ++i) printf("%d ", ans[i]);
}
int main()
{
read();
solve();
return 0;
}
Codeforces Round #504 D. Array Restoration的更多相关文章
- Codeforces Round #504 E. Down or Right
Codeforces Round #504 E. Down or Right 题目描述:交互题. 有一个\(n \times n\)的方阵,有一些格子是障碍,从\((1, 1)\)出发,只能向右向下走 ...
- Codeforces 1023 A.Single Wildcard Pattern Matching-匹配字符 (Codeforces Round #504 (rated, Div. 1 + Div. 2, based on VK Cup 2018 Fi)
Codeforces Round #504 (rated, Div. 1 + Div. 2, based on VK Cup 2018 Final) A. Single Wildcard Patter ...
- Codeforces Round #504:D. Array Restoration
D. Array Restoration 题目链接:https://codeforces.com/contest/1023/problem/D 题意: 给出一个序列,现在要求对一个全为0的序列执行q次 ...
- Codeforces 1023 D.Array Restoration-RMQ(ST)区间查询最值 (Codeforces Round #504 (rated, Div. 1 + Div. 2, based on VK Cup 2018 Fi)
D. Array Restoration 这题想一下就会发现是只要两个相同的数之间没有比它小的就可以,就是保存一下数第一次出现和最后一次出现的位置,然后查询一下这个区间就可以,如果有0的话就进行填充. ...
- Codeforces Round #504 (rated, Div. 1 + Div. 2, based on VK Cup 2018 Final)-D- Array Restoration
我们知道不满足的肯定是两边大中间小的,这样就用RMQ查询两个相同等值的区间内部最小值即可,注意边界条件 #include<bits/stdc++.h> #define x first #d ...
- Educational Codeforces Round 21 D.Array Division(二分)
D. Array Division time limit per test:2 seconds memory limit per test:256 megabytes input:standard i ...
- Educational Codeforces Round 11A. Co-prime Array 数学
地址:http://codeforces.com/contest/660/problem/A 题目: A. Co-prime Array time limit per test 1 second me ...
- E - Down or Right Codeforces Round #504 (rated, Div. 1 + Div. 2, based on VK Cup 2018 Final)
http://codeforces.com/contest/1023/problem/E 交互题 #include <cstdio> #include <cstdlib> #i ...
- Educational Codeforces Round 52D(ARRAY,模拟最短路)
#include<bits/stdc++.h>using namespace std;int n,x;int chess[17*17];//记录棋盘上的numberarray<int ...
随机推荐
- 简明Python教程自学笔记——命令行通讯录
[前言]学习Python已经有一段时间了,相关的书籍资料也下载了不少,但是没有一本完整的看完,也没有编出一个完整的程序.今天下午比较清闲就把<简明Python教程>看了一遍,然后根据书里面 ...
- Trailing Zeroes (I) LightOJ - 1028(求因子个数)
题意: 给出一个N 求N有多少个别的进制的数有后导零 解析: 对于一个别的进制的数要转化为10进制 (我们暂且只分析二进制就好啦) An * 2^(n-1) + An-1 * 2^(n-2) + `` ...
- #define后面只带有一个标识符
经常看到有#define后只有一个标识符的语句,这样是做宏开关用 宏定义编译前会被编译器进行替换,只有一个标识符的情况,如果在代码里使用了这个标识符,会被替换为空,也就是相当于没加. 用来做编译开关的 ...
- Android Studio快捷键设置之实现原eclipse中ctrl+m的全屏的效果
如下图,keymap设置成eclipse的的,但是鼠标双击最大化复原没有,ctr+M全屏也没有, 那么久在如下图的三个项目上添加鼠标双击和快捷键---自己方便好用,但不冲突的 Toggle full ...
- 【bzoj4264】小C找朋友
题解 $a$和$b$是好*友说明除了这两个人以外的邻接集合相同: 做两次$hash$,分别都处理和$a$相邻的点排序$hash$,①$a$要算进$a$的相邻集合,②$a$不算进: 当两个人不是好*友, ...
- 图像处理之色彩转换(CCM)
1 色彩校正原理 人眼对色彩的识别,是基于人眼对光谱存在三种不同的感应单元,不同的感应单元对不同波段的光有不同的响应曲线的原理,通过大脑的合成得到色彩的感知. 一般来说,我们可以通俗的用 RGB三基 ...
- Java入门:绘制简单图形
在上一节,我们学习了如何使用swing和awt工具创建一个空的窗口,本节学习如何绘制简单图形. 基本绘图介绍 Java中绘制基本图形,可以使用Java类库中的Graphics类,此类位于java.aw ...
- nginx访问日志出现大量的500状态信息,用postman返回 Internal Server Error,Too Many Attempts.错误的解决办法
用postman的post方法访问某个URL时,出现以下错误: { "status": "1", "message": " Int ...
- 为什么 “return s and s.strip()” 在用 filter 去掉空白字符时好使?
如题: 给定一个数组,其中该数组中的每个元素都为字符串,删除该数组中的空白字符串. _list = ["A", "", "", " ...
- Windows上虚拟环境的安装及使用
Why Install VirtualEnv? 可以方便的解决不同项目对类库的依赖问题. 可以在系统中Python解释器中避免包的混乱和版本的冲突. 为每个程序单独创建虚拟环境,可以保证程序只能访 ...