CodeForces 55D "Beautiful numbers"(数位DP+离散化处理)
参考资料:
[1]:CodeForces 55D Beautiful numbers(数位dp&&离散化)
题意:
求一个区间内的Beautiful numbers有多少个。Beautiful numbers指:一个数能整除所有组成它的非0数字。
例如15可以被1和5整除,所以15是Beautiful numbers。
我的理解:
起初,我先定义一个三维数组 dp[ i ][ j ][ k ]:来到 i 位置时,所有非零数的lcm = j,当前数位 k 时含有的 Beautiful numbers 的个数。
但是,由题意得,当前的数 k 可以是个很大的数(9e18),数组根本就开不下,那怎么办呢?
将当前的数 hash 一下,如何hash呢?
假设 a,b,c,d 为[0,9]的数,那么不存在另一个 a',b',c',d'
使得 a*11+b*13+c*17+d*19 = a' *11+b' *13+c' *17+d' *19;
也就是说,这 19 位数分别乘以大于10 的互不相等的前19个素数是不会存在重数的;
那么,hash完后,最大的数才7000多,那么开个 dp[20][50][8000] 的数组绰绰有余;
但问题来了,既然每个数的 hash 值都不想等,那,哪来的记忆化搜索?
所以说,这就是赤裸裸的暴力!!!!!!
so,举足无措,只好参考大佬博客辽。
下面具体说说我的理解:
首先解释一下上述博客中给出的公式 sum%(x*n)%x == sum%x 的证法以及作用;
证明:
sum%(x*n) = sum-(int)(sum / (x*n) )*(x*n);
sum%(x*n)%x = [sum-(int)(sum / (x*n) )*(x*n) ] % x = sum%x - 0 = sum % x;
那,接下来就要看看要怎么用这个公式了,预处理出 1~9的所有不同组合的 lcm ,我是用DFS预处理的;
参考代码如下:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define mem(a,b) memset(a,b,sizeof(a)) int a[];
bool vis[];//vis[i]:判断数字 i 是否访问过
bool vis2[*******];//vis2[lcm]:判断当前的lcm是否在之前出现过 int GCD(int a,int b)
{
return a== ? b:GCD(b%a,a);
}
int LCM(int a,int b)
{
return a*b/GCD(a,b);
} void DFS(int t,int lcm,int &k)
{
if(!vis2[lcm])
{
a[k++]=lcm;
vis2[lcm]=true;
} for(int i=;i <= ;++i)
{
if(vis[i])
continue;
vis[i]=true;
DFS(t+,LCM(lcm,i),k);
vis[i]=false;
}
}
int main()
{
int k=; mem(vis,false);
mem(vis2,false); DFS(,,k);
sort(a,a+);
for(int i=;i < k;++i)
printf("%d,",a[i]); return ;
}
预处理出所有的lcm
与处理完后,你会发现最大的lcm为2520,且 2520%(其余的lcm) = 0;
那么,这就好办了,对于当前所形成的数 curNum 和当前的 lcm,我们需要判断 curNum % lcm 是否等于0,那么问题就是
curNum 太大了怎么办?
通过上面的公式,可知 curNum%lcm = curNum%(lcm*(2520/lcm) )%lcm = (curNum%2520)%lcm;
所以,每次只需要将 curNum%2520 即可。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define ll long long
#define mem(a,b) memset(a,b,sizeof(a)) ll n,m;
int digit[];
ll dp[][][];
int a[]={ , , , , , , , , ,,
,,,,,,,,,,
,,,,,,,,,,
,,,,,,,,
,,,,,,,,,}; int GCD(int a,int b)
{
return a== ? b:GCD(b%a,a);
}
int LCM(int a,int b)
{
return a*b/GCD(a,b);
} ll DFS(int curPos,int curNum,int lcm,bool limit)
{
if(curPos == -)
return curNum%lcm == ? :;
int t=lower_bound(a,a+,lcm)-a;
if(!limit && dp[curPos][t][curNum] != -)
return dp[curPos][t][curNum]; int up=limit ? digit[curPos]:;
ll ans=;
for(int i=;i <= up;++i)
ans += DFS(curPos-,(curNum*+i)%,(i != ? LCM(lcm,i):lcm),limit&&i==digit[curPos]);
if(!limit)
dp[curPos][t][curNum]=ans; return ans;
}
ll Solve(ll x)
{
int k=;
while(x)
{
digit[k++]=x%;
x /= ;
}
return DFS(k-,,,true);
}
int main()
{
int test;
scanf("%d",&test);
mem(dp,-);
while(test--)
{
scanf("%lld%lld",&n,&m);
printf("%lld\n",Solve(m)-Solve(n-));
}
return ;
}
CodeForces 55D "Beautiful numbers"(数位DP+离散化处理)的更多相关文章
- codeforces 55D - Beautiful numbers(数位DP+离散化)
D. Beautiful numbers time limit per test 4 seconds memory limit per test 256 megabytes input standar ...
- CodeForces - 55D - Beautiful numbers(数位DP,离散化)
链接: https://vjudge.net/problem/CodeForces-55D 题意: Volodya is an odd boy and his taste is strange as ...
- CodeForces - 55D Beautiful numbers —— 数位DP
题目链接:https://vjudge.net/problem/CodeForces-55D D. Beautiful numbers time limit per test 4 seconds me ...
- Codeforces - 55D Beautiful numbers (数位dp+数论)
题意:求[L,R](1<=L<=R<=9e18)区间中所有能被自己数位上的非零数整除的数的个数 分析:丛数据量可以分析出是用数位dp求解,区间个数可以转化为sum(R)-sum(L- ...
- codeforces 55D. Beautiful numbers 数位dp
题目链接 一个数, 他的所有位上的数都可以被这个数整除, 求出范围内满足条件的数的个数. dp[i][j][k], i表示第i位, j表示前几位的lcm是几, k表示这个数mod2520, 2520是 ...
- FZU2179/Codeforces 55D beautiful number 数位DP
题目大意: 求 1(m)到n直接有多少个数字x满足 x可以整出这个数字的每一位上的数字 思路: 整除每一位.只需要整除每一位的lcm即可 但是数字太大,dp状态怎么表示呢 发现 1~9的LCM 是2 ...
- CF 55D. Beautiful numbers(数位DP)
题目链接 这题,没想出来,根本没想到用最小公倍数来更新,一直想状态压缩,不过余数什么的根本存不下,看的von学长的blog,比着写了写,就是模版改改,不过状态转移构造不出,怎么着,都做不出来. #in ...
- Codeforces 55D. Beautiful numbers(数位DP,离散化)
Codeforces 55D. Beautiful numbers 题意 求[L,R]区间内有多少个数满足:该数能被其每一位数字都整除(如12,24,15等). 思路 一开始以为是数位DP的水题,觉得 ...
- 2018 ACM 国际大学生程序设计竞赛上海大都会赛重现赛 J Beautiful Numbers (数位DP)
2018 ACM 国际大学生程序设计竞赛上海大都会赛重现赛 J Beautiful Numbers (数位DP) 链接:https://ac.nowcoder.com/acm/contest/163/ ...
随机推荐
- git 提交顺序
0. git branch # 查看自己是哪个分支:先确定自己现在是哪个分支 1. git fetch # 将远程主机的更新,全部取回本地.如果只想取回特定分支的更新,可以指定分支名:git ...
- 今日头条移动app广告激活数据API对接完整Java代码实现供大家参考》》》项目随记
这是自毕业后的第一篇博客,希望自己今后能养成写博客的一个好习惯.最近公司为了加速APP推广,采取在外部平台(如:今日头条)进行广告投放的方式,进行用户引流.因此我们需要对广告的激活数据进行一个检测,跟 ...
- iOS应用的性能调试
1.Static Analysis 使用之前先清理一下数据:product-->Clean 可能遇到的问题: a.发现工程中有多个“User-facing text should use loc ...
- 前端bug记录
记录一下前端入门时期遇到的一些bug. 1.同步.异步问题 背景:Javascript语言的执行环境是单线程.即一次只能完成一个任务.若有多个任务则需排队逐个执行——前一个任务完成,再执行后一个任务. ...
- 【XSY1537】五颜六色的幻想乡 数学 生成树计数 拉格朗日插值
题目大意 有一个\(n\)个点\(m\)条边的图,每条边有一种颜色\(c_i\in\{1,2,3\}\),求所有的包括\(i\)条颜色为\(1\)的边,\(j\)条颜色为\(2\)的边,\(k\) ...
- 爬虫_糗事百科(scrapy)
糗事百科scrapy爬虫笔记 1.response是一个'scrapy.http.response.html.HtmlResponse'对象,可以执行xpath,css语法来提取数据 2.提取出来的数 ...
- P3723 [AH2017/HNOI2017]礼物
题目链接:[AH2017/HNOI2017]礼物 题意: 两个环x, y 长度都为n k可取 0 ~ n - 1 c可取任意值 求 ∑ ( x[i] - y[(i + k) % n + 1] ...
- HNOI2019 退役记
窗外风雨大作. 凌晨两点又被噩梦惊醒,朦胧中看见书桌的灯还亮着. 走近书桌,有一个人正在敲打着键盘,屏幕上是一些似曾相识的东西. 被水浸湿的头发随意地搭在额头上,鼻梁上架着一副眼镜,镜片上全是小水珠, ...
- 实验六 MapReduce实验:二次排序
实验指导: 6.1 实验目的基于MapReduce思想,编写SecondarySort程序. 6.2 实验要求要能理解MapReduce编程思想,会编写MapReduce版本二次排序程序,然后将其执行 ...
- 【BZOJ4061】[Cerc2012]Farm and factory(最短路,构造)
[BZOJ4061][Cerc2012]Farm and factory(最短路,构造) 题面 BZOJ 然而权限题QwQ. 题解 先求出所有点到达\(1,2\)的最短路,不妨记为\(d_{u,1}, ...