SDOI2010代码拍卖会 (计数类DP)
P2481 SDOI2010代码拍卖会

$ solution: $
这道题调了好久好久,久到都要放弃了。洛谷的第五个点是真的强,简简单单一个1,调了快4个小时!
这道题第一眼怎么都是数位DP,奈何数据范围太大,只能找性质。而这道题最重要的一个性质也很套路(敲难想),因为我们所有的方案都是 $ 111112223333 $ 这样的数,我们要求的是它的大小模 $ p $ ,所以我们考虑将它用加法拆解。于是我们惊奇的发现它可以转化成(以上面那个数为例) $ 111111111111+1111111+1111=111112223333 $ 。正因为数位上的数字单调不降,所以它可以表示为不超过9个形似 $ 111...111 $ 的数的和!
而这种形似 $ 1111...1111 $ 的数,我们可以发现,如果我们依次枚举它的长度( $ 1,11,111,1111...... $ )这些数在模 $ p $ 意义下一定会产生循环节。于是我们将在模 $ p $ 意义下同余的这类数归到一起,然后用一个数组 $ g[i] $ 记录模 $ p $ 意义下为 $ i $ 的形如 $ 11111...111 $ 的数的个数。(因为 $ p $ 不大,所以不用枚举多久就能找到循环节)
于是题目的意思就被转换成:从所有的 $ g[i] $ 中找出(不超过)九个数,使得它们的下标之和能被 $ p $ 整除!(为什么是下标之和?因为上面说了 $ g[i] $ 中所有的形如 $ 1111...111 $ 的数在模 $ p $ 意义下为 $ i $ !)
所以这道题就变成了一道计数类DP,我们设 $ f[i][j][o] $ 表示:已经选完了下标为 $ [0,i-1] $ 的数,从中选了 $ o $ 个形如 $ 1111...11 $ 的数,它们的和模 $ p $ 等于 $ j $ 的方案数。 考虑它的转移,如果我们要从下标为 $ i $ 的数里选出 $ k $ 个数,显然这 $ k $ 个数是可以重复的,他们的贡献都为 $ i $ ,所以相当我们从 $ g[i] $ 个数里可重复的选出 $ k $ 个数(不论怎么选他们的和都为 $ k*i $ ),根据组合计数原理这样的方案有 $ C_{g[i]+k-1}^{~~~~~k} $ 种。为了方便博主代码里会用 $ C[i][k] $ 代替 $ C[g[i]+k-1][k] $ 。
$ f[i+1][(j+k*i)modp][o+k]=f[i][j][o]*C[i][k] $
$ code: $
#include<iostream>
#include<cstdio>
#include<iomanip>
#include<algorithm>
#include<cstring>
#include<cstdlib>
#include<ctime>
#include<cmath>
#include<vector>
#include<queue>
#include<map>
#include<set>
#define ll long long
#define db double
#define rg register int
using namespace std;
const int mod=999911659; //答案的模数
ll n;
ll g[505]; //记录余数为i的形如11111的数的个数
int p,vn,ans; //循环节长度,小模数
int a[505]; //记录余数上一次的位置(找循环节用)
int inv[11]; //逆元
int c[505][11]; //组合数
int f[505][505][11]; //计数类DP
inline int qr(){
register char ch; register bool sign=0; rg res=0;
while(!isdigit(ch=getchar()))if(ch=='-')sign=1;
while(isdigit(ch))res=res*10+(ch^48),ch=getchar();
if(sign)return -res; else return res;
}
int main(){
//freopen(".in","r",stdin);
//freopen(".out","w",stdout);
scanf("%lld%d",&n,&p); inv[1]=1;
if(n<=p) for(rg i=1;i<=n;++i) vn=(vn*10+1)%p,++g[vn];
else{ rg tot=1%p,m,l;
for(rg i=1;i<=p+1;++i){
if(a[tot]){ l=a[tot]; m=i-l; break;} //找到循环节,计算循环节长度
a[tot]=i; ++g[tot]; tot=(tot*10+1)%p; //记录当前余数出现位置,继续寻找
}
for(rg i=0;i<p;++i)
if(a[i]&&a[i]>=l){
g[i]+=(n-a[i])/m%mod; //记录所有余数为i的形如11111的数的个数
if((a[i]-l+1)%m==(n-l+1)%m) vn=i; //记录长度为 n 的形如1111的数的余数,后面初始DP时有用
}
}
for(rg i=2;i<9;++i) inv[i]=(ll)(mod-mod/i)*inv[mod%i]%mod; //线性递推求逆元
for(rg i=0;i<p;++i){ c[i][0]=1; if(!g[i])continue;
for(rg j=1;j<9;++j,g[i]=(g[i]+1)%mod) //注意最后还有个++g[i]!!!
c[i][j]=(ll)c[i][j-1]*g[i]%mod*inv[j]%mod; //用性质求i较大,j较小的逆元
}
f[0][vn][0]=1; //最开始有一个长度为n的形如1111111111的数
for(rg i=0;i<p;++i)
for(rg j=0;j<p;++j)
for(rg o=0;o<9;++o)
for(rg k=0;k<9-o;++k) //枚举取出多少个余数为i的形如的11111111的数
(f[i+1][(j+k*i)%p][k+o]+=(ll)f[i][j][o]*c[i][k]%mod)%=mod;
for(rg i=0;i<9;++i) ans=(ans+f[p][0][i])%mod; //计算答案
printf("%d\n",ans);
return 0;
}
SDOI2010代码拍卖会 (计数类DP)的更多相关文章
- 动态规划——区间DP,计数类DP,数位统计DP
本博客部分内容参考:<算法竞赛进阶指南> 一.区间DP 划重点: 以前所学过的线性DP一般从初始状态开始,沿着阶段的扩张向某个方向递推,直至计算出目标状态. 区间DP也属于线性DP的一种, ...
- CH5E26 扑克牌 (计数类DP)
$ CH~5E26~\times ~ $ 扑克牌: (计数类DP) $ solution: $ 唉,计数类DP总是这么有套路,就是想不到. 这道题我们首先可以发现牌的花色没有价值,只需要知道每种牌有 ...
- $Poj1737\ Connected\ Graph$ 计数类$DP$
AcWing Description 求$N$个节点的无向连通图有多少个,节点有标号,编号为$1~N$. $1<=N<=50$ Sol 在计数类$DP$中,通常要把一个问题划分成若干个子问 ...
- $CF559C\ Gerald\ and\ Fiant\ Chess$ 计数类$DP$
AcWing Description 有个$H$行$W$列的棋盘,里面有$N$个黑色格子,求一个棋子由左上方格子走到右下方格子且不经过黑色格子的方案数. $1<=H,M<=1e5,1< ...
- BZOJ 1974 [Sdoi2010] auction 代码拍卖会(数位dp)
题目描述 随着iPig在P++语言上的造诣日益提升,他形成了自己一套完整的代码库.猪王国想参加POI的童鞋们都争先恐后问iPig索要代码库.iPig不想把代码库给所有想要的小猪,只想给其中的一部分既关 ...
- [SDOI2010]代码拍卖会——DP
原题戳这里 绝对是一道好题 需要注意到两个东西 1.符合条件的数可以拆成一堆\(11...11\)相加的形式,比如\(1145=1111+11+11+11+1\) 2.\(1,11,111,1111, ...
- Luogu2481 SDOI2010 代码拍卖会 DP、组合
传送门 神仙DP 注意到\(N \leq 10^{18}\),不能够直接数位DP,于是考虑形成的\(N\)位数的性质. 因为低位一定不会比高位小,所以所有满足条件的\(N\)位数一定是不超过\(9\) ...
- [SDOI2010]代码拍卖会
题目描述 随着iPig在P++语言上的造诣日益提升,他形成了自己一套完整的代码库.猪王国想参加POI的童鞋们都争先恐后问iPig索要代码库.iPig不想把代码库给所有想要的小猪,只想给其中的一部分既关 ...
- bzoj 1974: [Sdoi2010]代码拍卖会
Description 随着iPig在P++语言上的造诣日益提升,他形成了自己一套完整的代 码库.猪王国想参加POI的童鞋们都争先恐后问iPig索要代码库.iPi g不想把代码库给所有想要的小猪,只想 ...
随机推荐
- 2017-03-04 idea破解
https://blog.csdn.net/qq_27686779/article/details/78870816 防止原址被删除,备份下,亲测可用 http://idea.java.sx/ 简单快 ...
- AAAI 2018 分析
AAAI 2018 分析 word embedding Learning Sentiment-Specific Word Embedding via Global Sentiment Represen ...
- 【mysql】错误代码1308 Invalid use of NULL value
错误原因是: 在最初设计表script_run_detail表时,resut_id忘记勾选不是null选项, 导致存储数据后发现result_id有NULL值,而实际上,我不希望这个字段可以存储NUL ...
- Eureka服务注册与发现-提供消费服务模型
1.工具及软件版本 JDK1.8 Spring Boot 1.4.3.RELEASE <parent> <groupId>org.springframework.boot< ...
- 在Vue文件中引入外部URL链接
前言:最近做一个vueNuxt的项,没有index.html 也没有main.js项目需要引入一些外部的包,没什么技术含量只是一种思路 在vue生命钩子函数中动态创建JavaScript标签追加到HT ...
- 【ABAP系列】SAP ABAP 关于四舍五入算法
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[ABAP系列]SAP ABAP 关于四舍五入算 ...
- cocos2dx基础篇(14) 滚动视图CCScrollView
[3.x] (1)去掉 "CC" (2)滚动方向 > CCScrollViewDirection 改为强枚举 ScrollView::Dire ...
- 数据结构系列之2-3树的插入、查找、删除和遍历完整版代码实现(dart语言实现)
弄懂了二叉树以后,再来看2-3树.网上.书上看了一堆文章和讲解,大部分是概念,很少有代码实现,尤其是删除操作的代码实现.当然,因为2-3树的特性,插入和删除都是比较复杂的,因此经过思考,独创了删除时分 ...
- 使用cython库对python代码进行动态编译达到加速效果及python第三方包的制作安装
1.测试代码:新建 fib.pyx # coding:utf-8 import matplotlib.pyplot as plt import numpy as np from sklearn.cl ...
- 【计算机视觉】HDR之tone mapping简介
tone Mapping原是摄影学中的一个术语,因为打印相片所能表现的亮度范围不足以表现现实世界中的亮度域,而如果简单的将真实世界的整个亮度域线性压缩到照片所能表现的亮度域内,则会在明暗两端同时丢失很 ...