[BZOJ]1046 上升序列(HAOI2007)
和字典序有关的题型啊。
Description
对于一个给定的S={a1,a2,a3,…,an},若有P={ax1,ax2,ax3,…,axm},满足(x1 < x2 < … < xm)且(ax1 < ax2 < … < axm)。那么就称P为S的一个上升序列。如果有多个P满足条件,那么我们想求字典序最小的那个。任务给出S序列,给出若干询问。对于第i个询问,求出长度为Li的上升序列,如有多个,求出字典序最小的那个(即首先x1最小,如果不唯一,再看x2最小……),如果不存在长度为Li的上升序列,则打印Impossible.
Input
第一行一个N,表示序列一共有N个元素第二行N个数,为a1,a2,…,an 第三行一个M,表示询问次数。下面接M行每行一个数L,表示要询问长度为L的上升序列。
Output
对于每个询问,如果对应的序列存在,则输出,否则打印Impossible。
Sample Input
6
3 4 1 2 3 6
3
6
4
5
Sample Output
Impossible
1 2 3 6
Impossible
HINT
N<=10000,M<=1000。
Solution
看到字典序和上升子序列DP,小C想起了APIO2009的那道convention。
(不要吐槽小C的做题顺序)
既然要取字典序最小的,那肯定就是从小的取起,能取就取。
那什么情况算是能取的呢?
设要取的上升序列长度为len,已经取了x个数,且最后一个数大小为last。
设我们当前要取的数的位置为i,数的大小为a[i],以这个位置开头的最长上升子序列长度为d[i]。
那么如果a[i]>last且d[i]+x>=len那么i这个位置的数就能取,也就是必须取。
我们来分析为什么这两个条件可以决定这个数取不取。
a[i]>last是上升序列的必要条件,我们已经满足了last必须取,且上一个取last一定能构造出答案,所以从字典序最小的角度来说,我们不能用a[i]来替换last。
d[i]+x>=len则是能构造出长度为len的答案的充分条件,因为d[i]满足二分性,如果d[i]>=len-x,那么一定存在一个以a[i]开头长度为len-x的上升序列接在last后面,从字典序最小的角度来说,我们取它显然最优。
剩下就是求d[i]了嘛。求最长上升子序列谁不会啊?
其实是正常的字典序也好,还是这题的逗比字典序也好,用以上的思路都是可以做的。
时间复杂度O(nlogn+n*m)。
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <vector>
#define MN 10005
#define MM 1005
using namespace std;
struct meg{int val,pos;}b[MM];
int f[MN],a[MN],p[MN],lb[MM],ans[MM][MN],t[MN<<];
int nin,n,m,MQ; inline int read()
{
int n=,f=; char c=getchar();
while (c<'' || c>'') {if(c=='-')f=-; c=getchar();}
while (c>='' && c<='') {n=n*+c-''; c=getchar();}
return n*f;
} void dfs(int depth,int x,int L,int R,int ps)
{
register int i,j,dx,mx=L-;
for (i=x;i<=n;++i)
{
if (a[i]<=ps) continue;
dx=lower_bound(lb+,lb+m+,f[i]+depth)-lb-;
if (dx>mx)
{
for (j=mx+;j<=dx;++j) ans[b[j].pos][depth]=a[i],mx=lb[j]==depth?j:mx;
if (dx>mx) {dfs(depth+,i+,mx+,dx,a[i]); mx=dx;}
if (mx==R) break;
}
}
} inline void getad(int x,int z) {for (x+=MQ;x&&z>t[x];x>>=) t[x]=z;}
inline int getmx(int L,int R)
{
if (L>R) return ;
register int lt=;
for (L+=MQ,R+=MQ;L<=R;L>>=,R>>=)
{
if ( L&) lt=max(lt,t[L++]);
if (~R&) lt=max(lt,t[R--]);
}
return lt;
}
bool cmp(const meg& a,const meg& b) {return a.val<b.val;} int main()
{
register int i,j;
n=read();
for (i=;i<=n;++i) a[i]=p[i]=read();
sort(p+,p+n+);
nin=unique(p+,p+n+)-p-;
for (i=;i<=n;++i) a[i]=lower_bound(p+,p+nin+,a[i])-p;
for (MQ=;MQ<nin;MQ<<=); --MQ;
for (i=n;i;--i) f[i]=getmx(a[i]+,nin)+,getad(a[i],f[i]);
m=read();
for (i=;i<=m;++i) b[i].val=read(),b[i].pos=i;
sort(b+,b+m+,cmp);
for (i=;i<=m;++i) lb[i]=b[i].val;
dfs(,,,m,);
for (i=;i<=m;++i)
{
if (!ans[i][]) {puts("Impossible"); continue;}
printf("%d",p[ans[i][]]);
for (j=;ans[i][j];++j) printf(" %d",p[ans[i][j]]); puts("");
}
}
Last Word
不知道为什么突然想把询问排序然后一起做想降低常数,不过常数好像更大了233。
因为看错字典序的缘故,小C的代码画风有点崩坏……(其实本来就很崩)
[BZOJ]1046 上升序列(HAOI2007)的更多相关文章
- BZOJ 1046 上升序列
Description 对于一个给定的S={a1,a2,a3,…,an},若有P={ax1,ax2,ax3,…,axm},满足(x1 < x2 < … < xm)且( ax1 < ...
- BZOJ 1046 上升序列(LIS变形)
要保证长度为L的序列下标字典序最小,当然要尽量选前面的数. 如何判断前面的数是否满足条件?,只需要知道这个数开头的递增序列的最长长度是多少,如果不小于L,那么必然可以加入这个数.还需判断一下它是否大于 ...
- 【BZOJ】【1046】【HAOI2007】上升序列
DP+贪心 啊……其实是个水题,想的复杂了 令f[i]表示以 i 为起始位置的最长上升子序列的长度,那么对于一个询问x,我们可以贪心地从前往后扫,如果f[i]>=x && a[i ...
- [BZOJ 1046] [HAOI2007] 上升序列 【DP】
题目链接:BZOJ - 1046 题目分析 先倒着做最长下降子序列,求出 f[i],即以 i 为起点向后的最长上升子序列长度. 注意题目要求的是 xi 的字典序最小,不是数值! 如果输入的 l 大于最 ...
- BZOJ 1046 最长不降子序列(nlogn)
nlogn的做法就是记录了在这之前每个长度的序列的最后一项的位置,这个位置是该长度下最后一个数最小的位置.显然能够达到最优. BZOJ 1046中里要按照字典序输出序列,按照坐标的字典序,那么我萌可以 ...
- [BZOJ 4350]括号序列再战猪猪侠 题解(区间DP)
[BZOJ 4350]括号序列再战猪猪侠 Description 括号序列与猪猪侠又大战了起来. 众所周知,括号序列是一个只有(和)组成的序列,我们称一个括号 序列S合法,当且仅当: 1.( )是一个 ...
- (WAWAWAWAWAWA) BZOJ 1858: [Scoi2010]序列操作
二次联通门 : BZOJ 1858: [Scoi2010]序列操作 /* BZOJ 1858: [Scoi2010]序列操作 已经... 没有什么好怕的的了... 16K的代码... 调个MMP啊.. ...
- BZOJ 1046: [HAOI2007]上升序列 LIS -dp
1046: [HAOI2007]上升序列 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 3438 Solved: 1171[Submit][Stat ...
- 【BZOJ 1046】 1046: [HAOI2007]上升序列
1046: [HAOI2007]上升序列 Description 对于一个给定的S={a1,a2,a3,-,an},若有P={ax1,ax2,ax3,-,axm},满足(x1 < x2 < ...
随机推荐
- hi-nginx-1.4.2发布,多项重要更新
支持多种编程语言混合开发web应用的通用服务器hi-nginx-1.4.2已经发布了. 此次发布包含多项重要更新: 支持python2和3,通过编译选项--with-http-hi-python-ve ...
- linux系统命令学习系列-用户切换命令su,sudo
先复习一下上节内容: 用户组添加groupadd 用户组修改groupmod 用户组删除groupdel 作业创建一个id为501的组group1,然后改成group2, 同时id变为502,最后删除 ...
- python 之 列表list && 元组tuple
目录: 列表 列表基本操作 列表的操作符 列表的函数和方法 元组 介绍: 列表是一种可变的有序集合,可以进行访问.添加和删除操作. 元组是一种不可变的有序集合,可以访问. 1.列表的基本操作 创建列表 ...
- Linq 集合操作符 Except,Intersect,Union
IList<string> s1 = new List<string>() { "One", "Two", "Three&qu ...
- SpringCloud的Hystrix(五) Hystrix机制
参考链接:http://www.jianshu.com/p/e07661b9bae8 一.前言 大型复杂的分布式系统中,高可用相关的技术架构非常重要.高可用架构非常重要的一个环节,就是如何将分布式系统 ...
- SpringBoot入门:Spring Data JPA 和 JPA(理论)
参考链接: Spring Data JPA - Reference Documentation Spring Data JPA--参考文档 中文版 纯洁的微笑:http://www.ityouknow ...
- hadoop2.7.3+spark2.1.0+scala2.12.1环境搭建(3)http://www.cnblogs.com/liugh/p/6624491.html
一.文件准备 scala-2.12.1.tgz 下载地址: http://www.scala-lang.org/download/2.12.1.html 二.工具准备 2.1 Xshell 2.2 X ...
- OAuth2.0学习(1-7)授权方式4-客户端模式(Client Credentials Grant)
授权方式4-客户端模式(Client Credentials Grant) 客户端模式(Client Credentials Grant)指客户端以自己的名义,而不是以用户的名义,向"服务提 ...
- Oracle update 执行更新操作后的数据恢复
操作数据库,经常会出现误操作,昨天执行的更新操作之后发现更新错了,只能想办法数据恢复了,现在整理一下 第一步:查询执行更新操作的时间 select r.FIRST_LOAD_TIME,r.* from ...
- yagmail让发邮件更简单
这是我迄今为止碰到的最良心的库,真tm简单啊 import yagmail # 连接邮箱服务器 yag = yagmail.SMTP(user="wuyongqiang2012@163.co ...