Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)
Total Submission(s): 905 Accepted Submission(s): 324
今天,它发明了一个游戏:D游戏。
度度熊的英文并不是很高明,所以这里的D,没什么高深的含义,只是代指等差数列[(等差数列百科)](http://baike.baidu.com/view/62268.htm)中的公差D。
这个游戏是这样的,首先度度熊拥有一个公差集合{D},然后它依次写下N个数字排成一行。游戏规则很简单:
1. 在当前剩下的有序数组中选择X(X≥2) 个连续数字;
2. 检查1选择的X个数字是否构成等差数列,且公差 d∈{D};
3. 如果2满足,可以在数组中删除这X个数字;
4. 重复 1−3 步,直到无法删除更多数字。
度度熊最多能删掉多少个数字,如果它足够聪明的话?
每组数据以两个整数 N,M 开始 。接着的一行包括 N 个整数,表示排成一行的有序数组 Ai。接下来的一行是 M 个整数,即给定的公差集合 Di。
1≤N,M≤300
−1 000 000 000≤Ai,Di≤1 000 000 000
#include <map> #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> using namespace std; typedef long long ll; map<ll,int> s; ll T,n,m,can[310][310],f[310],a[310]; int main() { scanf("%lld",&T); while (T--) { s.clear(); memset(can,0,sizeof(can)); memset(f,0,sizeof(f)); scanf("%lld%lld",&n,&m); for (int i = 1; i <= n; i++) scanf("%lld",&a[i]); for (int i = 1; i <= m; i++) { ll x; scanf("%lld",&x); s[x] = 1; } for (int len = 2; len <= n; len++) { for (int i = 1; i + len - 1 <= n; i++) { int j = i + len - 1; if (len == 2) { if (s[a[j] - a[i]]) can[i][j] = 1; continue; } if (len == 3) { if (a[j] - a[j - 1] == a[j - 1] - a[i] && s[a[j] - a[j - 1]]) can[i][j] = 1; continue; } for (int k = i + 1; k < j - 1; k++) if (can[i][k] && can[k + 1][j]) can[i][j] = 1; if (can[i + 1][j - 1] && s[a[j] - a[i]]) can[i][j] = 1; for (int k = i + 3; k < j - 2; k++) if (can[i + 1][k - 1] && can[k + 1][j - 1] && a[j] - a[k] == a[k] - a[i] && s[a[j] - a[k]]) can[i][j] = 1; } } for (int i = 1; i <= n; i++) for (int j = 1; j < i; j++) { f[i] = max(f[i],f[i - 1]); if (can[j][i]) f[i] = max(f[j - 1] + i - j + 1,f[i]); } printf("%lld\n",f[n]); } return 0; }