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<=1000

Output

  对于每个询问,如果对应的序列存在,则输出,否则打印Impossible.

Sample Input

6
3 4 1 2 3 6
3
6
4
5

Sample Output

Impossible
1 2 3 6
Impossible

HINT

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]上升序列的更多相关文章

  1. BZOJ 1046: [HAOI2007]上升序列 LIS -dp

    1046: [HAOI2007]上升序列 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3438  Solved: 1171[Submit][Stat ...

  2. 【BZOJ】1046 : [HAOI2007]上升序列

    1046: [HAOI2007]上升序列 题意:给定S={a1,a2,a3,…,an}问是否存在P={ax1,ax2,ax3,…,axm},满足(x1 < x2 < … < xm)且 ...

  3. Bzoj 1046: [HAOI2007]上升序列 二分,递推

    1046: [HAOI2007]上升序列 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3671  Solved: 1255[Submit][Stat ...

  4. BZOJ 1046: [HAOI2007]上升序列(LIS)

    题目挺坑的..但是不难.先反向做一次最长下降子序列.然后得到了d(i),以i为起点的最长上升子序列,接下来贪心,得到字典序最小. ----------------------------------- ...

  5. bzoj 1046 : [HAOI2007]上升序列 dp

    题目链接 1046: [HAOI2007]上升序列 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 3620  Solved: 1236[Submit] ...

  6. BZOJ 1046: [HAOI2007]上升序列【贪心+二分状态+dp+递归】

    1046: [HAOI2007]上升序列 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4987  Solved: 1732[Submit][Stat ...

  7. 1046: [HAOI2007]上升序列(dp)

    1046: [HAOI2007]上升序列 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4999  Solved: 1738[Submit][Stat ...

  8. BZOJ1046 [HAOI2007]上升序列 【LIS + 字典序最小】

    1046: [HAOI2007]上升序列 Time Limit: 10 Sec  Memory Limit: 162 MB Submit: 5410  Solved: 1877 [Submit][St ...

  9. 【BZOJ1046】[HAOI2007]上升序列

    [BZOJ1046][HAOI2007]上升序列 题面 bzoj 洛谷 题解 \(dp\)完之后随便搞一下即可,注意不要看错题 代码 #include <iostream> #includ ...

随机推荐

  1. C#学习笔记之线程 - 使用线程

    基于事件的异步模式 (EAP -- The Event-Based Asynchronous Pattern) EAP提供了一个简单的办法,通过它的类能够提供多线程能力,而不需显式的开启或管理线程.它 ...

  2. 委托和事件[delegate and event]_C#

    委托和事件: 1. 委托:一个能够表示方法的数据类型:它将方法作为对象封装起来,允许在运行时间接地绑定一个方法调用. 2. 声明委托数据类型: public delegate  bool Greate ...

  3. zz Release memory in Linux (Unused or Cached)

    In computer science, Memory Leakage occurs when a computer software or program consumes required mem ...

  4. 119. Pascal's Triangle II

    Given an index k, return the kth row of the Pascal's triangle. For example, given k = 3,Return [1,3, ...

  5. CAF(C++ actor framework)(序列化之复杂类,分析 还有自己不懂的细思恐极函数实现)(三)

    这里应该是序列化的最后一篇.感觉自己写的不是很好,也一点点在学习.这次就不贴上代码了.代码在github上的announce5.cpp.代码简单,但是分析下去会有细思恐极的感觉! 先看一下几个函数是干 ...

  6. Python 三大神器

    Python 三大神器 Python 中有很多优秀的包,本文主要讲一下 pip, virtualenv, fabric 1. pip 用来包管理 文档:https://pip.pypa.io/en/l ...

  7. windows github 搭建与使用

    git/github使用以下是全部在命令行使用(windows/github) 注册账户以及创建仓库先在github建立账号和创建仓库,此时为空的仓库 配置git下载并安装 git windows版本 ...

  8. 关于Socket编写简单聊天工具的总结(原创)

    这段时间再看socket编程,虽然现在是刚刚接触,但是还是忍不住想写一篇总结,来激励自己努力学习,写的不好的地方,还请大家指教啊! 下面针对一个简单的发送消息和文件的程序说说吧.   首先是服务器需要 ...

  9. Shell根据年月日创建文件夹

    #!/bin/sh dir_path="/vol/project/log/test/" ..} do #echo "$year" cd $dir_path mk ...

  10. 在windows服务器中,将MongoDB服务化。

    将mongodb在windows中服务化,就是将其注册成一个服务组件,并可以设置成,手动/自动 启动. 一般的我们都会在command窗口运行如下: d:\mongodb\bin>mongod ...