【BZOJ 1046】 1046: [HAOI2007]上升序列
1046: [HAOI2007]上升序列
Description
对于一个给定的S={a1,a2,a3,…,an},若有P={ax1,ax2,ax3,…,axm},满足(x1 < x2 < … < xm)且( ax1 < ax
2 < … < axm)。那么就称P为S的一个上升序列。如果有多个P满足条件,那么我们想求字典序最小的那个。任务给
出S序列,给出若干询问。对于第i个询问,求出长度为Li的上升序列,如有多个,求出字典序最小的那个(即首先
x1最小,如果不唯一,再看x2最小……),如果不存在长度为Li的上升序列,则打印Impossible.Input
第一行一个N,表示序列一共有N个元素第二行N个数,为a1,a2,…,an 第三行一个M,表示询问次数。下面接M
行每行一个数L,表示要询问长度为L的上升序列。N<=10000,M<=1000Output
对于每个询问,如果对应的序列存在,则输出,否则打印Impossible.
Sample Input
6
3 4 1 2 3 6
3
6
4
5Sample Output
Impossible
1 2 3 6
ImpossibleHINT
Source
【分析】
好坑啊,字典序最小指的是下标字典序。。。
那么,我们首先求出以每个数为开头上升序列长度,就是倒着做最长下降子序列。
然后,遇到第一个f[i]>=x的,就输出,然后更新条件,继续在后面找,时间是每次询问是线性的。
==个人觉得数值字典序最小更难吧。。。我打了一个数值字典序最小的,不知道对不对。。
LIS做法:
d[i]表示长度为i的IS中,最后一个数字最小的是什么(因为最后一个数字最小最有利于后面的)
显然d数组是单调递增的。显然我们每次插一个数进去只会改变一个d值或者让LIS增加一位。
若a[i]>a[d[mx]] ,那么可以吧a[i]放在目前的LIS后面,那么mx+1。
否则,二分查找一个x,使得x满足a[i]>d[x]&&a[i]<=d[x+1],那么可以把a[i]放在长度为x的IS后面,替换长度为x+1的IS。
下降序列大于小于号反一下就好了。
本题code:
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define Maxn 10010
#define INF 0xfffffff int a[Maxn],b[Maxn],f[Maxn];
int lt[Maxn]; int mymax(int x,int y) {return x>y?x:y;} int n,mx; int tdiv(int x,int l,int r)
{
while(l<r)
{
int mid=(l+r+)>>;
if(x<a[lt[mid]]) l=mid;
else r=mid-;
}
return l;
} void ffind()
{
mx=;
memset(lt,,sizeof(lt));
a[]=INF;
for(int i=n;i>=;i--)
{
if(mx==||a[i]<a[lt[mx]])
{
mx++;
lt[mx]=i;
f[i]=mx;
}
else
{
int x=tdiv(a[i],,mx);
f[i]=x+;
lt[x+]=i;
}
}
} void output(int x)
{
int now=-INF;
for(int i=;i<=n;i++)
{
if(f[i]>=x&&a[i]>now)
{
if(now==-INF) printf("%d",a[i]);
else printf(" %d",a[i]);
now=a[i];
x--;
}
if(x==) break;
}
} int main()
{
freopen("a.in","r",stdin);
freopen("a.out","w",stdout);
scanf("%d",&n);
for(int i=;i<=n;i++) scanf("%d",&a[i]);
ffind();
int q;
scanf("%d",&q);
while(q--)
{
int x;
scanf("%d",&x);
if(x>mx) printf("Impossible\n");
else
{
output(x);
printf("\n");
}
}
return ;
}
数值最小(不知道对不对的)
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define Maxn 10010
#define INF 0xfffffff int a[Maxn],b[Maxn],f[Maxn];
int lt[Maxn],nt[Maxn]; int mymax(int x,int y) {return x>y?x:y;} int n,mx; int tdiv(int x,int l,int r)
{
while(l<r)
{
int mid=(l+r+)>>;
if(x>a[lt[mid]]) l=mid;
else r=mid-;
}
return l;
} void ffind()
{
mx=;
memset(nt,,sizeof(nt));
memset(lt,,sizeof(lt));
a[]=-INF;
for(int i=;i<=n;i++)
{
if(mx==||a[i]>a[lt[mx]])
{
mx++;
lt[mx]=i;
nt[i]=lt[mx-];
}
else
{
int x=tdiv(a[i],,mx);
lt[x+]=i;
nt[i]=lt[x];
}
}
} void output(int x)
{
if(x==) return;
output(nt[x]);
printf("%d ",a[x]);
} int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++) scanf("%d",&a[i]);
ffind();
int q;
scanf("%d",&q);
while(q--)
{
int x;
scanf("%d",&x);
if(x>mx) printf("Impossible\n");
else
{
output(lt[x]);
printf("\n");
}
}
return ;
}
2016-12-15 18:29:39
【BZOJ 1046】 1046: [HAOI2007]上升序列的更多相关文章
- BZOJ 1046: [HAOI2007]上升序列 LIS -dp
1046: [HAOI2007]上升序列 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3438 Solved: 1171[Submit][Stat ...
- 【BZOJ】1046 : [HAOI2007]上升序列
1046: [HAOI2007]上升序列 题意:给定S={a1,a2,a3,…,an}问是否存在P={ax1,ax2,ax3,…,axm},满足(x1 < x2 < … < xm)且 ...
- Bzoj 1046: [HAOI2007]上升序列 二分,递推
1046: [HAOI2007]上升序列 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3671 Solved: 1255[Submit][Stat ...
- BZOJ 1046: [HAOI2007]上升序列(LIS)
题目挺坑的..但是不难.先反向做一次最长下降子序列.然后得到了d(i),以i为起点的最长上升子序列,接下来贪心,得到字典序最小. ----------------------------------- ...
- bzoj 1046 : [HAOI2007]上升序列 dp
题目链接 1046: [HAOI2007]上升序列 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3620 Solved: 1236[Submit] ...
- BZOJ 1046: [HAOI2007]上升序列【贪心+二分状态+dp+递归】
1046: [HAOI2007]上升序列 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 4987 Solved: 1732[Submit][Stat ...
- 1046: [HAOI2007]上升序列(dp)
1046: [HAOI2007]上升序列 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 4999 Solved: 1738[Submit][Stat ...
- BZOJ1046 [HAOI2007]上升序列 【LIS + 字典序最小】
1046: [HAOI2007]上升序列 Time Limit: 10 Sec Memory Limit: 162 MB Submit: 5410 Solved: 1877 [Submit][St ...
- 【BZOJ1046】[HAOI2007]上升序列
[BZOJ1046][HAOI2007]上升序列 题面 bzoj 洛谷 题解 \(dp\)完之后随便搞一下即可,注意不要看错题 代码 #include <iostream> #includ ...
随机推荐
- (转)MSSQL 各个发行版本版本号以及Compact 版本号
终于开始写博客了. 不要笑啊. 下面是MSSQL 的发行版本以及版本号.自己整理的. http://support.microsoft.com/kb/321185/zh-cn SQL Server 2 ...
- nodejs7.0 试用 async await
nodejs 7.0.0 已经支持使用 --harmony-async-await 选项来开启async 和 await功能. 在我看来,yield 和 async-await 都是在特定范围内实现了 ...
- CodeForces 679B(Bear and Tower of Cubes)
题意:Limak要垒一座由立方体垒成的塔.现有无穷多个不同棱长(a>=1)的立方体.要求:1.塔的体积为X(X<=m).2.在小于X的前提下,每次都选体积最大的砖块.3.在砖块数最多的前提 ...
- Linux ---> 简单socket
#include <stdio.h> #include <stdlib.h> #include <errno.h> #include <string.h> ...
- 按照自己的理解实现比特交换协议(alternating-bit protocol)
一开始的思路是想写两个程序的,发送端和接收端.但是后来想了下,因为是模拟的,所以不用这么麻烦,直接自己定制场景(比如说丢包阿,包出错阿,超时之类的). 基本上是根据上图所写的,一个函数发包,一个函数接 ...
- C#代码分层的好处
1.对于复杂的系统,分层让代码结构清晰,便于开发人员对系统进行整体的理解.把握.如果代码没有分层,把逻辑都写在一个方法里面的代码就好比是一本没有目录的文档,要找出其中某一节都要对全文遍览一次. 2.基 ...
- Android开发系列之按钮事件的4种写法
经过前两篇blog的铺垫,我们今天热身一下,做个简单的例子. 目录结构还是引用上篇blog的截图. 具体实现代码: public class MainActivity extends Activity ...
- discuz管理中心无法登陆
检查下配置文件,当前管理是不是创始人. 如是,那试下修改数据库,在数据表出错时也会这样,还有一个也同时试下 \config\config_global.php 文件 $_config['admincp ...
- 使用c#检测文件正在被那个进程占用
要检测文件被那个进程占用,需要使用微软提供的工具Handle.exe,这里有微软提供的下载 我们可以在c#中调用Handle.exe 来检测到底哪个进程占用了文件 string fileName = ...
- MySQL基础学习之函数
数学函数 绝对值 abs() 圆周率 PI() 平方根 sqrt() 模除取余 mod(被除数,除数) 随机数 rand() 四舍五入 round(数字) 次方 ...