题解:P12697 [KOI 2022 Round 2] 更换卡片
Nancy_Cherry · · 题解
题目简述
将序列修改为:单调递增等差数列、单调递减等差数列或常数序列(公差为0的等差数列),并使修改的卡片数量最少。
思路介绍
核心思路是找到与原序列匹配元素最多的等差数列,因为最少修改次数 = 总元素数 - 最大匹配数。
一个等差数列由首项
因此,公差
枚举可能的等差数列。遍历所有可能的元素对
对每个有效等差数列,遍历整个序列,统计满足
最少修改次数 = 总元素数
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
int main() {
int N;
cin >> N;
vector<int> x(N);
for (int i = 0; i < N; ++i) {
cin >> x[i];
}
int cnt = 1; // 最少有1个元素匹配(极端情况)
// 枚举所有可能的i < j对,确定等差数列的d和a1
for (int i = 0; i < N; ++i) {
for (int j = i + 1; j < N; ++j) {
int dx = x[j] - x[i];
int dj = j - i;
// 公差d必须为整数,否则跳过
if (dx % dj != 0) {
continue;
}
int d = dx / dj;
// 计算首项a1:第i项为a1 + i*d = x[i] → a1 = x[i] - i*d
int a1 = x[i] - i * d;
// 统计该等差数列在原序列中的匹配数
int cnt1 = 0;
for (int k = 0; k < N; ++k) {
if (x[k] == a1 + k * d) {
cnt1++;
}
}
// 更新最大匹配数
if (cnt1 > cnt) {
cnt = cnt1;
}
}
}
// 最少修改次数 = 总元素数 - 最大匹配数
cout << N - cnt << endl;
return 0;
}