题目链接:拿糖果
思路:首先给小于根号n的素数打表。d(i)表示当前剩余i颗糖,最多可以拿到多少糖。
转移方程:d(i) = max(d(i), k + d(i - 2 * k)),此处k表示她可以从i颗糖中拿的糖数量,即k是素数并且i % k == 0。
注意:妈妈可以拿的糖的数量如果不足P,则拿糖结束。
AC代码:
#include <cstdio>
#include<cmath>
#include <algorithm>
#include <cstring>
#include <utility>
#include <string>
#include <iostream>
#include <map>
#include <set>
#include <vector>
#include <queue>
#include <stack>
using namespace std;
#define eps 1e-10
#define inf 0x3f3f3f3f
#define PI pair<int, int>
const int maxn = 1e5 + 5;
int vis[500], d[maxn];
void init(int n) {
//vis[i] = 0表示是素数
int m = sqrt(n + 0.5);
memset(vis, 0, sizeof(vis));
for(int i = 2; i <= m; ++i) if(!vis[i])
for(int j = i*i; j <= n; j += i) vis[j] = 1;
}
int dfs(int n) {
if(d[n]) return d[n];
if(n <= 3) return d[n] = 0;
int m = sqrt(n);
for(int i = 2; i <= m; ++i) {
if(!vis[i] && n % i == 0) { //质因数
d[n] = max(d[n], i + dfs(n - 2*i));
}
}
return d[n];
}
int main() {
init(500);
int n;
while(scanf("%d", &n) == 1) {
memset(d, 0, sizeof(d));
printf("%d\n", dfs(n));
}
return 0;
}
如有不当之处欢迎指出!