链接:hdu 5087
题意:求第二大的最长升序子序列
分析:这里的第二大指的是,全部的递增子序列的长度(包含相等的),
从大到小排序后。排在第二的长度
思路1.每次将最长的两个上升子序列长度记录。最后再排序,取第二大的就可以
思路2.假设最长的上升子序列长度(ans)唯一,那第二大应为ans-1
否则,第二大的就为 ans
-
#include<stdio.h>
-
#include<algorithm>
-
using namespace std;
-
int a[1010],dp[1010][2],ans[2010],n;
-
int main()
-
{
-
int T,i,j,k,x,y;
-
scanf("%d",&T);
-
while(T--){
-
scanf("%d",&n);
-
for(i=0;i<n;i++)
-
scanf("%d",&a[i]);
-
k=0;
-
for(i=0;i<n;i++){
-
dp[i][0]=1;
-
dp[i][1]=0;
-
for(j=0;j<i;j++){
-
if(a[j]<a[i]){
-
x=dp[j][0]+1;
-
y=dp[j][1]+1;
-
if(x>dp[i][0])
-
swap(x,dp[i][0]);
-
if(x>y)
-
swap(x,y);
-
if(y>dp[i][1])
-
dp[i][1]=y;
-
}
-
}
-
ans[k++]=dp[i][0];
-
ans[k++]=dp[i][1];
-
}
-
sort(ans,ans+k);
-
printf("%d\n",ans[k-2]);
-
}
-
return 0;
-
}
-
#include<stdio.h>
-
#include<string.h>
-
#define max(a,b) a>b?
a:b
-
int a[1010],dp[1010],num[1010],n;
-
int main()
-
{
-
int T,i,j,cnt,ans;
-
scanf("%d",&T);
-
while(T--){
-
scanf("%d",&n);
-
for(i=0;i<n;i++)
-
scanf("%d",&a[i]);
-
ans=1;
-
for(i=0;i<n;i++){
-
num[i]=dp[i]=1;
-
for(j=0;j<i;j++){
-
if(a[j]<a[i]&&dp[j]+1>dp[i]){
-
dp[i]=dp[j]+1;
-
num[i]=num[j];
-
}
-
else if(a[j]<a[i]&&dp[j]+1==dp[i])
-
num[i]+=num[j];
-
}
-
ans=max(ans,dp[i]);
-
}
-
cnt=0;
-
for(i=0;i<n;i++)
-
if(dp[i]==ans)
-
cnt+=num[i];
-
if(cnt==1)
-
printf("%d\n",ans-1);
-
else
-
printf("%d\n",ans);
-
}
-
return 0;
-
}