D. Array Restoration

题目链接:https://codeforces.com/contest/1023/problem/D

题意:

给出一个序列,现在要求对一个全为0的序列执行q次操作,每次操作都要选定一段区间然后将区间上面的值变为i(i为操作的次数)。最终使得0序列变为之前给出的序列。

原序列中如果存在0,那么说明这个值是任意的。

最后要求输出经过q次操作之后的序列。

题解:

我们首先可以想到不可行的情况:在原序列中若存在一个数a,假设其出现次数大于1,那么两端为a的中间区间部分,是没有值比a小的。

还有一种情况也不可行,就是原序列中没有0且最大值小于q。

当把这两种情况排除过后,就是可行的情况了,之后输出方案就好了。

输出方案的时候还要解决0的问题,我的解决方案是让0等于其右边或左边的数,我是将一段连续的0缩点后进行处理的。

代码如下:

#include <bits/stdc++.h>
#define pii pair<int,int>
#define INF 99999999
using namespace std; const int N = 2e5+;
int n,q;
int a[N],cnt[N];
struct node{
int l,r;
}z[N],All[N];
int f[N][];
vector <int> vec[N];
int main(){
cin>>n>>q;
int num = ,mx=;
a[]=-;a[n+]=-;
for(int i=;i<=n;i++){
scanf("%d",&a[i]);
mx=max(mx,a[i]);
if(a[i]==&&a[i-]!=){
num++;
}
if(a[i]==){
cnt[i]=num;
if(a[i-]!=) z[num].l=i-;
}else if(a[i-]==) z[num].r=i;
vec[a[i]].push_back(i);
}
if(a[n]==) z[num].r=n+;
int flag = ;
for(int i=;i<=n;i++){
if(a[i]== || vec[a[i]].size()<=) continue ;
int len = vec[a[i]].size()-;
All[a[i]].l=vec[a[i]][];
All[a[i]].r=vec[a[i]][len];
}
for(int i=;i<=n;i++) for(int j=;j<=;j++) f[i][j]=INF;
for(int i=;i<=n;i++) f[i][]=(a[i]== ? INF : a[i]);
for(int i=;i<=n;i++)
for(int j=;j<=;j++)
f[i][j]=min(f[i][j-],f[i+(1ll<<(j-))][j-]);
for(int i=;i<=n;i++){
if(vec[a[i]].size()<=) continue ;
int l = All[a[i]].l,r = All[a[i]].r;
l++;r--;
int K = log((r-l+));
int mn = min(f[l][K],f[r-(<<K)+][K]);
if(mn<a[i]){
flag=;
break ;
}
}
int ok=;
if(q>mx) ok=;
if(ok) for(int i=;i<=n;i++){
if(a[i]==){
a[i]=q;
ok=;
break ;
}
}
if(flag||ok==) cout<<"NO";
else{
cout<<"YES"<<endl;
for(int i=;i<=n;i++){
if(a[i]!=){
cout<<a[i]<<" ";
}
else{
int l = z[cnt[i]].l,r = z[cnt[i]].r;
if(l< && r>n) cout<<q<<" ";
else if(l<) cout<<a[r]<<" ";
else cout<<a[l]<<" ";
}
}
}
return ;
}

之后看了下标程,用的是并查集orz...

标程的想法就比较巧妙了,首先把之前说的第二种不可行情况判断一下,然后开始骚操作。

其基本思想为:值从大到小进行区间修改的操作,对于当前位置的数而言,用并查集来维护i之后小于或等于它的第一个位置

因为我们是从大到小开始涂色,对于第一次涂色,肯定能找到第一个小于或等于它的位置;对于第i次涂色,假设当前位置为pos,如果pos+1的值比pos的值大,那么它之前已经操作过了,说明已经用并查集处理过了,那么f(pos+1)就是比pos+1的值小的第一个位置;如果此时找到的值还比pos的值大,那么就继续寻找;如果比pos的值小,那么就不可行。

注意上面查找值时要满足值的存在区间范围,并且为了输出方案,我们还要对区间进行修改,这时只用对0进行修改就好了~比它大的都不用管。

具体代码如下:

