要求找出范围内含有“13”且能被13整除的数字的个数

可以使用数位dp

dp[i][j][0] 表示长度为i,余数为j,不含13的数字的个数

dp[i][j][1] 表示长度为i,余数为j,3开头的数字的个数

dp[i][j][2] 表示长度为i,余数为j,含有"13"的数字的个数

index[1] = 1;

for(i=2; i<11; ++i)

  index[i] = (index[i-1] * 10) % 13;

index[i] 存储的为1,10,100,1000,10000 % 13 的余数

那么,状态转移方程详见源代码

 1 #include <stdio.h>
 2 int dp[11][13][3];
 3 int index[11];
 4 int num[11];
 5 void init()
 6 {
 7     int i,j,k;
 8     index[1] = 1;
 9     for(i=2; i<11; ++i)
10         index[i] = (index[i-1]*10) % 13;
11     dp[0][0][0] = 1;
12     for(i=1; i<11; ++i)
13     {
14         for(k=0; k<13; ++k)
15         {
16             //(1)
17             dp[i][(index[i]+k)%13][0] -= dp[i-1][k][1];
18             //长度为i-1,余数为k的不含13的数字前面加上3,-->长度为i-1,余数为k的3开头的个数
19             dp[i][(index[i]*3+k)%13][1] += dp[i-1][k][0];
20             //长度为i-1,余数为k的3开头的数字前面加上1-->长度为i,余数为(index[i]+k)%13含13的数字个数
21             dp[i][(index[i]+k)%13][2] += dp[i-1][k][1];
22             for(j=0; j<10; ++j)
23             {
24                 //长度为i-1,余数为k的不含13的数字前面加上j-->长度为i,余数为(index[i]*j+k)%13不含13的数字个数
25                 //但是dp[i-1][k][0] 里面是包含dp[i-1][k][1]的,当加上数字1时,成为了含有13的数字,这里多加,所以在(1)处减去
26                 dp[i][(index[i]*j+k)%13][0] += dp[i-1][k][0];
27                 //长度为i-1,余数为k的含13的数字前面加上j-->长度为i,余数为(index[i]*j+k)%13含13的数字个数
28                 dp[i][(index[i]*j+k)%13][2] += dp[i-1][k][2];
29             }
30         }
31     }
32 }
33 int getAns(int n)
34 {
35     int i,j,k,len=0,ans=0;
36     while(n)
37     {
38         num[++len] = n % 10;
39         n /= 10;
40     }
41     num[len+1] = 0;
42     bool flag = false;
43     int t = 0,mod;
44     for(i=len; i>=1; --i)
45     {
46         for(k=0; k<13; ++k)
47         {
48             if(num[i]>1 && !flag)//第i位取1
49             {
50                 mod = (index[i]+k+t)%13;//第i位取1时余数为mod
51                 if(mod==0) ans += dp[i-1][k][1];//如果余数为0,那么就加上3开头的数字个数
52             }
53             if(num[i+1]==1 && num[i]>3 &&!flag)//第i+1位为1,第i位取3.
54             {
55                 mod = (t + k) % 13;//第i+1位为1,第i位取3的余数为mod
56                 if(mod==0) ans += dp[i][k][1];//如果余数为0,那么就加上3开头的数字个数
57             }
58             for(j=0; j<num[i]; ++j)//第i位为j时,
59             {
60                 mod = (index[i]*j+k+t)%13;//第i位为j时余数为mod
61                 if(mod==0) ans += dp[i-1][k][2];//如果余数为0,那么就加上含有13的数字的个数
62                 if(mod==0 && flag) ans += dp[i-1][k][0];//如果余数为0,且前面的数字含有13,那么就加上不含13的数字个数
63             }
64         }
65         t = (t + num[i]*index[i])%13;//第len位到第i位的数字固定后产生的余数
66         if(num[i+1]==1 && num[i]==3)
67             flag = true;
68     }
69     return ans;
70 }
71 int main()
72 {
73     int n;
74     init();
75
76     while(scanf("%d",&n)!=EOF)
77     {
78         printf("%d\n",getAns(n+1));
79     }
80     return 0;
81 }

View Code

数位dp的难点就在于状态的转移,还有统计。

关键要弄懂它统计的原理。

http://www.cnblogs.com/justPassBy/p/4275226.html

转载于:https://www.cnblogs.com/justPassBy/p/4277263.html

