luoguP1020 导弹拦截
题意
题目描述
某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统。但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度。某天,雷达捕捉到敌国的导弹来袭。由于该系统还在试用阶段,所以只有一套系统,因此有可能不能拦截所有的导弹。
输入导弹依次飞来的高度(雷达给出的高度数据是\le 50000≤50000的正整数),计算这套系统最多能拦截多少导弹,如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。
输入格式
11行,若干个整数(个数\le 100000≤100000)
输出格式
22行,每行一个整数,第一个数字表示这套系统最多能拦截多少导弹,第二个数字表示如果要拦截所有导弹最少要配备多少套这种导弹拦截系统。
输入输出样例
输入 #1
389 207 155 300 299 170 158 65
输出 #1
6
2
说明/提示
为了让大家更好地测试n方算法,本题开启spj,n方100分,nlogn200分
每点两问,按问给分
分析
LIS的两种优化(好像不用这个做也行....但我不会呀)
问题1: 显然,是求最长不上升子序列
问题2: 分析一下, 问需要多少个系统才能把导弹全部拦截。 一个系统所能拦截的导弹是“不上升的”, 所以我们手算一下样例, 发现有转折的地方(“155”->“300”), 是必须要多加一个系统的。 而“207”->“300”其实也是一个转折, 但上一个加的系统就能够拦截到“207”了, 所以,我们再多造几组数据,发现,这个就是一个最长上升子序列。(其实我也不会...)
这里主要讲讲它的优化, 我们看见题目明确的要nlogn算法, LIS的nlogn算法(我知道的)有树状数组优化和二分+贪心+类似栈的东西
树状数组优化: 因为它只添加,而且t数组也是只增不减的, 所以可以用树状数组做。 注意实现的细节: 求以x结尾的最长上升子序列的时候, 不能把x算上, 因为可能在之前出现过x。
#include<cstdio>
#include<string.h>
#include<algorithm>
using namespace std;
#define lowbit(x) (x&-x)
const int MAX = 50000+9;
const int N = 100000+9;
int ft[MAX], mx, num;
int a[N];
void add1(int x, int k) {while(x <= MAX) ft[x] = max(ft[x], k), x += lowbit(x);}
int query1(int x) {
int mx = -1;
x -= 1;//LIS:是上升! 所以不包括相等
while(x) {
mx = max(ft[x], mx);
x -= lowbit(x);
}
return mx;
}
void add2(int x, int k) {while(x) ft[x] = max(ft[x], k), x -= lowbit(x);}
int query2(int x) {//最长不上升序列: 可以相等
int mx = -1;
while(x <= MAX) {
mx = max(ft[x], mx);
x += lowbit(x);
}
return mx;
}
int main() {
int n = 0;
while(scanf("%d", &a[++n]) == 1) ;
--n;//多读了一个
mx = 1;
for(int i = 1; i <= n; i++) {
ft[a[i]] = query2(a[i]) + 1;
mx = max(ft[a[i]], mx);
add2(a[i], ft[a[i]]);
}
printf("%d\n", mx);
memset(ft, 0, sizeof(ft));
num = 0;
for(int i = 1; i <= n; i++) {//求LIS
ft[a[i]] = query1(a[i]) + 1;
num = max(num, ft[a[i]]);
add1(a[i], ft[a[i]]);
// printf("%d ", ft[a[i]]);
}
printf("%d", num);
return 0;
}
二分+贪心(这里只给出LIS的二分代码):
其实这个我也不知道为什么能A,有可能是因为我看它的代码的时间比较长, 背下来了吧(不然怎么自己打出来了呢?
定义f[i]: 长度为i的上升子序列的末尾元素的最小值, 贪心的思路, f[i]越小, 我们后面的选择就越多。显然, f是递增的(自己写下数据)
f[1] = a[1], len = 1,(len表示当前的最长长度)
接下来i从2开始向后枚举 如果a[i]>f[len] f[++len]=i;//放到后面 不然每次在f里面二分查找第一个大于等于a[i]的数来更新f数组,设它的下标为l
(因为f[l]为第一个大于等于a[i]的数, 所以f[l-1] < a[i], 所以我们是不可以用a[i]更新f[l-1]及l-1以前的, 而我们却可以更新的f[l], 这样改只会使f[l]更小或不变, 也就是使f[l]更优。) (过程: 要是f[mid]<a[i], 那么显然f[mid] 包括 f[mid-1]及以前的都不能更新, 所以l = mid+1。 如果f[mid] >= a[i], 那么l~mid是可能需要更新的,而mid的右边的f[]>a[i], 不能用a[i]更新f[], 所以r = mid)(注: 左闭右开
最后比较大小,f[l]=min(f[l],a[i])
int len = 1;
ft[1] = a[1];
for(int i = 2; i <= n; i++) {//求LIS
if(a[i] > ft[len]) ft[++len] = a[i];
else {
int l = 1, r = len, mid;
while(l < r) {
mid = (l+r)>>1;
if(ft[mid] >= a[i]) r = mid;
//其实如果ft[mid] == a[i], 按理说应该是l=mid,但那样循环不出来;如果这样写的话,l 最后也会拿到正解的
else l = mid+1;
//其实可以反着思考, 既要使二分能循环出来(l = mid+1)又要符合逻辑(ft[mid] < a[i]时 l = mid), 就只能这么写了
}
ft[l] = min(ft[l], a[i]);
}
}
printf("%d", len);
luoguP1020 导弹拦截的更多相关文章
- LuoguP1020 导弹拦截 (LIS)
最长不降和单升 #include <iostream> #include <cstdio> #include <cstring> #include <algo ...
- Luogu-P1020(导弹拦截)(DP,LIS ,二分优化)
Luogu-P1020(导弹拦截)(DP) 题意: 给n(n<=100000) 个数字,求最长不上升子序列的长度和最少的不上升子序列的个数. 分析: 第一问: 求最长不上升子序列有 O(n^2) ...
- HDU-1257 导弹拦截系统 http://acm.hdu.edu.cn/showproblem.php?pid=1257
Problem Description 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能超过前一发的高 ...
- AC日记——导弹拦截 洛谷 P1020 (dp+模拟)
题目描述 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度.某天,雷达捕捉到敌国的导弹 ...
- 【codevs1044】导弹拦截问题与Dilworth定理
题目描述 Description 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度.某 ...
- TYVJ P1020 导弹拦截 Label:水
题目描述 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度.某天,雷达捕捉到敌国的导弹 ...
- bzoj2044: 三维导弹拦截
Description 一场战争正在A国与B国之间如火如荼的展开. B国凭借其强大的经济实力开发出了无数的远程攻击导弹,B国的领导人希望,通过这些导弹直接毁灭A国的指挥部,从而取得战斗的胜利!当然,A ...
- nyoj 79 导弹拦截
点击打开链接 拦截导弹 时间限制:3000 ms | 内存限制:65535 KB 难度:3 描述 某国为了防御敌国的导弹袭击,发展中一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发 ...
- UESTC_导弹拦截 2015 UESTC Training for Dynamic Programming<Problem N>
N - 导弹拦截 Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Submit ...
随机推荐
- vue引入ElementUI库
element国内网站:https://element.eleme.cn/#/zh-CN 引入ElementUI命令:npm install element-ui --save (网速不好用cnp ...
- Ubuntu 18.04安装MySQL
安装 MySQL 服务端 sudo apt-get install mysql-server 等待安装完成. 检查 mysql 服务状态 servive mysql status 登录 mysql 客 ...
- leaflet-webpack 入门开发系列五地图卷帘(附源码下载)
前言 leaflet-webpack 入门开发系列环境知识点了解: node 安装包下载webpack 打包管理工具需要依赖 node 环境,所以 node 安装包必须安装,上面链接是官网下载地址 w ...
- 从0系统学Android--3.1编写UI界面
从0系统学Android--3.1编写UI界面 本系列文章目录:更多精品文章分类 本系列持续更新中.... 界面设计和功能开发同样重要,界面美观的应用程序不仅可以大大增加用户粘性,还能帮我们吸引到更多 ...
- 一文解读JSON (转)
JSON作为目前Web主流的数据交换格式,是每个IT技术人员都必须要了解的一种数据交换格式.尤其是在Ajax和REST技术的大行其道的当今,JSON无疑成为了数据交换格式的首选! 今天我们一起来学习一 ...
- 为什么 netstat 对某些服务只显示了 tcp6 监听端口
最近偶尔发现一个比较奇怪的现象,netstat 查看监听的服务端口时,却只显示了 tcp6 的监控, 但是服务明明是可以通过 tcp4 的 ipv4 地址访问的,那为什么没有显示 tcp4 的监听呢? ...
- nfs存储服务器
1.nfs的基础简介 1.1:什么是nfs? 它的主要功能是通过网络让不同的机器系统之间可以彼此共享文件和目录.NFS服务器可以允许NFS客户端将远端NFS服务器端的共享目录挂载到本地的NFS客户端中 ...
- 【pat】C++之刷题常用STL容器整理
1.vector 动态数组,方便的动态扩容,方便的变量初始化(int类型默认初始化为0,bool默认初始化为false),可以用来实现邻接表(结点数太多的图). 头文件 #include<vec ...
- Java描述设计模式(11):观察者模式
本文源码:GitHub·点这里 || GitEE·点这里 一.观察者模式 1.概念描述 观察者模式是对象的行为模式,又叫发布-订阅(Publish/Subscribe)模式.观察者模式定义了一种一对多 ...
- 分布式应用监控: SkyWalking 快速接入实践
分布式应用,会存在各种问题.而要解决这些难题,除了要应用自己做一些监控埋点外,还应该有一些外围的系统进行主动探测,主动发现. APM工具就是干这活的,SkyWalking 是国人开源的一款优秀的APM ...