#include <bits/stdc++.h>
using namespace std; const int N =2e5+;
int n,q;
int f[N],l[N],r[N];
int a[N]; int find(int x){
return f[x]==x ? f[x] : f[x]=find(f[x]);
} int main(){
cin>>n>>q;
for(int i=;i<=q;i++) l[i]=n+,r[i]=;
for(int i=;i<=n+;i++) f[i]=i;
for(int i=;i<=n;i++){
scanf("%d",&a[i]);
l[a[i]]=min(l[a[i]],i);
r[a[i]]=max(r[a[i]],i);
}
if(l[q]>r[q]){
if(l[]>r[]){
cout<<"NO";
return ;
}
a[l[]]=q;
f[l[]]=l[]+;
}
for(int i=q;i>=;i--){
for(int j=find(l[i]);j<=r[i];j=find(j)){
if(a[j] && a[j]<i){
cout<<"NO";
return ;
}
a[j]=i;
f[j]=find(j+);
}
}
puts("YES");
for(int i=;i<=n;i++) printf("%d ",a[i]?a[i]:);
return ;
}

Codeforces Round #504:D. Array Restoration的更多相关文章

  1. Codeforces Round #504 D. Array Restoration

    Codeforces Round #504 D. Array Restoration 题目描述:有一个长度为\(n\)的序列\(a\),有\(q\)次操作,第\(i\)次选择一个区间,将区间里的数全部 ...

  2. Codeforces Round #504 E. Down or Right

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

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

  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. [Educational Codeforces Round 63 ] D. Beautiful Array (思维+DP)

    Educational Codeforces Round 63 (Rated for Div. 2) D. Beautiful Array time limit per test 2 seconds ...

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

  7. Educational Codeforces Round 11 A. Co-prime Array 水题

    A. Co-prime Array 题目连接: http://www.codeforces.com/contest/660/problem/A Description You are given an ...

  8. Educational Codeforces Round 23 D. Imbalanced Array 单调栈

    D. Imbalanced Array time limit per test 2 seconds memory limit per test 256 megabytes input standard ...

  9. Educational Codeforces Round 11——A. Co-prime Array(map+vector)

    A. Co-prime Array time limit per test 1 second memory limit per test 256 megabytes input standard in ...

随机推荐

  1. Servlet学习笔记02——什么是http协议?

    1.http协议 (了解) (1)什么是http协议? 是一种网络应用层协议,规定了浏览器与web服务器之间 如何通信以及相应的数据包的结构. 注: a.tcp/ip: 网络层协议,可以保证数据可靠的 ...

  2. thymeleaf单选回显,多选回显,选回显,下拉默认选中第一个

    //默认选中第一个<input type ="radio" name="repaymentType" th:each ="repaymentTy ...

  3. 安装破解IDEA(个人使用)

    安装的过程,许多的教程都会有,我在这里附上一两个链接吧:https://blog.csdn.net/newabcc/article/details/80601933 他这里也有破解过程,但是比较麻烦, ...

  4. Android和IOS网页不一致汇总

    1.input type=text 内容输入框的不一致,ios会默认给输入框添加自己的样式,导致在横向的输入框长度精准控制的时候,ios的输入框一般都比android上要长一点,还有内部阴影 解决此问 ...

  5. JSON后台处理特殊字符方法,在JSONArray.fromObject转换时处理

    /** * 替换一个字符串中的某些指定字符 * @param strData String 原始字符串 * @param regex String 要替换的字符串 * @param replaceme ...

  6. Android开发——弹性滑动的两种实现方式

    0. 前言   欢迎转载,转载请注明出处:http://blog.csdn.net/seu_calvin/article/details/52538723 我们在Android开发--View滑动的三 ...

  7. viewpager 无网络的时候滑动异常

    不知道大家有没有遇到过这种情况,就是框架是viewpager+fragment的架构.然后呢,fragment里面是webview.一般情况下,当没有网的时候,webviwe会说什么找不到网页,然后很 ...

  8. 程序第一次启动推送跳转,handleOpenURL没法跳转的原因

    iOS 程序启动时总会调用application:didFinishLaunchingWithOptions:,其中第二个参数launchOptions为NSDictionary类型的对象,里面存储有 ...

  9. 【tmux环境配置】在centos6.4上配置tmux

    我学习tmux的动力如下: (1)tmux大法好.原因是被同学安利过tmux. (2)多个terminal下ssh到开发机太麻烦.还是之前实习的时候,总要开N个terminal去ssh开发机,这种东西 ...

  10. U盘的容量变小了怎么办?

    之前买了个U盘,后来给朋友装系统弄成U盘启动盘了,就发现U盘容量变少了几百兆,原来是因为做U盘启动盘的时候,U盘启动盘制作软件都是把写入U盘的PE文件隐藏了,防止用户不小心删除文件. 所以说这些空间应 ...