题目:
给定大小为 1≤N≤1e41≤N≤1e4 的多重集合 AA,每个元素。求由集合元素构成的最长等差数列长度。
分析:
开局先排序,剩下全靠猜。一看 1e41e4,就想 O(N2)O(N2) 暴力一下,结果写着写着就变成 O(N2logN)O(N2logN) 了,还感觉挺有道理:设 dp[i][j]dp[i][j] 为以 ii 结尾公差为 的最长等差数列长度。对于一个位置 ii,枚举 ,设 d=ai−ajd=ai−aj,则记 dp[i][d]=maxj<i{dp[j][d]+1}dp[i][d]=maxj<i{dp[j][d]+1},然而这个 dd 显然很大,需要离散化一下,于是加一个 就 TLE 了。
题解教做人:记 dp[i][j]dp[i][j] 为以 i,ji,j 结尾的最长等差数列长度(一看就很靠谱的样子)。对于每个元素 aiai,记录双指针 l,rl,r,代表 al,ai,ajal,ai,aj 构成了等差数列,则当 ll 向左移动时, 显然应该向右移动。于是就变成枚举 ll,滑 ,同时用 dp[l][i]dp[l][i] 更新 dp[i][r]dp[i][r]。这样复杂度就变成了 O(N2)O(N2),注意初始化为所有 dp[i][j]=2dp[i][j]=2。
优化:对于一个 dd,最长等差数列不会超过 ,所以更新 dp[i][j]dp[i][j] 时记录一下当前最大值,可以进行剪枝。实测效果良好。至于如何开 1e4×1e41e4×1e4 的数组,那就只有瞎搞了。比如说开 short, 比如说开一半等。
代码:
#include <bits/stdc++.h>
using namespace std;
const int N_Max = 1e4 + 10;
int N;
short dp[N_Max][N_Max];
int arr[N_Max];
inline short Max(short a, short b) {
return a > b ? a : b;
}
int main() {
scanf("%d", &N);
for (int i = 1; i <= N; i++) {
scanf("%d", arr + i);
}
sort(arr + 1, arr + N + 1);
int ans = 0;
for (int i = 1; i < N; i++) {
for (int j = i + 1; j <= N; j++) {
dp[i][j] = 2;
ans = Max(ans, dp[i][j]);
}
}
int bound = arr[N] - arr[1];
for (int i = 2; i < N; i++) {
int l = i - 1, r = i + 1;
while (l >= 1 && r <= N) {
if (arr[l] + arr[r] > 2 * arr[i]) l--;
else if (arr[l] + arr[r] < 2 * arr[i]) r++;
else {
dp[i][r] = Max(dp[i][r], dp[l][i] + 1);
ans = Max(ans, dp[i][r]);
if (bound / (arr[r] - arr[i]) <= ans) break;
if (l == 1) r++;
else l--;
}
}
}
printf("%d\n", ans);
}