bzoj 1799: [Ahoi2009]self 类似的分布 解读
【原标题】
1799: [Ahoi2009]self 同类分布
Time Limit: 50 Sec Memory Limit: 64 MB id=1799" style="color:blue; text-decoration:none">Submit id=1799" style="color:blue; text-decoration:none">Status
Submit: 554 Solved: 194
[
Description
Input
Output
Sample Input
Sample Output
HINT
【约束条件】1 ≤ a ≤ b ≤ 10^18
Source
【分析】肯定是数位DP。只是望着50s的时限我大笑:这么宽?
于是匆匆整理了思路。
最多时18个9。也就是和最大值是162。首先我要先枚举和P。
状态怎么表示呢?哦。f[i][j][k][sum]表示到第i位。首位是j,眼下总和是sum,且眼下模P的答案是K。推起来简单。就是统计的时候还是得一位一位的来,略麻烦。
可是写完后。我发现例子时过了,可连例子都跑得飞慢!。。
计算效率:162*18*10*162*162。咦?怎么算都是超时的!!。
【代码1】
#include<cstdio>
#include<cstring>
#include<cmath>
#define D 19
#define S 163
#define G 10
using namespace std;
typedef long long LL;
LL f[D][G][S][S],A,B,M[D][S];
LL PRE()
{
int a[D]={0},cnta=0;LL ansa=0;
for (;A;A/=10ll) a[++cnta]=A%10;
int b[D]={0},cntb=0;LL ansb=0;
for (;B;B/=10ll) b[++cntb]=B%10;
for (int P=1;P<=162;P++)
{
memset(f,0,sizeof(f));
for (int i=0;i<10;i++) f[1][i][i%P][i]=1ll;
for (int i=1;i<cntb;i++)
for (int j=0;j<=9;j++)
for (int k=0;k<P;k++)
for (int sum=0;sum<=P;sum++)
if (f[i][j][k][sum])
{
for (int now=0;now<=9;now++)
f[i+1][now][(k+now*M[i][P])%P][sum+now]+=f[i][j][k][sum];
}
LL ta=ansa,tb=ansb;
int s=0,d=0,now=0;
for (int i=cnta;i;i--)
{
for (int j=(i==1);j<a[i];j++) ansa+=f[i][j][now][P-d];
s=(s+a[i])*10%P;d+=a[i];now=(P-s)*(P!=s);
}
s=0;d=0;now=0;
for (int i=cntb;i;i--)
{
for (int j=(i==1);j<b[i];j++) ansb+=f[i][j][now][P-d];
s=(s+b[i])*10%P;d+=b[i];now=(P-s)*(P!=s);
}
}
if (!ansa) ansa--;
return ansb-ansa;
}
int main()
{
scanf("%lld%lld",&A,&B);A--;
M[0][1]=0;for (int j=2;j<=162;j++) M[0][j]=1;
for (int i=1;i<=18;i++)
for (int j=1;j<=162;j++)
M[i][j]=M[i-1][j]*10ll%j;
printf("%lld",PRE());
return 0;
}
于是无节操的去看题解。
有一位P党的大神是按位去枚举sum的。效率非常高。
于是就。
。
。嘿嘿。
。。
赶紧借鉴一下。
然后发现空间上过不去。。。。
上面内存限制64M(当然BZOJ上超一点点没什么事),我68M!
于是開始压内存。把无用的全删除了——66M!
怎么办?打了一个非常小的点(事实上也不算是打点。仅仅是把sum=162的这样的情况去掉)
65M多!
!
还是卡只是去!
!
气死了。就用数组自然溢出吧!咦?居然过了。!
【代码】
#include<cstdio>
#include<cstring>
#include<algorithm>
#define D 19
#define S 160 //原来是163
using namespace std;
typedef long long LL;
LL f1[S-1][S][S-1],f2[S-1][S][S-1];LL A,B;
LL PRE(LL A)
{
int a[D]={0},cnt=0,y,i,j,k,l,z,sum=0;LL ans=0,x=A,mul,now;
if (!A) return 0;
for (;A;A/=10ll) a[++cnt]=A%10,sum+=a[cnt];
for (i=1;i<=cnt/2;i++) swap(a[i],a[cnt-i+1]);
memset(f1,0,sizeof(f1));
for (i=1;i<S;i++) f1[0][i][0]=1ll;
if (x%sum==0) ans++;mul=1ll;
for (i=0;i<cnt;i++)
{
x/=10ll;mul*=10ll;now=x*mul;sum-=a[cnt-i];
for (z=0;z<a[cnt-i];z++)
{
for (j=sum+z+(sum+z==0);j<=sum+z+i*9;j++)
y=(now%j)?j-now%j:0,ans+=f1[j-sum-z][j][y];
now+=mul/10;
}
memcpy(f2,f1,sizeof(f1));
memset(f1,0,sizeof(f1));
for (j=0;j<=i*9;j++)
for (k=j;k<S;k++)
for (l=0;l<k;l++)
for (z=0;z<=9;z++)
f1[j+z][k][(l*10+z)%k]+=f2[j][k][l];
}
return ans;
}
int main()
{
scanf("%lld%lld",&A,&B); if (B>=999999999999999999ll) B=999999999999999998ll,printf("%lld",PRE(B)-PRE(A-1)+1);
else printf("%lld",PRE(B)-PRE(A-1));
return 0;
}
版权声明:本文博客原创文章,博客,未经同意,不得转载。
bzoj 1799: [Ahoi2009]self 类似的分布 解读的更多相关文章
- bzoj 1799: [Ahoi2009]self 同类分布 数位dp
1799: [Ahoi2009]self 同类分布 Time Limit: 50 Sec Memory Limit: 64 MB[Submit][Status][Discuss] Descripti ...
- BZOJ 1799 - [AHOI2009]self 同类分布 - 枚举 数位DP
Description 找出$[L, R]$ 区间内有多少数, 各位数字和 能整除原数 Solution 枚举每个可能的数字和, 进行数位DP即可 , 水爆 Code #include<cstd ...
- [BZOJ 1799] self 同类分布
Link: BZOJ 1799 传送门 Solution: 一句话的题目,看得爽,做得烦 一般这类和数位相关的都是数位$dp$吧 不过一开始还是感觉不太可做,毕竟每个数模数不同 但要发现,模数最高也只 ...
- Bzoj 1081 [Ahoi2009] chess 中国象棋
bzoj 1081 [Ahoi2009] chess 中国象棋 题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1801 状态比较难设,的确 ...
- 【BZOJ】1799: [Ahoi2009]self 同类分布
[题意]给出a,b,求出[a,b]中各位数字之和能整除原数的数的个数.1 ≤ a ≤ b ≤ 10^18 [算法]数位DP [题解] 感觉这种方法很暴力啊. 枚举数位和1~162(不能枚举0,不然会模 ...
- BZOJ 1799 同类分布
一开始没想出来..一看题解 我艹直接枚举数位的和啊.....怪不得给50s. 还是太蠢. #include<iostream> #include<cstdio> #includ ...
- BZOJ 1799 同类分布(数位DP)
给出a,b,求出[a,b]中各位数字之和能整除原数的数的个数.1<=a<=b<=1e18. 注意到各位数字之和最大是153.考虑枚举这个东西.那么需要统计的是[0,a-1]和[0,b ...
- BZOJ 1797: [Ahoi2009]Mincut 最小割
1797: [Ahoi2009]Mincut 最小割 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 2076 Solved: 885[Submit] ...
- bzoj 1800: [Ahoi2009]fly 飞行棋 暴力
1800: [Ahoi2009]fly 飞行棋 Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline ...
随机推荐
- 7.oracle学习门户系列七---网络管理和配置
oracle学习门户系列七 网络管理和配置 们学习了模式和用户.包含模式定义以及模式的作用. 这篇我么来看下ORACLE数据库中的网络管理和配置.只是这篇好像和上篇没有继承啊.这怎么看? Ok,事实上 ...
- UVA714- Copying Books(最大最小化)
意甲冠军:k手稿的部分成m部分,使每一个和最小 思路:典型最大值最小化问题,使用贪心+二分. 贪心的是每次尽量将元素往右边划分,二分查找最小的x满足m个连续的子序列和S(i)都不超过x. 由于输出的原 ...
- Android Studio Debug
小米4usb调试怎么打开?miui6进入开发者模式想要打开USB调试首先开启开发者模式.过去在MIUI V5版本时,小米手机开启开发者模式的方法是连续点击Anroid版本号.不过最新上市的小米4都搭载 ...
- spring+websocket综合(springMVC+spring+MyBatis这是SSM框架和websocket集成技术)
java-websocket该建筑是easy.儿童无用的框架可以在这里下载主线和个人教学好java-websocket计划: Apach Tomcat 8.0.3+MyEclipse+maven+JD ...
- 谈话ZooKeeper(一个)分析ZooKeeper的Quorums机制--预防Split-Brain问题
使用ZooKeeper学生们应该看到一个参数.它是ZooKeeper超过一半的群集必须节点(Majority)可用的.外来人才在整个集群中可用.在大多数情况下,这种说法是正确的. 谈论这篇文章背后的原 ...
- 64位sql server 如何使用链接服务器连接Access
原文:64位sql server 如何使用链接服务器连接Access 测试环境 操作系统版本:Windows Server 2008 r2 64位 数据库版本:Sql Server 2005 64位 ...
- Eclipse正在使用Ant扑灭Android数据包错误的解决方案 – Perhaps JAVA_HOME does not point to the JDK
问题描写叙述: 在Eclipse中执行ant批量打包工具出错,日志信息例如以下: D:\Android\android-sdk-windows\tools\ant\build.xml:601: The ...
- ORACLE在表中添加的目光凝视和读取列
在ORACLE中给表.列添加凝视以及读取凝视 1.给表填加凝视:SQL>comment on table 表名 is '表凝视"; 2.给列加凝视:SQL>comment on ...
- 十天学Linux内核之第十天---总结篇(kconfig和Makefile & 讲不出再见)
原文:十天学Linux内核之第十天---总结篇(kconfig和Makefile & 讲不出再见) 非常开心能够和大家一起分享这些,让我受益匪浅,感激之情也溢于言表,,code monkey的 ...
- POJ 1743 Musical Theme Hash+二分法
标题效果:有一个美丽的旋律,它们是由一些不大于88音调.如果计为五个音调的量度,问:是否有相同的节奏的多个部分(相同的差,以及两者之间的相同的节奏不能重叠),并寻求最长长度. 思考:这个问题是八人中的 ...