最长公共子序列LCS


根据题目数据范围可以知道算法的时间复杂度应当是O(nlogn)。具体思路应当是将最长公共子序列问题转到最长上升子序列(LIS问题有时间复杂度O(nlogn))。具体可以参考这篇文章:最长上升子序列
例如:

求上面两个序列的最长公共子序列,可以将上面的的序列 1 6 5 4 3 2 转换成 1 2 3 4 5 6。即有1->1 , 6->2 , 5->3 , 4->4 , 3->5 , 2->6。的对应方法。最后转换成下面这个样子。

求两个序列的公共子序列就是求下面序列的最长上升子序列。下面的代码中translate()函数就是转换的方法。

代码如下:

#include<iostream>//根据题目范围知道这道题要用nlogn 的算法,用n^2的算法会超时,将LCS的问题转化为LIS的问题是解决本题的关键。
#include<algorithm>
using namespace std;
const int maxn = 100005;
int n;
int a[maxn], b[maxn],v[maxn],id[maxn];// id[x] : x被编成了几号
//int dp[maxn];
inline void input()//输入函数
{
	cin >> n;
	for (int i = 1;i <= n;i++)
		cin >> a[i];
	for (int i = 1;i <= n;i++)
		cin >> b[i];
}
inline int query()
{
	v[1] = a[1];
	int ans = 1;
	for (int i = 2;i <= n;i++)
	{
		if (a[i] > v[ans])
			v[++ans] = a[i];
		else//为了使上升子序列最长,应当使序列中的值之间相差尽可能的小。   *p=a[i];的目的就是舍弃之前的值,用新的a[i]来顶替它。
		{
			int* p = lower_bound(v + 1, v + 1 + ans, a[i]);
			*p = a[i];
		}
        
	}
	return ans;
	/*for(int i=1;i<=n;i++)//朴素做法,时间复杂度为n^2
		for (int j = n;j >=1;j--)
		{
			if (a[i] == b[j])
				dp[j] = max(dp[j - 1] + 1, dp[j]);
			else
				dp[j] = max(dp[j], dp[j - 1]);
		}
	return dp[n];*/
}
inline void translate()
{
	for (int i = 1;i <= n;i++)//将数字进行转化的精髓
		id[b[i]] = i;     
	for (int i = 1;i <= n;i++)
		a[i] = id[a[i]];
    /*for (int i = 1;i <= n;i++)
        b[i]=i;*/
}
int main()
{
	ios::sync_with_stdio(false);
	input();
	translate();
    /*for(int i=1;i<=n;i++)
    cout<<a[i]<<" ";
    cout<<endl;
     for(int i=1;i<=n;i++)
    cout<<b[i]<<" ";
    cout<<endl;*/
	int k = query();
	cout << k << endl;
}

以上是关于这道模板题的讲解。下面来对LCS问题进行分析。复杂度为O(n^2)。
对于长度为n的序列A和长度为m的序列B。
具体可以这样思考:
dp[i][j]是长度为i的序列A和长度为j的序列B的最长公共子序列。
1.A[i]==B[j],则有dp[i][j]=dp[i-1][j-1]+1;
2.A[i]!=B[j], 则有dp[i][j]=max(dp[i-1][j],dp[i][j-1]);
代码如下:

#include<iostream>
#include<algorithm>
using namespace std;
int A[105],B[105];
int dp[105][105];
int main()
{
    int n,m;
    cin>>n>>m;
    for(int i=1;i<=n;i++)
    cin>>A[i];
    for(int i=1;i<=m;i++)
    cin>>B[i];
    for(int i=1;i<=n;i++)
       for(int j=1;j<=m;j++)
       {
           if(A[i]==B[j])
               dp[i][j]=dp[i-1][j-1]+1;
           else
               dp[i][j]=max(dp[i-1][j],dp[i][j-1]);
       }
    cout<<dp[n][m]<<endl;
}

参考资料:
https://www.zhihu.com/question/23995189
https://www.luogu.com.cn/blog/pks-LOVING/junior-dynamic-programming-dong-tai-gui-hua-chu-bu-ge-zhong-zi-xu-lie

posted @ 2020-07-13 12:25  肥泽~  阅读(57)  评论(0编辑  收藏  举报