【题解】P2602 [ZJOI2010]数字计数

$Description: $

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

\(Sample\) \(Input:\)

1 99

\(Sample\) \(Output:\)

9 20 20 20 20 20 20 20 20 20

\(Solution:\)

状态 \(f[i][j]\) 表示前 \(i\) 位当前这个数出现了 \(j\) 次 。

那么就可以愉快的dfs了,但是还有一点初始化没懂,暂时得先把他撂下了。

#include<bits/stdc++.h>
#define int long long
using namespace std;
int l,r;
const int N=15,M=10;
int f[N][N],digit[N],ans[N];
// f[i][j] => 对于前 i 位,当前作的这个数出现了sum次。
// lead => 是否为前导0,如果是lead==1 不然 lead==0
inline int dfs(int len,int num,int sum,bool lead,bool limit){
    if(len<=0) return f[len][sum]=sum;
    if(!limit && f[len][sum]!=-1) return f[len][sum];

    int ret=0,up_bound=(limit)?digit[len]:9;
    for(int i=0;i<=up_bound;++i)
        ret+=dfs(len-1,num,sum+((i||lead) && (i==num)),lead||i,limit&&(i==up_bound));
                            // i||lead => 当前这位是否属于前导0,i==num => 是否是要求的数字
    if(!limit && lead) f[len][sum]=ret;
    return ret;
}
inline int solve(int n,int x){
    int cnt=0;
    memset(f,-1,sizeof(f));
    while(n){
        digit[++cnt]=n%10;
        n/=10;
    }
    return dfs(cnt,x,0,0,true);
}
signed main(){
    scanf("%lld%lld",&l,&r);
    for(int i=0;i<M;++i)
        printf("%lld ",solve(r,i)-solve(l-1,i));
    return 0;
}

原文地址:https://www.cnblogs.com/JCNL666/p/10711001.html

时间: 2024-08-30 13:37:15

【题解】P2602 [ZJOI2010]数字计数的相关文章

数位dp详解&amp;&amp;LG P2602 [ZJOI2010]数字计数

数位dp,适用于解决一类求x~y之间有多少个符合要求的数或者其他. 例题 题目描述 杭州交通管理局经常会扩充一些的士车牌照,新近出来一个好消息,以后上牌照,不再含有不吉利的数字了,这样一来,就可以消除个别的士司机和乘客的心理障碍,更安全地服务大众. 不吉利的数字为所有含有4或62的号码.例如: 62315 73418 88914 都属于不吉利号码.但是,61152虽然含有6和2,但不是62连号,所以不属于不吉利数字之列. 你的任务是,对于每次给出的一个牌照区间号,推断出交管局今次又要实际上给多少

Luogu P2602 [ZJOI2010]数字计数

这算是一道数位DP的入门题了吧虽然对于我来说还是有点烦 经典起手式不讲了吧,\(ans(a,b)\to ans(1,b)-ans(1,a-1)\) 我们首先预处理一个东西,用\(f_i\)表示有\(i\)位数字的时候,每个数字有几个(注意是和).若不考虑前导零,则所有数字都是等价的,转移为: \(f_i=10\cdot f_{i-1}+10^{i-1}\) 这个还是比较好理解的吧,前面一项表示无论这一位放什么直接从前面拿过来已有的,所以这一位可以放\(0\to9\)十个数,后面一项表示当这一位放

P2602 [ZJOI2010]数字计数

题目描述 给定两个正整数a和b,求在[a,b]中的所有整数中,每个数码(digit)各出现了多少次. 输入输出格式 输入格式: 输入文件中仅包含一行两个整数a.b,含义如上所述. 输出格式: 输出文件中包含一行10个整数,分别表示0-9在[a,b]中出现了多少次. 输入输出样例 输入样例#1: 1 99 输出样例#1: 9 20 20 20 20 20 20 20 20 20 说明 30%的数据中,a<=b<=10^6: 100%的数据中,a<=b<=10^12. Solution

题解——[ZJOI2010]数字计数 数位DP

最近在写DP,今天把最近写的都放上来好了,,, 题意:给定两个正整数a和b,求在[a,b]中的所有整数中,每个数码(digit)各出现了多少次. 首先询问的是一个区间,显然是要分别求出1 ~ r ,1 ~ l的答案,然后相减得到最终答案 首先我们观察到,产生答案的区间是连续的,且可以被拆分, 也就是说0 ~ 987的贡献= 0 ~ 900 + 901 ~ 987的恭喜, 同理,把位拆开也是等价的,所以我们可以单独计算每个位的贡献 这样讲可能有点不太清晰,举个例子吧 3872 我们先把它按数拆开来

[ZJOI2010]数字计数 题解

题面 这道题是一道数位DP的模板题: 因为窝太蒟蒻了,所以不会递推,只会记忆化搜索: 首先,咋暴力咋来: 将一个数分解成一个数组,这样以后方便调用: 数位DP的技巧:(用1~b的答案)-(1~a的答案)就是(a~b的答案): 那么对于每个数码i,我们做两次dfs(分别以a为上界和以b为上界): 设正在搜索的数码是digit: 枚举每一位,当这位==digit时,便将答案+1,并记忆化: 然后就没了; 可是这样做忽略了两个重要的事情: 1.可能存在前导零: 2.目前搜到的数比目标值要大: 对于这两

[ZJOI2010]数字计数 理解

题目链接:https://www.luogu.org/problemnew/show/P2602 设d[k][i]为当前情况下选取到第 k 位时数字 i 的数量, sum[k][i]为第 k 位选取 i 时最终的dfa情况数量(能凑成的数字总量)显然 d[k][i]+=d[k+1][i];而对于第 k+1 位不是 i 的数, 也要加上它们的情况(因为第 k 位出现了 1 个 i)所以 d[k][i]+=d[k+i][j](0<=j<=9) 原文地址:https://www.cnblogs.co

[ZJOI2010] 数字计数

题目描述 给定两个正整数a和b,求在[a,b]中的所有整数中,每个数码(digit)各出现了多少次. 输入输出格式 输入格式: 输入文件中仅包含一行两个整数a.b,含义如上所述. 输出格式: 输出文件中包含一行10个整数,分别表示0-9在[a,b]中出现了多少次. 输入输出样例 输入样例#1: 1 99 输出样例#1: 9 20 20 20 20 20 20 20 20 20 说明 30%的数据中,\(a<=b<=10^6:\) 100%的数据中,\(a<=b<=10^{12}.\

BZOJ_1833_[ZJOI2010]_数字计数_(数位dp)

描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1833 统计\(a~b\)中数字\(0,1,2,...,9\)分别出现了多少次. 分析 数位dp真是细节又多又容易出错,我都懒得看题解,所以也就懒得写题解了... 注意细节吧还是... 1 #include <bits/stdc++.h> 2 using namespace std; 3 4 typedef long long ll; 5 ll a,b; 6 ll A[10],B[10],n

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

1833: [ZJOI2010]count 数字计数 题目:传送门 题解: 今天是躲不开各种恶心DP了??? %爆靖大佬啊!!! 据说是数位DP裸题...emmm学吧学吧 感觉记忆化搜索特别强: 定义f[i][j][k]表示若前i个位置有k个j的此时的全局方案数,然后就可以记忆化搜索了(具体看代码吧) 代码: 1 #include<cstdio> 2 #include<cstring> 3 #include<cstdlib> 4 #include<cmath>