[51NOD1230]幸运数(数位DP)

题目链接:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1230

dp(l,s,ss)表示长度为l的数各位和为s,各位平方和为ss的幸运数的个数。

 1 #include <bits/stdc++.h>
 2 #pragma comment(linker, "/STACK:10240000,10240000")
 3 using namespace std;
 4
 5 typedef long long LL;
 6 const int maxn = 22;
 7 const int maxa = 162;
 8 const int maxm = 1610;
 9 LL l, r;
10 LL dp[maxn][maxa][maxm];
11 int digit[maxn];
12 bool isprime[maxm];
13
14 void printlist() {
15   memset(isprime, true, sizeof(isprime));
16   isprime[0] = isprime[1] = false;
17   int pedge = int(sqrt(maxm));
18   for(int i = 2; i <= pedge; i++) {
19     if(isprime[i]) {
20       int o = maxm / i;
21       for(int j = 2; j <= o; j++) {
22         isprime[i*j] = false;
23       }
24     }
25   }
26 }
27
28 LL dfs(int l, int s, int ss, bool flag) {
29   if(l == 0) return isprime[s] && isprime[ss];
30   if(!flag && ~dp[l][s][ss]) return dp[l][s][ss];
31   int pos = flag ? digit[l] : 9;
32   LL ret = 0;
33   for(int i = 0; i <= pos; i++) {
34     ret += dfs(l-1, s+i, ss+i*i, flag&&(pos==i));
35   }
36   if(!flag) dp[l][s][ss] = ret;
37   return ret;
38 }
39
40 LL f(LL x) {
41   if(x < 0) return 0;
42   int pos = 0;
43   while(x) {
44     digit[++pos] = x % 10;
45     x /= 10;
46   }
47   return dfs(pos, 0, 0, true);
48 }
49
50 int main() {
51   //freopen("in", "r", stdin);
52   printlist();
53   memset(dp, -1, sizeof(dp));
54   int T;
55   scanf("%d", &T);
56   while(T--) {
57     scanf("%lld%lld",&l,&r);
58     printf("%lld\n", f(r)-f(l-1));
59   }
60   return 0;
61 }
时间: 2024-11-01 19:31:16

[51NOD1230]幸运数(数位DP)的相关文章

uestc250windy数数位dp

#include <cstdio> #include <cstring> #include <cmath> #include <algorithm> #include <climits> #include <string> #include <iostream> #include <map> #include <cstdlib> #include <list> #include <

USETC 250 windy数 数位DP

注意处理数字只有一位的情况(其实不用怎么处理)= = 简单数位DP #include <cstdio> #include <cstring> #include <cmath> #include <algorithm> #include <climits> #include <string> #include <iostream> #include <map> #include <cstdlib> #

bzoj1026: [SCOI2009]windy数 数位dp

题目: http://www.lydsy.com/JudgeOnline/problem.php?id=1026 题意: Description windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道, 在A和B之间,包括A和B,总共有多少个windy数? Input 包含两个整数,A B. Output 一个整数 思路: 数位dp,记忆化搜索. 1 #include <bits/stdc++.h> 2 3 using namesp

1043 幸运号码 数位DP

http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1043 设dp[i][j]表示前i位数中,i位数的和为j时的所有情况. 转移的时候和普通的数位dp是一样转移的,但是如果你压缩了空间的话,就是用滚动数组的话,记录情况数就要多开一个变量来保存, 然后看看怎么排除前导0的情况. 如果产生的和值是j,然后前i - 1位产生的和值也是j,那么第i为就是前导0了.需要排除. 然后前n部分的和值的所有情况(需要排除前导0) * 后n

【BZOJ-1026】windy数 数位DP

1026: [SCOI2009]windy数 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 5230  Solved: 2353[Submit][Status][Discuss] Description windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道,在A和B之间,包括A和B,总共有多少个windy数? Input 包含两个整数,A B. Output 一个整数 Sample I

[BZOJ 1026][SCOI 2009]windy数(数位DP)

题目链接:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1026 很基础的数位DP题,很早之前我就尝试做这题,不过当时我被这题吓死了,现在回过头做这题,感觉简单多了. 做这个题时我想到了POJ一道类似的组合数学的题,同样是按数位统计,有异曲同工之妙. 题目要求[a,b]区间上的windy数个数,我们可以转化成求[1,a]上的windy数个数-[1,b-1]上的windy数个数.题目转化成了求[1,x]上的windy数个数,我们就写个函数c

【BZOJ1662】[Usaco2006 Nov]Round Numbers 圆环数 数位DP

[BZOJ1662][Usaco2006 Nov]Round Numbers 圆环数 Description 正如你所知,奶牛们没有手指以至于不能玩"石头剪刀布"来任意地决定例如谁先挤奶的顺序.她们甚至也不能通过仍硬币的方式. 所以她们通过"round number"竞赛的方式.第一头牛选取一个整数,小于20亿.第二头牛也这样选取一个整数.如果这两个数都是 "round numbers",那么第一头牛获胜,否则第二头牛获胜. 如果一个正整数N的二

bzoj 1026 [SCOI2009]windy数 数位dp

1026: [SCOI2009]windy数 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/problem.php?id=1026 Description windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道,在A和B之间,包括A和B,总共有多少个windy数? Input 包含两个整数,A B. Output 一个整数.

CCF 201312-4 有趣的数 (数位DP, 状压DP, 组合数学+暴力枚举, 推公式, 矩阵快速幂)

问题描述 我们把一个数称为有趣的,当且仅当: 1. 它的数字只包含0, 1, 2, 3,且这四个数字都出现过至少一次. 2. 所有的0都出现在所有的1之前,而所有的2都出现在所有的3之前. 3. 最高位数字不为0. 因此,符合我们定义的最小的有趣的数是2013.除此以外,4位的有趣的数还有两个:2031和2301. 请计算恰好有n位的有趣的数的个数.由于答案可能非常大,只需要输出答案除以1000000007的余数. 输入格式 输入只有一行,包括恰好一个正整数n (4 ≤ n ≤ 1000). 输