P2602 [ZJOI2010]数字计数

题目描述

给定两个正整数 \(a\) 和 \(b\) ,求在 \([a,b]\) 中的所有整数中,每个数码(digit)各出现了多少次。

输入格式

输入文件中仅包含一行两个整数\(a,b\),含义如上所述。

输出格式

输出文件中包含一行 \(10\) 个整数,分别表示 \(0-9\) 在 \([a,b]\) 中出现了多少次。

说明/提示

\(30\%\)的数据中,\(a<=b<=10^6\);

\(100\%\)的数据中,\(a<=b<=10^{12}\)。


Solution

此题解仅讲想法,不讲有关数位 \(dp\)的基础知识或写法,如果还没有学过数位 \(dp\) 的可以先看别的题目

看到题解里用 \(dfs\) 做的都是设的两维或以上的状态,实际上这道题只需要一维状态就够了

设目前已经填到了第 \(pos\) 位,则不管 \(num\ -\!-\ \ pos + 1\) 位上填的是什么,之后的 \(1\ -\!-\ pos\)位上的贡献是不变的(除非是已经到了 \(limit\) 的限制了,这个之后再讨论)

那么状态很明显为 \(f[pos]\)

\(eg:\)现在要填五位的数,目前状态为 \(12XXX\), \(limit\) 为 \(30000\),则后面的三位可以直接由 \(f[3]\) 转移过来,因为这属于子结构,不对前面造成影响

为什么可以这样转移?

前面所填的数(类似于 \(eg\) 中的 \(12XXX\) 的 \(12\))的贡献如何计算?

我们可以发现,对于 \(12XXX\) 中 第四位上的 \(2\) 的贡献,是 \(10^{pos - 1}\) 的。因为 \(12XXX\) 的后三位可以填 \(000\) - \(999\) 中的任意一种,则第四位的 \(2\) 就被计算了 \(10^3\) 次,即贡献就是 \(10^{pos - 1}\)。注意:这里所讨论的 \(2\) 的贡献值,仅考虑第四位上的 \(2\) ,对于后面位置上的为子结构,在之后会考虑到,而前面位置上的,在之前已经预先考虑过了,所以不会重复也不会漏情况。

现在再来讨论 \(limit\) 的限制情况。

假设将 \(eg\) 中的 \(limit\) 改为 \(12300\),则填后三位时就只能填 \(000\) - \(300\),总共是 \(12300 - 12000 + 1\) 种,于是只用在计算贡献时加这样一个判断就可以了。


Code

#include<bits/stdc++.h>
#define ll long long
#define F(i, x, y) for(int i = x; i <= y; ++ i)
using namespace std;
const int N = 15;
ll L, R;
int cnt[N];
ll f[N];
ll add(int pos)//计算lim限制时的贡献
{
ll ans = 0;
for(int i = pos - 1; i >= 1; -- i) ans = ans * 10 + cnt[i];
return ans + 1;
}
ll dp(int pos, int x, int lim, int last)
{/* pos为第几位 x为现在在算的数码
lim为是否为限制 last为上一次的值(处理前导零)*/
if(! pos) return 0;
if(! lim && f[pos] != -1 && last != 10) return f[pos];
ll ret = 0;
F(i, (last == 10 ? 1 : 0), (lim ? cnt[pos] : 9))
{
if(i == x && (i != cnt[pos] || ! lim)) ret += pow(10, pos - 1);
else if(i == x) ret += add(pos);//分情况计算贡献
ret += dp(pos - 1, x, lim && i == cnt[pos], i);
}
if(last == 10) ret += dp(pos - 1, x, 0, last);
if(! lim) f[pos] = ret;
return ret;
}
ll work(int x, ll r)
{
memset(f, -1, sizeof(f));
int num = 0;
for(r; r; r /= 10) cnt[++ num] = r % 10;
return dp(num, x, 1, 10);
}
int main()
{
cin >> L >> R;
F(i, 0, 9) printf("%lld ", work(i, R) - work(i, L - 1));
printf("\n");
return 0;
}

Thanks