hdu3652(数位dp)相关推荐

  1. [Hdu3652]B-number(数位DP)

    Description 题目大意:求小于n是13的倍数且含有'13'的数的个数. (1 <= n <= 1000000000) Solution 数位DP,题目需要包含13,且被13整除, ...

  2. 动态规划 —— 数位 DP

    [概述] 数位 DP 实际是一种计数用的 DP,一般就是统计一个区间 [le,ri] 内满足一些条件数的个数. 所求的限定条件往往与数的位数有关,例如:数位之和.指定数码个数.数的大小顺序分组等. 题 ...

  3. 【bzoj 1833】【codevs 1359】 [ZJOI2010]count 数字计数(数位dp)

    1833: [ZJOI2010]count 数字计数 Time Limit: 3 Sec  Memory Limit: 64 MB Submit: 2774  Solved: 1230 [Submit ...

  4. 不要62 ---数位DP

    题意:求m到n中不含62和4的数的个数. 题目链接 思路:数位dp模板求满足的数字或不满足的数字,刚学,就求不满足的数. #include<stdio.h> #include<str ...

  5. bzoj 3598 [ Scoi 2014 ] 方伯伯的商场之旅 ——数位DP

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3598 数位DP...东看西看:http://www.cnblogs.com/Artanis/ ...

  6. cojs 简单的数位DP 题解报告

    首先这道题真的是个数位DP 我们考虑所有的限制: 首先第六个限制和第二个限制是重复的,保留第二个限制即可 第五个限制在转移中可以判断,不用放在状态里 对于第一个限制,我们可以增加一维表示余数即可 对于 ...

  7. 数位DP 不断学习中。。。。

    1, HDU 2089  不要62 :http://acm.hdu.edu.cn/showproblem.php?pid=2089 题意:不能出现4,或者相邻的62, dp[i][0],表示不存在不吉 ...

  8. [数位dp] spoj 10738 Ra-One Numbers

    题意:给定x.y.为[x,y]之间有多少个数的偶数位和减去奇数位和等于一. 个位是第一位. 样例: 10=1-0=1 所以10是这种数 思路:数位dp[i][sum][ok] i位和为sum 是否含有 ...

  9. 数位dp(求1-n中数字1出现的个数)

    题意:求1-n的n个数字中1出现的个数. 解法:数位dp,dp[pre][now][equa] 记录着第pre位为now,equa表示前边是否有降数字(即后边可不能够任意取,true为没降,true为 ...

最新文章

  1. php自动打印小票_错题打印机哪个品牌质量好?【2020双12】错题打印机品牌排行...
  2. couchdb 任意命令执行漏洞(cve-2017-12636)
  3. android logger的使用
  4. ios自带NSURLConnection下载文件
  5. iOS开发:通过经纬度获得城市、省份等信息
  6. C# 中打印、预览、打印机设置和打印属性的方法
  7. python环境变量的配置_python基础教程-第一讲-带你进入python的世界
  8. 伯克利、OpenAI等提出基于模型的元策略优化强化学习
  9. 重新学习操作系统的知识
  10. 剑指offer——7.重建二叉树
  11. 把图像划分为patch以及用图像块重建图像
  12. PowerDesigner数据库设计工具
  13. 利用GPU加速的软件
  14. Error: Unable to execute “/usr/bin/vmware-uninstall-tools.pl.终极解决方案
  15. 鸿蒙系统4月24日升级哪几款手机,大器晚成!鸿蒙系统或4月24号推送更新,你的机型在内吗?...
  16. python math 数学函数教程
  17. 有python画螺旋线
  18. 【OpenGL】glm库的配置
  19. 搭建注册中心Eureka运行时报错:[ main] o.s.b.d.LoggingFailureAnalysisReporter :
  20. 从0开始搭建Web自动化测试框架

热门文章

  1. 定时器_Qt定时器小坑
  2. 计算机二级考试开考多久能出来,【计算机二级】明天就要开考了,你们准备好了吗?...
  3. jsp 中div居中_让div在屏幕中居中(水平居中+垂直居中)
  4. Java 集合容器篇面试题(上)-王者笔记《收藏版》
  5. 基于前端HTML+CSS+JS实现2022城市新年贺卡特效
  6. 字谜 java_java - 2字的字谜 - 堆栈内存溢出
  7. python发布_python网站发布
  8. 点击编辑框全选内容java_Android 中使用EditText 点击全选再次点击取消全选功能
  9. Java 在指定目录中创建文件
  10. Java自动类型转换和强制类型转换