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的更多相关文章

  1. Codeforces Round #504 E. Down or Right

    Codeforces Round #504 E. Down or Right 题目描述:交互题. 有一个\(n \times n\)的方阵,有一些格子是障碍,从\((1, 1)\)出发,只能向右向下走 ...

  2. 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 ...

  3. Codeforces Round #504:D. Array Restoration

    D. Array Restoration 题目链接:https://codeforces.com/contest/1023/problem/D 题意: 给出一个序列,现在要求对一个全为0的序列执行q次 ...

  4. 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的话就进行填充. ...

  5. 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 ...

  6. 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 ...

  7. 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 ...

  8. 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 ...

  9. Educational Codeforces Round 52D(ARRAY,模拟最短路)

    #include<bits/stdc++.h>using namespace std;int n,x;int chess[17*17];//记录棋盘上的numberarray<int ...

随机推荐

  1. Linux相关——记录gdb基本操作(持续更新)

    -----------2018.9.26更新标记----------- gdb的确是个很强大的东西啊,这里记录一下gdb的基本操作吧 后续可能会补充,但暂时感觉够用了就不写多了. 首先是ubuntu终 ...

  2. Java应用中使用ShutdownHook友好地清理现场

    在线上Java程序中经常遇到进程程挂掉,一些状态没有正确的保存下来,这时候就需要在JVM关掉的时候执行一些清理现场的代码.Java中得ShutdownHook提供了比较好的方案. JDK在1.3之后提 ...

  3. 【转】64位Ubuntu 16.04搭建嵌入式交叉编译环境arm-linux-gcc过程图解

    64位Ubuntu 16.04搭建嵌入式交叉编译环境arm-linux-gcc过程图解,开发裸机环境之前需要先搭建其开发环境,毕竟工欲善其事必先利其器嘛.  安装步骤 1.准备工具安装目录 下载 ar ...

  4. Find Common Characters - LeetCode

    目录 题目链接 注意点 解法 小结 题目链接 Find Common Characters - LeetCode 注意点 不能单纯的以字母出现的次数来判断是否是公共的字母 解法 解法一:将第一个字符串 ...

  5. SpringMVC 使用@ResponseBody返回json 中文乱码

    这确实是个蛋疼的问题,Spring中解析字符串的转换器默认编码居然是ISO-8859-1 既然找到问题了,那就必须想办法改过来,不同版本的Spring好像方法还不一样,网上不少说的都是Spring3. ...

  6. 【费用流】【网络流24题】【P4013】 数字梯形问题

    Description 给定一个由 \(n\) 行数字组成的数字梯形如下图所示. 梯形的第一行有 \(m\) 个数字.从梯形的顶部的 \(m\) 个数字开始,在每个数字处可以沿左下或右下方向移动,形成 ...

  7. Python中__init__.py文件的作用详解

    转自http://www.jb51.net/article/92863.htm Python中__init__.py文件的作用详解 http://www.jb51.net/article/86580. ...

  8. bashttpd使用手册

    http://note.youdao.com/noteshare?id=15775dca9fcdc7326e80158082572ed5

  9. Java入门:一些初学者需要掌握的基础算法程序——二分查找

    本例演示如何通过二分算法查找一个链表中的指定元素. import java.util.Scanner; class BinarySearchExample { public static void m ...

  10. 利用oneproxy实现mysql读写分离搭建笔记

      功能: 1.具有SQL白名单(防SQL注入)及IP白名单功能的SQL防火墙软件 2.数据库故障切换 3.读写分离 4.分库分表     一.下载 官网下载:http://www.onexsoft. ...