zl程序教程

您现在的位置是:首页 >  其它

当前栏目

1010. 拦截导弹(LIS+贪心)

贪心 LIS
2023-09-14 09:14:58 时间

蓝桥杯国赛指南,详情见专栏

文章目录

Question

某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统。

但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度。

某天,雷达捕捉到敌国的导弹来袭。

由于该系统还在试用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹。

输入导弹依次飞来的高度(雷达给出的高度数据是不大于30000的正整数,导弹数不超过1000),计算这套系统最多能拦截多少导弹,如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。

输入格式
共一行,输入导弹依次飞来的高度。

输出格式
第一行包含一个整数,表示最多能拦截的导弹数。

第二行包含一个整数,表示要拦截所有导弹最少要配备的系统数。

数据范围
雷达给出的高度数据是不大于 30000 的正整数,导弹数不超过 1000。

输入样例:
389 207 155 300 299 170 158 65
输出样例:
6
2

Ideas

最长上升子序列模型+贪心

Code

# 最长上升子序列变形 LIST + 贪心
a = list(map(int,input().strip().split()))
n = len(a)
f = [0 for i in range(n)]
g = [0 for i in range(n)]

res = 0
for i in range(n):
    f[i] = 1
    for j in range(i):
        if a[j] >= a[i]:
            f[i] = max(f[i],f[j]+1)
    res = max(res,f[i])
print(res)

# 贪心策略:
# 1. 当前的数大于任何一个子序列的结尾,重新创建一个子序列
# 2. 当前的数小于等于一个子序列的结尾,寻找子序列结尾最小的那个

cnt = 0
for i in range(n):
    k = 0
    # print(g)
    while k<cnt and a[i]>g[k]:
        k += 1
    g[k] = a[i]
    if k >= cnt:
        cnt += 1
print(cnt)