最长公共子序列(上海交通大学考研机试题)

题目描述

给出两个长度为 n 的整数序列,求它们的最长公共子序列(LCS)的长度,保证第一个序列中所有元素都不重复。

注意:

第一个序列中的所有元素均不重复。

第二个序列中可能有重复元素。

一个序列中的某些元素可能不在另一个序列中出现。

输入样例

bash 复制代码
5
2 1 3 8 7
2 9 3 4 5

输入样例

bash 复制代码
2

数据范围

1 ≤ n ≤ 1 0 6 1≤n≤10^6 1≤n≤106, 序列内元素取值范围 [ 1 , 1 0 6 ] [1,10^6] [1,106]

分析

此题的数据量达到了1e6故不能用传统的 o ( n 2 ) o(n^2) o(n2)的dp做法,需考虑 o ( n l o g n ) o(nlogn) o(nlogn)的做法。

由于第一个序列中元素不重复,这是一个典型的最长公共子序列转换为最长上升子序列问题。


最长上升子序列求法 O ( n l o g n ) O(nlogn) O(nlogn)

首先我们看看最长公共子序列的求解过程是什么样子的?

A: 2 1 3 8 7
B: 2 9 3 4 5

ans = {2, 3};

就是从B中按照下标从小到大的顺序(从左至右)去A中找相同的数字,且在A中数字的下标也需要是递增的(也需要从左至右)

我们用另一种方式模拟这个过程

① 我们先将A中的数字和下标存储在idx数组中idx[key] = value,key 对应的是A中的值,value对应的是A中元素值对应的下标;

② 再按照坐标序遍历B中的元素,找到其在A中的位置idx[key], 找到一个我们就处理这个元素的下标到f数组中,将f数组维护为递增数组(刚刚我们说到A,B的子序列下标都需要是递增的);

f 数组维护的是B数组元素在A数组元素中的下标

时间复杂度

O ( n l o g n ) O(nlogn) O(nlogn)

C++ 代码

c 复制代码
#include <bits/stdc++.h>

using namespace std;

const int N = 1e6 + 7;

int idx[N], f[N];
int cnt;
int n;

int find(int x) {
    int l = 0, r = cnt;
    while (l < r) {
        int mid = l + r >> 1;
        if (f[mid] >= x) r = mid;
        else l = mid + 1;
    }
    
    return l;
}

int main() {
    ios::sync_with_stdio(false);
    cin.tie(0);
    
    cin >> n;
    memset(idx, -1, sizeof idx);
    
    for (int i = 1; i <= n; i ++ ) {
        int x;
        cin >> x;
        idx[x] = i;     // 记录数组中每个值的下标是多少
    }
    
    
    for (int i = 1; i <= n; i ++ ) {
        int x;
        cin >> x;
        
        int k = idx[x];
        if (k == -1) continue;
        
        if (k > f[cnt]) f[ ++ cnt ] = k;    // 如果当前下标大于f数组末尾元素下标就直接加入f数组
        else {
            int p = find(k);    // 找到大于等于k的第一个数的下标
            f[p] = k;           // 将下标的对应值替换掉
        }
    }
    
    cout << cnt << endl;
    return 0;
}
相关推荐
Shang180989357261 天前
THC63LVD1027D一款10位双链路LVDS信号中继器芯片,支持WUXGA分辨率视频数据传输THC63LVD1027支持30位数据通道方案
人工智能·考研·信息与通信·信号处理·thc63lvd1027d·thc63lvd1027
元亓亓亓2 天前
考研408--组成原理--day3--数字电路(补)&加减乘除
考研·408·组成原理·门电路·alu
蒙奇D索大2 天前
【计算机网络】考研408 | 数据链路层的“安全卫士”:探秘检错编码之奇偶校验码
经验分享·笔记·计算机网络·考研·改行学it
元亓亓亓3 天前
考研408--数据结构--day2--顺序表及其增删改查
数据结构·考研·顺序表·408
元亓亓亓3 天前
考研408--计算机网络--day4--组帧&差错控制&可靠传输
计算机网络·考研·数据链路层
aitoolhub4 天前
考研论文引用格式 AI 校验实操:工具合集 + 技术原理
c语言·人工智能·考研·aigc
南风微微吹4 天前
【管综】考研199管理类联考真题试卷及答案解析PDF电子版(2009-2025年)
考研·pdf
立志成为大牛的小牛4 天前
数据结构——四十一、分块查找(索引顺序查找)(王道408)
数据结构·学习·程序人生·考研·算法
蒙奇D索大4 天前
【计算机网络】[特殊字符] 408高频考点 | 数据链路层组帧:从字符计数到违规编码,一文学透四大实现方法
网络·笔记·学习·计算机网络·考研
蒙奇D索大4 天前
【算法】 递归实战应用:从暴力迭代到快速幂的优化之路
笔记·考研·算法·改行学it