洛谷 P1020 导弹拦截 【最长上升子序列】 || 【线段树】

题目链接:https://www.luogu.org/problemnew/show/P1020

题目描述

某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统。但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度。某天,雷达捕捉到敌国的导弹来袭。由于该系统还在试用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹。

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

输入格式:

一行,若干个整数(个数少于100000)

输出格式:

2行,每行一个整数,第一个数字表示这套系统最多能拦截多少导弹,第二个数字表示如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。

输入样例#1:

389 207 155 300 299 170 158 65

输出样例#1:

6
2

下面我的代码是WA的,只不过我还不知道为什么WA,先记录下来,至于线段树做法,等以后学会了再来补吧。
#include <bits/stdc++.h>              //此题主要是不知道如何用stl中的upper_bound()函数对结构体中的某一特定元素进行二分
using namespace std;                  //当然,也可也写一个二分子函数对该结构体进行二分,但是还是过不了,此题先记录着吧
int arr[100010];
int vis[100010] = {0};

struct node
{
    int val;
    int loc;
}lis[100010];

struct cmp1
{
    bool operator () (const node &a, const node &b) const {
        return a.val < b.val;
    }
};

int main()
{
    int num = 0;    //记录下标记了多少个导弹
    int a, len = 0,res=0;
    while (cin >> a) {
        arr[++res] = a;      //res为导弹的数量
    }
    int ans = 0; int cas = 0;
    while (num != res) {
        for (int i = 1; i <= res; i++)
        {
            if (!vis[i])
            {
                if (arr[i] > lis[len].val) {
                    lis[++len].val = arr[i];
                    lis[len].loc = i;
                    vis[i] = 1;
                }
                else
                {        //*******对结构体中的某一个元素进行二分查找的方法,详情见上面的cmp()函数
                    node cur; cur.val = arr[i];
                    int j = upper_bound(lis + 1, lis + len + 1, cur,cmp1()) - lis;      //由于这是最长不上升子序列,所以用upper_bound()函数
                    vis[lis[j].loc] = 0;         //由于原来最长上升子序列中的数此时不用了,所以要取消标记
                    lis[j].val = arr[i];
                    lis[j].loc = i;
                    vis[i] = 1;        //标记此时要用的数
                }
            }
        }
        if (!cas)cout << len<<endl;
        num += len;
        len = 0; memset(lis, 0, sizeof(lis));
        ++cas;
        ++ans;         //导弹系统的数量
    }
    cout << ans << endl;
    return 0;
}

2018-05-20

原文地址:https://www.cnblogs.com/00isok/p/9063729.html

时间: 2024-10-08 23:53:47

洛谷 P1020 导弹拦截 【最长上升子序列】 || 【线段树】的相关文章

codevs1044 拦截导弹==洛谷 P1020 导弹拦截

P1020 导弹拦截 题目描述 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度.某天,雷达捕捉到敌国的导弹来袭.由于该系统还在试用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹. 输入导弹依次飞来的高度(雷达给出的高度数据是不大于30000的正整数),计算这套系统最多能拦截多少导弹,如果要拦截所有导弹最少要配备多少套这种导弹拦截系统. 输入输出格式 输入格式: 一行,若干个正

洛谷P1020导弹拦截——LIS

题目:https://www.luogu.org/problemnew/show/P1020 主要是第二问,使用了dilworth定理:一个序列中最长不上升子序列的最大覆盖=最长上升子序列长度. dilworth定理:http://www.cnblogs.com/nanke/archive/2011/08/11/2134355.html 代码如下: #include<iostream> #include<cstdio> using namespace std; int d,b[10

洛谷 [P1020] 导弹拦截 (N*logN)

首先此一眼就能看出来是一个非常基础的最长不下降子序列(LIS),其朴素的 N^2做法很简单,但如何将其优化成为N*logN? 我们不妨换一个思路,维护一个f数组,f[x]表示长度为x的LIS的最大的最后一个数字是f[x].(为什么是最大的?可以应用贪心的思想,发现对于相同的x,f[x]越大其后可能扩展的情况就越多,即就越优)我们可以发现f数组单调递减(为什么?也可使用反证法证明,在此不赘述)对于决策单调性问题,一般使用二分法优化,这就是logN的来历.二分的边界条件一定要写对. 代码如下: #i

洛谷P1158 导弹拦截 排序

---恢复内容开始--- 洛谷P1158 导弹拦截 排序 算是有技巧的枚举吧 题意 用两套系统来拦截导弹,一个系统的费用等于这个系统拦截的导弹中离他最远的那颗导弹和系统的距离 的平方 排序 将每颗导弹按距离系统1 的距离排序,然后枚举n--0 选这些导弹为系统1最远能够拦截的导弹 然后就可以更新下一次 系统2要拦截的导弹 中离系统2 最远的一颗 1 #include <cstdio> 2 #include <cmath> 3 #include <cstdlib> 4 #

【题解】P1020 导弹拦截

[题解]P1020 导弹拦截 从n^2到nlogn 第二问就是贪心 第一问: 简化题意:求最长不下降子序列 普通n^2: for (int i = 1; i <= n; i++) for (int j = 1; j < i; j++) if(a[j] >= a[i]) f[i] = max(f[i], f[j] + 1); cout << f[n]; 另一种n^2级,可能快一点点(还没交,不知对不对) f[0] = 1; for (int i = 1; i <= n;

p1020导弹拦截

传送门 P1020导弹拦截 题目描述 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度.某天,雷达捕捉到敌国的导弹来袭.由于该系统还在试用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹. 输入导弹依次飞来的高度(雷达给出的高度数据是不大于30000的正整数),计算这套系统最多能拦截多少导弹,如果要拦截所有导弹最少要配备多少套这种导弹拦截系统. 输入输出格式 输入格式: 一行,若

洛谷 P3380 【模板】二逼平衡树(树套树)

洛谷 P3380 [模板]二逼平衡树(树套树) 线段树套treap: 就是线段树每个节点放一个treap.建树复杂度应该是$n log n$,操作1,3,4,5的复杂度是$(log n)^2$,操作2的复杂度是$(log n)^3$. 操作3:找到线段树的对应叶子节点后找到要删除的值,在回溯的时候更新线段树相关的每一个节点(在treap中去掉要删除的值,再加入要加入的值) 操作1:将操作转化为统计(这个区间[l,r]内小于x的数的个数)+1.那么通过线段树将区间分解,然后对分解出的每一个区间对应

AC日记——导弹拦截 洛谷 P1020 (dp+模拟)

题目描述 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度.某天,雷达捕捉到敌国的导弹来袭.由于该系统还在试用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹. 输入导弹依次飞来的高度(雷达给出的高度数据是不大于30000的正整数),计算这套系统最多能拦截多少导弹,如果要拦截所有导弹最少要配备多少套这种导弹拦截系统. 输入输出格式 输入格式: 一行,若干个正整数最多100个. 输

洛谷 P2487 [SDOI2011]拦截导弹

题目描述 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度.并且能够拦截任意速度的导弹,但是以后每一发炮弹都不能高于前一发的高度,其拦截的导弹的飞行速度也不能大于前一发.某天,雷达捕捉到敌国的导弹来袭.由于该系统还在试用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹. 在不能拦截所有的导弹的情况下,我们当然要选择使国家损失最小.也就是拦截导弹的数量最多的方案.但是拦截导弹数量的最多的方案有可能有多个,如果有多个最优方案