14届蓝桥杯 C/C++ B组 T5 接龙排序 (最长上升子序列DP+优化)

不难发现这是一个LIS问题,但是如果直接套用LIS的模版,在数据范围到达 1 e 5 1e5 1e5 的情况下,就只能够得到一半的分数,所以我们需要对其进行优化。

首先给出暴力的代码:

cpp 复制代码
#include<iostream>
using namespace std;
const int N = 1e5+10;

string a[N];	//为了方便比较数的首尾,直接用string类型存
int f[N];

int main(){
	int n;cin >> n;
	for(int i = 1;i <= n;i++)cin >> a[i];

	for(int i = 1;i <= n;i++){
		f[i] = 1;
		for(int j = 1;j < i;j++){
			if(a[i][0] == a[j][a[j].length() - 1])
				f[i] = max(f[i],f[j] + 1);
		}
	}
	
	int res = 0;
	for(int i = 1;i <= n;i++)res = max(res,f[i]);

	cout << n - res;
	return 0;
}

那么如何优化,注意到暴力程序只有一个地方达到了两层的循环,所以我们只要优化掉一层循环即可。

那么如何优化以下代码:

复制代码
for(int j = 1;j < i;j++){
	if(a[i][0] == a[j][a[j].length() - 1])
		f[i] = max(f[i],f[j] + 1);
}

此处代码写出来是为了枚举比较首尾,那么如果我们能够直接定位和a[i]的首部相同尾部的子序列的长度不就不需要判断了吗。

所以使用一个数组来存尾部是 1 1 1 ~ 9 9 9 中某一个数结尾的接龙子序列的最长长度,在状态转移时直接省掉了判断的步骤。

优化代码:

cpp 复制代码
#include<iostream>
#include<map>
using namespace std;
const int N = 1e5 + 10;

string a[N];
map<char,int>m;
int f[N];

int main() {
    int n; cin >> n;
    for (int i = 1; i <= n; i++)
        cin >> a[i];

    for (int i = 1; i <= n; i++) {
        f[i] = 1;
        f[i] = max(f[i],m[a[i][0]] + 1);
        m[a[i][a[i].length() - 1]] = max(f[i],m[a[i][a[i].length() - 1]]);//这里必须取max,因为f[i]不一定就更大
    }

    int res = 0;
    for(int i = 1;i <= n;i++)res = max(res,f[i]);
    cout << n - res;
    return 0;
}
相关推荐
王老师青少年编程5 分钟前
信奥赛C++提高组csp-s之组合数学专题课:鸽巢原理详解及案例实践
c++·组合数学·信奥赛·抽屉原理·csp-s·提高组·鸽巢原理
寒秋花开曾相惜9 分钟前
(学习笔记)3.8 指针运算(3.8.3 嵌套的数组& 3.8.4 定长数组)
java·开发语言·笔记·学习·算法
Гений.大天才10 分钟前
2026年计算机领域的年度主题与范式转移
算法
njidf38 分钟前
C++与Qt图形开发
开发语言·c++·算法
ZoeJoy839 分钟前
算法筑基(一):排序算法——从冒泡到快排,一文掌握最经典的排序算法
数据结构·算法·排序算法
qwehjk20081 小时前
代码动态生成技术
开发语言·c++·算法
承渊政道1 小时前
【优选算法】(实战体会位运算的逻辑思维)
数据结构·c++·笔记·学习·算法·leetcode·visual studio
Frostnova丶1 小时前
LeetCode 2573. 找出对应 LCP 矩阵的字符串
算法·leetcode·矩阵
m0_716765232 小时前
C++提高编程--STL常用容器(set/multiset、map/multimap容器)详解
java·开发语言·c++·经验分享·学习·青少年编程·visual studio
承渊政道2 小时前
【优选算法】(实战推演模拟算法的蕴含深意)
数据结构·c++·笔记·学习·算法·leetcode·排序算法