hdu 1257 最少拦截系统(动态规划 || 构造法-贪心 || 线段树)

本文介绍了求解最少拦截系统问题的动态规划、贪心及线段树三种算法,通过分析样例数据展示了如何找到最长升序链,并利用这些链来确定最少需要的拦截系统数量。动态规划解决方案的时间复杂度为O(n^2),贪心算法为O(nlogn),线段树为O(nlogn),并给出了相应的C++代码实现。
摘要由CSDN通过智能技术生成

最少拦截系统 - http://acm.hdu.edu.cn/showproblem.php?pid=1257

分析

样例数据9 8 389 207 155 300 299 170 158 65(导弹9枚,与原题不同;画图时把8也看作了导弹高度,将错就错。)】
在这里插入图片描述

  • 图中画出所有升序链(任意一条升序链上的任意两个元素都不能被同一个拦截系统拦截)
  • 最长升序链的长度为k,则拦截系统不少于k个;否则,会至少存在1个拦截系统拦截了这个最长升序链中至少2个元素,不符题意。样例 k = 3,即拦截系统不少于3个。
  • 可以构造出k个非升序集合(即:反链),构造方法是:每次用所有升序链的头(最小)元素构造一个非升序集合;k次构造后剩余集合为空集。例如:{8}、{389,207,155,65}、{300,299,170,158}。
  • 结论:拦截系统最少要k个。参考:狄尔沃斯(Dilworth)定理 / 偏序集的分解定理
  • 问题转化为:最长升序链的长度k

代码

动态规划【O(n2), 187MS】

// hdu 1257 最少拦截系统
#include<bits/stdc++.h>
using namespace std;	
#define MXH 30010
int n, h, dp[MXH];
int main(){
	while(cin >> n){
		memset(dp, 0, sizeof dp);
		for(int i = 1; i <= n; i++){
			cin >> h;
			dp[h]++;
			for(int j = 1; j < h; j++){ // 此循环O(n),可优化为logn,优化后成线段树
				if(dp[h] < dp[j]+1) dp[h] = dp[j]+1;
			}
		}
		for(int i = 1; i < MXH; i++) if(dp[0] < dp[i]) dp[0] = dp[i];
		cout << dp[0] << endl;
	}
	return 0;
}

反链构造法 — 又称:贪心【O(nlogn), 31MS]

  • 数组a的1…k存放已有的k个导弹拦截系统各自拦截的最后一颗导弹的高度,递增排列
  • 初始时刻 k = 1
  • 新出现的导弹尽量用已有的导弹拦截系统拦截,且该拦截系统的最后一颗导弹的高度是尽量低
  • 新出现的导弹无法用已有的拦截系统拦截,则增加一套新的拦截系统,且是在k+1的位置
  • 构造过程:
    —> {8} // 初始状态,a数组的值:[8], k=1
    —> {8} 、{389} // 与8构成升序链,a数组的值:[8, 389], k=2
    —> {8} 、{389,207} // 与8构成升序链,a数组的值:[8, 207], k=2
    —> {8} 、{389,207,155} // 与8构成升序链,a数组的值:[8, 155], k=2
    —> {8} 、{389,207,155} 、{300} // 与207、155构成升序链,a数组的值:[8, 155, 300], k=3
    —> {8} 、{389,207,155} 、{300,299} // 与207、155构成升序链,a数组的值:[8, 155, 299], k=3
    —> {8} 、{389,207,155} 、{300,299,177} // 与155构成升序链,a数组的值:[8, 155, 177], k=3
    —> {8} 、{389,207,155} 、{300,299,177,158} // 与155构成升序链,a数组的值:[8, 155, 158], k=3
    —> {8} 、{389,207,155, 65} 、{300,299,177,158} // 与8构成升序链,a数组的值:[8, 65, 158], k=3
// hdu 1257 最少拦截系统
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
ll a[100005], k, n, t;
int main() {
    while (cin >> n) {
        k = 1;
        memset(a, 0, sizeof(a));
        cin >> t;
        a[1] = t;
        for (ll i = 2; i <= n; ++i) {
            cin >> t;
            if (t > a[k]) a[++k] = t;
            else *lower_bound(a + 1, a + k + 1, t) = t;
        }
        cout << k << endl;
    }
    return 0;
}

线段树【O(nlogn),15MS】

// hdu 1257 最少拦截系统
#include <bits/stdc++.h>
using namespace std;
#define MXH 30010
int n, h;
struct Node{ int l, r, mx = 0;}tr[MXH<<2];
void build(int root, int l, int r){
    Node &rt = tr[root];
    rt.l = l, rt.r = r, rt.mx = 0;
    if(l == r) return;
    int mid = (l+r)>>1;
    build(root << 1, l, mid);
    build(root << 1 | 1, mid+1, r);
}
int query(int root, int l, int r){
	Node &rt = tr[root];
	if(rt.l == l && rt.r == r) return rt.mx;
	int mid = (rt.l+rt.r)>>1;
	if(r <= mid) return query(root<<1, l, r);
	else if(l > mid) return query(root<<1|1, l, r);
	else return max(query(root<<1, l, mid), query(root<<1|1, mid+1, r));
}
void update(int root, int h, int v){
	Node &rt = tr[root];
	if(rt.l == rt.r) {
		rt.mx = v;
		return;
	}
	int mid = (rt.l + rt.r)>>1;
	if(h <= mid) update(root << 1, h, v);
	else update(root<<1|1, h, v);
	rt.mx = max(tr[root<<1].mx,tr[root<<1|1].mx);
}
int main() {
    while (cin >> n) {
        build(1, 1, MXH);
        for (int i = 1; i <= n; ++i) {
            scanf("%d", &h);
			update(1, h, query(1, 1, h)+1);
        }
        cout << query(1, 1, MXH) << endl;
    }
    return 0;
}
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

jpphy0

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值