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 ...
随机推荐
- MT【139】公比为有理数
已知正整数\(a_1,a_2,\cdots ,a_{2016}\)成等比数列,公比\(q\in (1,2)\),则\(a_{2016}\) 取最小值时,\(q=\)______ 解答: 显然\(q\) ...
- 洛谷P4902乘积
题面链接 洛谷 题意简述 求\(\prod_{i=A}^B\prod_{j=1}^i \lgroup \frac{i}{j} \rgroup ^{\lfloor \frac{i}{j} \rfloor ...
- 【BZOJ1864】三色二叉树(动态规划)
[BZOJ1864]三色二叉树(动态规划) 题面 BZOJ 题解 首先把树给构出来. 设\(f[i][0/1]\)表示当前节点\(i\),是否是绿色节点的子树中最大/最小的绿色节点的个数和. 转移很显 ...
- 安装linux系统后调优及安全设置
环境说明: [root@server1 ~]# cat /etc/redhat-release CentOS release 6.9 (Final) [root@server1 ~]# uname - ...
- Linux内核设计与实现第八周读书笔记
第四章 进程调度 进程在操作系统看来是程序的运行态表现形式. 4.1多任务 多任务操作系统就是能同时并发地交互执行多个进程的操作系统. 多任务操作系统会使多个进程处于堵塞或者睡眠状态.这些任务尽管位于 ...
- MySQL总结小妙招
mysql5.7版本,免登陆修改管理员密码: vim /etc/my.cnf 加入skip-grant-tables,重启MySQL 终端输入 mysql ,直接登录MySQL数据库,然后use my ...
- 解题: SDOI 2011 染色
题面 强行把序列问题通过树剖套在树上...算了算是回顾了一下树剖的思想=.= 每次树上跳的时候注意跳的同时维护当前拼出来的左右两条链的靠上的端点,然后拼起来的时候讨论一下拼接点,最后一下左右两边的端点 ...
- Kubernetes Downward API
目录 说明 环境变量方式 将pod信息注入为环境变量 将容器资源信息注入为环境变量 volume挂载方式 作用 说明 我们知道,每个Pod在成功创建出来之后,都会被系统分配唯一的名字.IP地址,并且处 ...
- BIO、NIO、AIO三者的比较
消息时的系统通信,通常基于网络协议实现.常见的协议包括TCP/IP,UDP/IP. TCP/IP等协议用于数据传输,但要完成通信,还需要对数据进行处理.例如读取和写入数据. I/O可以分为两种:同步I ...
- 互斥量、条件变量与pthread_cond_wait()函数的使用,详解(一)
1. 首先pthread_cond_wait 的定义是这样的 The pthread_cond_wait() and pthread_cond_timedwait() functions are us ...