题目描述
给定一个长度为N的数列,求数值严格单调递增的子序列的长度最长是多少。
输入格式
第一行包含整数N。
第二行包含N个整数,表示完整序列。
输出格式
输出一个整数,表示最大长度。
数据范围
1≤N≤1000,
−109≤数列中的数≤109
输入样例:
7
3 1 2 1 8 5 6
输出样例:
4
思路:
采用动态规划的思想,用dp[ i ]来表示序列中第i个元素结尾的最长上升子序列的长度那么状态转移方程为:
if (a[i] > a[j]) dp[i] = MAX (dp[i], dp[j] + 1);
#include
using namespace std;
const int N = 1010;
int d[N], a[N];
int main()
{
int n;
cin >> n;
for(int i = 0; i < n; i++)
cin >> a[i];
int ans = 0;
for(int i = 0; i < n; i++){
d[i] = 1;
for(int j = 0; j < i; j++){
if(a[i] > a[j])
d[i] = max(d[i], d[j] + 1);
}
ans = max(ans, d[i]);
}
cout << ans << endl;
return 0;
}
给定一个长度为N的数列,求数值严格单调递增的子序列的长度最长是多少。
输入格式
第一行包含整数N。
第二行包含N个整数,表示完整序列。
输出格式
输出一个整数,表示最大长度。
数据范围
1≤N≤100000
−109≤数列中的数≤109
输入样例:
7
3 1 2 1 8 5 6
输出样例:
4
题解:
这里的数据范围很大,不能再用上面的那个方法来做这道题
nlgn做法 定义:a[1…n]为原始序列,d[k]表示长度为k的不下降子序列末尾元素的最小值,len表示当前已知的最长子序列的长度
考虑两个数a[x]和a[y],x
按dp[t]=k来分类,只需保留dp[t]=k的所有a[t]中的最小值,设d[k]记录这个值,d[k]=min{a[t],dp[t]=k}。
这时注意到d的两个特点(重要):
**1. d[k]在计算过程中单调不升; **
2. d数组是有序的,d[1]
** 利用这两个性质,可以很方便的求解:**
1.设当前已求出的最长上升子序列的长度为len(初始时为1),每次读入一个新元素x:
2.若x>d[len],则直接加入到d的末尾,且len++;(利用性质2)
3.否则,在d中二分查找,找到第一个比x小的数d[k],并d[k+1]=x,在这里x<=d[k+1]一定成立(性质1,2)。
#include
using namespace std;
const int N = 1e5 +10;
int a[N], q[N];
int main()
{
int n;
cin >> n;
for(int i = 0; i <n; i++)
cin >> a[i];
int len = 0;
for(int i = 0; i < n; i++){
int l = 0, r = len;
while(l < r){
int mid = l + r + 1 >> 1;
if(q[mid] < a[i])l = mid; //找到小于a[i]的第一个数
else r = mid - 1;
}
//r + 1 表示大于等于a[i]的的第一个数,并将该数替换掉
len = max(len, r + 1);
q[r + 1] = a[i];
}
cout << len << endl;
}
题目描述:给定你一个数字序列,找出数字和最大的递增子序列
思路:跟前一个思路一样采用dp的思想,现在dp【i】的含义表示为序列中第i个元素结尾的数字和的最大值
#include
#include
#include
#define maxn 100000
int max(int a,int b)
{
if(a>b)return a;
else return b;
}
int a[maxn+5];
int dp[maxn+5];
int main()
{
int n,Max=0;
scanf("%d",&n);
for(int i=0;i<n;i++){
scanf("%d",&a[i]);
dp[i]=a[i];
}
for(int i=0;i<n;i++){
for(int j=0;j<i;j++){
if(a[i]>a[j])dp[i] = max(dp[j]+a[i], dp[i]);//dp【i】表示以第i个元素结尾的数字和的最大值
}
Max=max(Max,dp[i]); //记录最大的数字和
}
printf("%d\n",Max);
return 0;
}
例题:杭电OJ 1003
问题描述:给你一个数字序列,里面有正数,也有负数,求出这个序列中连续某一段的和的最大值,并求出是从什么地方开始到什么地方结束
思路:dp的思想,dp【i】依然表示的是序列中以第i个元素结尾的连续数组的最大值,从第二项依次遍历,如果前一项<0,则不加,若大于0就加上.
#include
#define maxn 100000
int a[maxn+5];
int main()
{
int t,n,p=1;
scanf("%d",&t);
while(t--){
a[0]=0;
int start=1,end=0,fstart=1,fend=1,Max=-1e9;
scanf("%d",&n);
for(int i=1;i<=n;i++)scanf("%d",&a[i]);
for(int i=1;i<=n;i++){
if(a[i-1]<0){
start=i;
end=i;
}
else {
a[i]+=a[i-1];
end=i;
}
if(a[i]>Max){ //Max,fstart,fend分别表示和的最大值以及对应的起始位置
Max=a[i];
fstart=start;
fend=end;
}
}
printf("Case %d:\n",p);
printf("%d %d %d\n",Max,fstart,fend);
p++;
if(t!=0)printf("\n");
}
}
Description
咱们就不拐弯抹角了,如题,需要你做的就是写一个程序,得出最长公共子序列。
tip:最长公共子序列也称作最长公共子串(不要求连续),英文缩写为LCS(Longest Common Subsequence)。
其定义是,一个序列 S ,如果分别是两个或多个已知序列的子序列,且是所有符合此条件序列中最长的,则 S 称为已知序列的最长公共子序列。
Input
第一行给出一个整数N(0
接下来每组数据两行,分别为待测的两组字符串。每个字符串长度不大于1000.
Output
每组测试数据输出一个整数,表示最长公共子序列长度。每组结果占一行。
Sample Input
2
asdf
adfsd
123abc
abc123abc
Sample Output
3
6
题解:
动态规划,dp[ i ] [ j ]表示a的前i个字符,b的前j个字符中的最大公共子序列长度。
状态转移方程:
当a[i] 和b[j]相同时,多了一条转移路线。
#include
#include
#include
int max(int a,int b)
{
if(a>b)return a;
else return b;
}
int main()
{
int t;
char a[1000],b[1000];
int c[1010][1010];
scanf("%d",&t);
while(t--)
{
scanf("%s%s",&a,&b);
memset(c,0,sizeof(c));
for(int i=1;i<=strlen(a);i++){
for(int j=1;j<=strlen(b);j++)
if(a[i-1]==b[j-1])c[i][j]=c[i-1][j-1]+1;
else c[i][j]=max(c[i][j-1],c[i-1][j]);
}
printf("%d\n",c[strlen(a)][strlen(b)]);
}
return 0;
}