题目
题目描述
某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统。但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度。某天,雷达捕捉到敌国的导弹来袭。由于该系统还在试用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹。
输入导弹依次飞来的高度(雷达给出的高度数据是\(\le 50000\)的正整数),计算这套系统最多能拦截多少导弹,如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。
输入格式
\(1\)行,若干个整数(个数\(\le 100000\))
输出格式
\(2\)行,每行一个整数,第一个数字表示这套系统最多能拦截多少导弹,第二个数字表示如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。
算法分析
我们先看一个例子。
例子
输入数据
10
0 192 100 91 149 146 159 137 17 188
定义三个变量
opt[i]:表示长度为i的不升序列的末位数字最大为opt[i]
opt_n:当前最长不升序列的长度
f[i]:动态规划计算,以第i个导弹结尾的最长不升序列的长度为f[i]
定义一个函数
int s(int i):二分查找最大的k使得opt[k]>=h[i]
初始值
memset(opt,0,sizeof(opt));
opt[1]=h[1];f[1]=1;
状态转移方程
\(f[i]=k+1,k=max\{k\mid opt[k]\ge h[i]\}\)
如果\(\{k\mid opt[k]\ge h[i]\}=\varnothing\)则\(k=0\)
遍历
代码
for(int i=2;i<=n;++i){
k=s(i);f[i]=k+1; //求最大的k使得opt[k]>=h[i],则f[i]=k+1
if(f[i]<=opt_n){ //如果f[i]不超过opt_n,则考虑更新opt[f[i]]
if(opt[f[i]]<h[i]) opt[f[i]]=h[i];
}else opt_n++,opt[opt_n]=h[i];
}
手算理解(建议先看总结,看不懂再看这里):
-
当
i=2,h[i]=192
,没有1<=k<=n
使得opt[k]>192
,那么k=s(i)=0
,f[i]=0+1=1
。
此时f[i]>opt_n=0
,即 当前方案长度 超过了opt
中已知的最长序列的长度,则需要更新opt
(发现了更长的不升序列,补到opt
中)。opt_n++; //opt_n=1
opt[opt_n]=h[i] //opt[1]=192 -
当
i=3,h[i]=100
,存在最大的k=1
使得opt[k]=192>100
,那么f[i]=k+1=2
。
此时f[i]>opt_n
,更新opt
。opt_n++; //opt_n=2
opt[opt_n]=h[i] //opt[2]=100 -
当
i=4,h[i]=91
,存在最大的k=2
使得opt[k]=100>91
,那么f[i]=k+1=3
。
此时f[i]>opt_n
,更新opt
。opt_n++; //opt_n=3
opt[opt_n]=h[i] //opt[3]=91 -
当
i=5,h[i]=149
,存在最大的k=1
使得opt[k]=192>149
,那么f[i]=1+1=2
。
此时f[i]<opt_n
,即 当前序列长度 小于opt
中已知的最长序列的长度;也就是说,前面已经有过长度相同的不升序列。
需要判断此时h[i]
是否大于opt[f[i]]
,因为opt
要存最大的末位数字。if(opt[f[i]]<h[i]) //opt[f[i]]=opt[2]=100<149
opt[f[i]]=h[i]; //opt[2]=149
依此类推,之后的操作直接给出:
i=6,f[i]=3
opt[1]=192|opt[2]=149|opt[3]=146|opt[4]= 0|opt[5]= 0|opt[6]= 0|
i=7,f[i]=2
opt[1]=192|opt[2]=159|opt[3]=146|opt[4]= 0|opt[5]= 0|opt[6]= 0|
i=8,f[i]=4
opt[1]=192|opt[2]=159|opt[3]=146|opt[4]=137|opt[5]= 0|opt[6]= 0|
i=9,f[i]=5
opt[1]=192|opt[2]=159|opt[3]=146|opt[4]=137|opt[5]= 17|opt[6]= 0|
i=10,f[i]=2
opt[1]=192|opt[2]=188|opt[3]=146|opt[4]=137|opt[5]= 17|opt[6]= 0|
总结
而对于同长度的序列,要判断h[i]
是否能够接在其后,当然只要判断这些序列中最大的末位数字是否比h[i]
大。大了就一定可以,小了就一定不可以。所以opt[i]
存的是长度为i的不升序列的末位最大数字。
于是,计算f[i]
只要在k=1..i-1
中找满足opt[k]>=h[i]
的尽可能大的k,接在其后,新的序列长度就为k+1
。
而opt
数组是递减的(反证法:因为每个时刻opt[i]
都存长度为i
的不升序列的末位最大数字,若i<j
且opt[i]<opt[j]
,那么为什么opt[i]
的值不能为opt[j]
这个序列中的第i
个元素呢?),所以可以用二分查找最大的k。
在上面这些操作后,时间复杂度就顺利地降成了O(nlogn)
。(´▽` )
附:完整代码
#include<cstdio>
#define reg register
using namespace std;
int n,h[100001],f[100001],opt[100001],opt_n;
int s(int i){
//二分查找最大的k使得opt[k]>=h[i]
int l=1,r=n,m;
while(l<=r){
m=(l+r)/2;
if(opt[m]>=h[i]){
if(m+1<=r && opt[m+1]>=h[i])
l=m+1;
else return m;
}else r=m-1;
}
return 0;
}
void dp(){
//opt[i]:长度是i的最长不升子序列所有子串中末尾最大的那个数,
//根据这个数字,我们可以容易知道,
//只要当前考察的这个数比opt[i]小,那么当前这个数一定能通过opt[i]构成一个长度为i+1的下降子序列。
int k;
opt_n=1;
opt[1]=h[1];f[1]=1;
for(reg int i=2;i<=n;++i){
k=s(i);f[i]=k+1; //求最大的k使得opt[k]>=h[i],则f[i]=k+1
if(f[i]<=opt_n){ //如果f[i]不超过opt_n,则考虑更新opt[f[i]]
if(opt[f[i]]<h[i]) opt[f[i]]=h[i];
}else opt_n++,opt[opt_n]=h[i];
/*--debug--
printf("\n\ni=%d,f[i]=%d\n",i,f[i]);
for(reg int x=1;x<=6;++x)printf("opt[%d]=%3d|",x,opt[x]);
*/
}
}
int main(){
scanf("%d",&n);
for(reg int i=1;i<=n;++i)scanf("%d",&h[i]);
dp();
printf("%d",opt_n);
}
拦截导弹nlogn解法的更多相关文章
-
1260:【例9.4】拦截导弹(Noip1999)
题目来源:http://ybt.ssoier.cn:8088/problem_show.php?pid=1260 1260:[例9.4]拦截导弹(Noip1999) 时间限制: 1000 ms ...
-
拦截导弹类问题 (Codevs4888零件分组POJ1065Wooden Sticks)(LIS及其覆盖问题)
拦截导弹 题意:求最长不上升子序列长度:求一个序列最少分成几个非增子序. 第一问易求,已知序列a,令f[i]为a前i个元素的最长非增子序的长度,则有 f[i]=max{f[i],f[j]+1} (1& ...
-
codevs1409 拦截导弹2
[问题描述]一场战争正在 A 国与 B 国之间如火如荼的展开.B 国凭借其强大的经济实力开发出了无数的远程攻击导弹,B 国的*希望,通过这些导弹直接毁灭 A 国的指挥部,从而取得战斗的胜利!当然, ...
-
nyoj814_又见拦截导弹_DP
又见拦截导弹 时间限制:3000 ms | 内存限制:65535 KB 难度:3 描述 大家对拦截导弹那个题目应该比较熟悉了,我再叙述一下题意:某国为了防御敌国的导弹袭击,新研制出来一种导弹拦 ...
-
【动态规划】拦截导弹_dilworth定理_最长递增子序列
问题 K: [动态规划]拦截导弹 时间限制: 1 Sec 内存限制: 256 MB提交: 39 解决: 10[提交][状态][讨论版] 题目描述 张琪曼:“老师,修罗场是什么?” 墨老师:“修罗是 ...
-
ACM题目————又见拦截导弹
描述 大家对拦截导弹那个题目应该比较熟悉了,我再叙述一下题意:某国为了防御敌国的导弹袭击,新研制出来一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:它的第一发炮弹能够到达任意的高度,但是以后每一发炮 ...
-
nyoj------79拦截导弹
拦截导弹 时间限制:3000 ms | 内存限制:65535 KB 难度:3 描述 某国为了防御敌国的导弹袭击,发展中一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到 ...
-
百练_2945 拦截导弹(DP)
描述 某国为了防御敌国的导弹袭击,开发出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度.某天,雷达捕捉到敌国的导弹来袭 ...
-
nyoj 79 拦截导弹
拦截导弹 时间限制:3000 ms | 内存限制:65535 KB 难度:3 描述 某国为了防御敌国的导弹袭击,发展中一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到 ...
随机推荐
-
java hashCode方法返回值
hashCode 是和内存地址相关的一个整数. HashCode只是在需要用到哈希算法的数据结构中才有用 用途是为了方便快速地查找对象: HashMap 是根据键对象的 HashCode 来进行快速查 ...
-
nodePPT初认识启动与手机控制
最近要做个PPT,想起之前看到过个网页PPT,于是这次就想尝试下,搜了下弹出个nodeppt---有可能是最好的网页PPT,那,就这个吧. 按照文档来,貌似有点问题,百度,又是一堆粘贴复制,没点用.自 ...
-
xiaoxia的vim配置
这样已经很强大了 set nu sts=4 ts=4 sw=4 et si ai set ruler set hlsearch syntax on filetype plugin on
-
Codevs 1474 十进制转m进制
时间限制: 1 s 空间限制: 128000 KB 题目等级 : 白银 Silver 题目描述 Description 将十进制数n转换成m进制数 m<=16 n<=100 输 ...
-
Armitage主屏幕说明与命令行启动
(1)我们将Armitage主屏幕标注为A.B和C A:该区域显示预配置的模块.您可以在模块列表下面的文本框中输入要查找的模块进行查找. B:该区域显示我们可以进行漏洞测试的活跃主机. C:该区域显示 ...
-
python study
python django restul webservice返回json数据 2013-09-27 23:14 by lixingle, 249 visits, 网摘, 收藏, 编辑 摘要:做这个d ...
-
python之路(3)函数和匿名函数
函数 函数与过程 过程 def test(): "注释" print('1 am chen') test() : 过程调用 def : 定义函数的关键字 test : 函数名 pr ...
-
Implementation of WC in JAVA
Implementation of WC in JAVA github地址 相关要求 基本功能 -c [文件名] 返回文件的字符数 (实现) -w [文件名] 返回文件的词的数目 (实现) -l [文 ...
-
液晶电视插有线电视信号线的是哪个接口 HDMI是什么接口
1.液晶电视插有线电视信号线的接口(模拟信号)是射频接口(也叫RF接口,同轴电缆接口,闭路线接口),数字信号就得通过机顶盒转换成模拟信号视频输出至电视,才能正常收看电视节目. 2.电视机或高清机顶盒上 ...
-
Codeforces gym 100971 D. Laying Cables 单调栈
D. Laying Cables time limit per test 2 seconds memory limit per test 256 megabytes input standard in ...