中竞杯校赛 E.数论只会 for 循环 (数论分块 + 记忆化搜索)
题目链接:https://ac.nowcoder.com/acm/contest/9680/E
由定义式可以发现,如果 k>log(n),那么答案就一定是 0, 所以我们只需要计算 k<=27 的答案即可
记忆化搜索,使用 unorderedmap 存储 dp 数组的答案
对于每个块,如果左端点为 l, 那么对于 log(i), 右端点 r1=1<<log(i), 对于 ni,右端点为 r2=nnl
每次分块的右端点 r 取 r=min(r1,r2) 即可,具体实现参考代码
#include<iostream>
#include<algorithm>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<unordered_map>
using namespace std;
typedef long long ll;
const int M = 1000000007;
int n, k;
unordered_map<int, int> dp[30];
int dfs(int N, int K){
if(K == 0){
return dp[K][N] = 1;
}
if(dp[K].find(N) != dp[K].end()){
return dp[K][N];
}
int res = 0;
int l, r;
int kk = 0;
for(l = 2 ; l <= N ;){
while(1 << (kk + 1) < l) ++kk;
if((1 << kk) < l) ++kk;
r = min(min(N / (N / l), (1 << kk)), N);
res = (res + 1ll * (r - l + 1) * kk % M * dfs(N / l, K - 1) % M) % M;
l = r + 1;
}
return dp[K][N] = res;
}
ll read(){ ll s = 0, f = 1; char ch = getchar(); while(ch < '0' || ch > '9'){ if(ch == '-') f = -1; ch = getchar(); } while(ch >= '0' && ch <= '9'){ s = s * 10 + ch - '0'; ch = getchar(); } return s * f; }
int main(){
n = read(), k = read();
if(k >= 27) printf("0\n");
else {
printf("%d\n", dfs(n, k));
}
return 0;
}
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步