交叉匹配
Problem
给定两个正整数数列
我们想要对于给定的输入,找到画出最多匹配线段的方式,并且满足以下条件:
-
每条
-匹配线段恰好和一条 -匹配线段相交,且 ( , 表示任何值,并非某个特定的值)。我们称这样的匹配为交叉匹配。 -
不允许两条线段从同一个数出发,即不能从一点连出两条线段。
-
不允许一条线段和多条其它线段相交。
编一个程序对于给定输入数据,计算匹配线段的最多条数。
注意这个数总是偶数。
Input
输入第一行,两个整数
输入第二行,
输入第三行,
Output
输出仅一行,表示最多匹配线段的条数。
Sample
Input 1
6 6
1 3 1 3 1 3
3 1 3 1 3 1
Output 1
6
Input 2
12 11
1 2 3 3 2 4 1 5 3 5 10 4
3 1 2 3 2 4 12 1 5 5 3
Output 2
6
Solution
DP。
定义
假设当前在考虑
然后我们要来考虑交叉的情况。如果
代码:
#include <algorithm>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <map>
using namespace std;
const int kmax = 2005;
int n, m, a[kmax], b[kmax];
int f[kmax][kmax];
map<int, int> pa, pb; // 记录上一次出现的位置
int main() {
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++) {
f[i][j] = max(f[i][j - 1], f[i - 1][j]);
if (a[i] != b[j]) {
int ap = pa[b[j]], bp = pb[a[i]];
if (ap && bp) { // 存在上一个点
f[i][j] = max(f[i][j], f[ap - 1][bp - 1] + 2); // 转移
}
}
pb[b[j]] = j;
}
pa[a[i]] = i;
pb.clear(); // 要清空
}
cout << f[n][m] << '\n'; // 终态
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 单线程的Redis速度为什么快?
· 展开说说关于C#中ORM框架的用法!
· Pantheons:用 TypeScript 打造主流大模型对话的一站式集成库
· SQL Server 2025 AI相关能力初探
· 为什么 退出登录 或 修改密码 无法使 token 失效