动态规划之子序列问题

文章目录

        • 一.最长递增子序列I
        • 二. 最长递增子序列II
        • 二.数字和最大的递增子序列
        • 三.连续子数组最大和
        • 四.最长公共子序列

一.最长递增子序列I

题目描述

给定一个长度为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;
}

二. 最长递增子序列II

给定一个长度为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;
}

你可能感兴趣的:(动态规划)