如果有任何疑问欢迎提出或和我一起讨论(′▽`〃)

【题解】P2602 数字计数 - 数位dp的更多相关文章

  1. 洛谷 - P2602 - 数字计数 - 数位dp

    https://www.luogu.org/problemnew/show/P2602 第二道数位dp,因为“数位dp都是模板题”(误),所以是从第一道的基础上面改的. 核心思想就是分类讨论,分不同情 ...

  2. [bzoj1833][ZJOI2010]count 数字计数——数位dp

    题目: (传送门)[http://www.lydsy.com/JudgeOnline/problem.php?id=1833] 题解: 第一次接触数位dp,真的是恶心. 首先翻阅了很多很多一维dp,因 ...

  3. Luogu P2602 [ZJOI2010]数字计数 数位DP

    很久以前就...但是一直咕咕咕 思路:数位$DP$ 提交:1次 题解:见代码 #include<cstdio> #include<iostream> #include<c ...

  4. 洛谷P2602 [ZJOI2010]数字计数(数位dp)

    数字计数 题目传送门 解题思路 用\(dp[i][j][k]\)来表示长度为\(i\)且以\(j\)为开头的数里\(k\)出现的次数. 则转移方程式为:\(dp[i][j][k] += \sum_{t ...

  5. bzoj1833: [ZJOI2010]count 数字计数(数位DP+记忆化搜索)

    1833: [ZJOI2010]count 数字计数 题目:传送门 题解: 今天是躲不开各种恶心DP了??? %爆靖大佬啊!!! 据说是数位DP裸题...emmm学吧学吧 感觉记忆化搜索特别强: 定义 ...

  6. 【BZOJ-1833】count数字计数 数位DP

    1833: [ZJOI2010]count 数字计数 Time Limit: 3 Sec  Memory Limit: 64 MBSubmit: 2494  Solved: 1101[Submit][ ...

  7. BZOJ 1833 数字计数 数位DP

    题目链接 做的第一道数位DP题,听说是最基础的模板题,但还是花了好长时间才写出来..... 想深入了解下数位DP的请点这里 先设dp数组dp[i][j][k]表示数位是i,以j开头的数k出现的次数 有 ...

  8. UVA.1640.The Counting Problem / BZOJ.1833.[ZJOI2010]数字计数(数位DP)

    题目链接 \(Description\) 求\([l,r]\)中\(0,1,\cdots,9\)每个数字出现的次数(十进制表示). \(Solution\) 对每位分别DP.注意考虑前导0: 在最后统 ...

  9. 1833: [ZJOI2010]count 数字计数——数位dp

    传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=1833 省选之前来切一道裸的数位dp.. 题意 统计[a,b]中0~9每个数字出现的次数(不算 ...

随机推荐

  1. React Hook挖坑

    React Hook挖坑 如果已经使用过 Hook,相信你一定回不去了,这种用函数的方式去编写有状态组件简直太爽啦. 如果还没使用过 Hook,那你要赶紧升级你的 React(v16.8+),投入 H ...

  2. 用c#求一元二次方程

    题目:编一个程序,输入a .b.c 的值,求出一元二次方程a*x*x+b*x+c=0的二个实数根. 我的思路: 我们都知道数学中求一元二次方程有很多方法:直接开方法.配方法.公式法.分解因式法等等,在 ...

  3. 【tensorflow2.0】处理图片数据-cifar2分类

    1.准备数据 cifar2数据集为cifar10数据集的子集,只包括前两种类别airplane和automobile. 训练集有airplane和automobile图片各5000张,测试集有airp ...

  4. 1044 Shopping in Mars (25分)(二分查找)

    Shopping in Mars is quite a different experience. The Mars people pay by chained diamonds. Each diam ...

  5. spring中BeanPostProcessor之一:InstantiationAwareBeanPostProcessor(03)

    前面介绍了InstantiationAwareBeanPostProcessor后置处理器的postProcessBeforeInstantiation和postProcessAfterInstant ...

  6. 玩转redis-延时消息队列

    上一篇基于redis的list实现了一个简单的消息队列:玩转redis-简单消息队列 源码地址 使用demo 产品经理经常说的一句话,我们不光要有X功能,还要Y功能,这样客户才能更满意.同样的,只有简 ...

  7. Git应用详解第四讲:版本回退的三种方式与stash

    前言 前情提要:Git应用详解第三讲:本地分支的重要操作 git作为一款版本控制工具,其最核心的功能就是版本回退,没有之一.熟悉git版本回退的操作能够让你真真正正地放开手脚去开发,不用小心翼翼,怕一 ...

  8. MySQL学习之路1-Mac下启动连接MySQL服务

    MySQL简介 (MySQL是目前最流行的关系型数据库管理系统,现属于Oracle公司.) MySQL主要特点: 支持大型数据库,支持5000万条记录的数据仓库,32位系统表文件最大可支持4GB,64 ...

  9. template_共享模板

    方法: 定义一个基本框架html文件  举例:定义{标题.内容.页尾}区块   定义相应的html文件实现区块的具体样式或内容   定义具体静态网页html文件时调用这些区块html文件, 实现公共元 ...

  10. hadoop(四)centos7克隆|静态ip|机器名|映射关系|别名配置(完全分布式准备一)|6

    hadoop完全分布式准备工作 克隆默认基础虚拟机三台102/103/104目标:在win10主机上能连上这三台机器,三台机器之间可以互相ping通,用机器名也可ping通.基础虚拟机:创建了文件op ...