【题解】P1020 导弹拦截
题面
前言
本来开开心心敲个
我嘞个超绝
正文
首先嗷,这个题是可以
转化一下题意:第一问即求最长不上升子序列的长度,第二问即求最少能被划分成多少个不上升子序列
第一问
DP 是非常好想的,记
转移方程:
答案的计算也是显然的……形如:
但是捏,我们要追求一下
这个东西长得就很像决策单调性优化 DP 的套路
如何证明该 DP 转移有决策单调性?
记
显然有
不严格的证明
考虑使用反证法。假设,满足
考虑长度为的单调不升子序列,根据定义它以 结尾
而我们可以从该序列中构造出一个长度为的单调不升子序列,它的结尾同样是 ,那么由于 ,与 最大相矛盾,故假设不成立
嗯嗯,对于
现在考虑以
由于我们需要计算所有满足
考察这个
因此总有
以上我们证明了 DP 具有的决策单调性,采用二分法即可(当然 upper_bound
和 lower_bound
好用捏!)
时间复杂度
第二问
前置知识:Dilworth 定理
Dilworth 定理
最长反链最小链覆盖
然后第二问就进一步转化为求最长上升子序列的长度
同上即可!
代码
#include<iostream>
#include<algorithm>
#define int long long
using namespace std;
const int maxn=1e6+10;
int n=1,a[maxn];
int dp1[maxn],dp2[maxn];
int len1,len2;
signed main(){
while(cin>>a[n]){
n++;
}
n--;
len1=1;
len2=1;
dp1[1]=a[1];
dp2[1]=a[1];
for(int i=2;i<=n;i++){
if(a[i]<=dp1[len1]){
dp1[++len1]=a[i];
}else{
int p=upper_bound(dp1+1,dp1+len1+1,a[i],greater<int>())-dp1;
dp1[p]=a[i];
}
if(a[i]>dp2[len2]){
dp2[++len2]=a[i];
}else{
int p=lower_bound(dp2+1,dp2+len2+1,a[i])-dp2;
dp2[p]=a[i];
}
}
cout<<len1<<endl<<len2<<endl;
return 0;
}
后记
也许,云落真的很……
完结撒花!
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· winform 绘制太阳,地球,月球 运作规律
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· AI 智能体引爆开源社区「GitHub 热点速览」
· Manus的开源复刻OpenManus初探
· 写一个简单的SQL生成工具