最长递增子序列-动态规划dp-51node

教程地址:点击打开链接


给出长度为N的数组,找出这个数组的最长递增子序列。(递增子序列是指,子序列的元素是递增的)

例如:5 1 6 8 2 4 5 10,最长递增子序列是1 2 4 5 10。
输入

第1行:1个数N,N为序列的长度(2 <= N <= 50000)
第2 - N + 1行:每行1个数,对应序列的元素(-10^9 <= S[i] <= 10^9)

输出

输出最长递增子序列的长度。

输入示例

8
5
1
6
8
2
4
5
10

输出示例

5

【解法1】:

普通dp解法,时间复杂度O(n^2)

dp[i] 表示前i个序列最长子序列

状态方程dp[i]=max(dp[i],dp[j]+1);   0<j<i;条件判断(a[j]<a[i]);

[cpp]  view plain  copy
  1. #include <iostream>    
  2. #include <algorithm>   
  3. using namespace std;  
  4. int main()  
  5. {  
  6.     int n,dp[50050],a[50050];  
  7.     while(cin>>n)  
  8.     {  
  9.         for(int i=1;i<=n;i++)  
  10.             cin>>a[i];  
  11.         for(int i=1;i<=n;i++)  
  12.         {  
  13.             dp[i]=1;  
  14.             for(int j=1;j<i;j++)  
  15.                 if(a[j]<a[i]&&dp[i]<dp[j]+1)  
  16.                     dp[i]=dp[j]+1;  
  17.         }  
  18.         int Max=0;  
  19.         for(int i=1;i<=n;i++)  
  20.             Max=max(Max,dp[i]);  
  21.         cout<<Max<<endl;  
  22.     }  
  23. }  

【解法2】:

dp+二分查找。时间复杂度O(n*log(n))

dp[j],递增序列长度为j时,最后一项最小是多少

每次加入一个元素,找到这个元素能接的长度。比如dp[3]=6,那可以接7(8,9...),然后更新dp[4]=7;

而且以此做法,dp始终会是递增的。

[cpp]  view plain  copy
  1. #include <iostream>    
  2. #include <algorithm>   
  3. using namespace std;  
  4. int main()  
  5. {  
  6.     int n,dp[50050],a[50050];  
  7.     while(cin>>n)  
  8.     {  
  9.         for(int i=1;i<=n;i++)  
  10.             cin>>a[i];  
  11.         dp[1]=a[1];  
  12.         int top=1;  
  13.         for(int i=2;i<=n;i++)  
  14.         {  
  15.             if(a[i]<dp[1])dp[1]=a[i];//遇到了当前最小的数   
  16.             else if(a[i]>dp[top])dp[++top]=a[i];//遇到了一个当前最大的数   
  17.             else{  
  18.                 int l=1,r=top;//二分查找  
  19.                 while(l<r)  
  20.                 {  
  21.                     int mid=(l+r)/2;  
  22.                     if(dp[mid]<=a[i])  
  23.                         l=mid+1;  
  24.                     else r=mid;  
  25.                 }  
  26.                 dp[l]=a[i];  
  27.             }  
  28.         }  
  29.         cout<<top<<endl;  
  30.     }  
  31. }  
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

雪的期许

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

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

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

打赏作者

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

抵扣说明:

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

余额充值