原题戳这里

绝对是一道好题

需要注意到两个东西

**

1.符合条件的数可以拆成一堆\(11...11\)相加的形式,比如\(1145=1111+11+11+11+1\)

2.\(1,11,111,1111,...\)模\(p\)会出现循环,循环节长度不超过\(p\)

**

还有就是\(11...11\)最多为\(9\)个,然后就可以\(dp\)了

首先需要统计长度为\(1-n\)的全\(1\)串中有多少个模\(p\)为\(r\),记为\(cnt[r]\),这个可以\(O(p)\)的预处理出来

接着设\(f[k][i][j]\)表示在\(cnt[1]-cnt[i]\)中已经选了\(k\)个,且它们的和模\(p\)为\(j\)的方案数,那么转移如下:

\[f[k+l][i][(j+l\times i)\% p]+=\binom{cnt[i]+l-1}{l}f[k][i-1][j],0\leqslant k+l\leqslant 8
\]

为什么是\(8\)而不是\(9\)呢?因为我们至少要选一个长度为\(n\)的全\(1\)串,为了方便,我们在\(dp\)之前就把它选出来,相当于占用了\(9\)个中的\(1\)个

还有就是那个组合是可重组合

细节看代码吧:

#include <bits/stdc++.h>

using namespace std;

#define $SHOW(x) cout << #x" = " << x << endl

#define ll long long
#define MOD 999911659
#define MAXN 500 ll n, P, all, cnt[MAXN + 5], inv[10]; // all代表长度为n的全1串模p的值
int st, len, pos[MAXN + 5], f[10][MAXN + 5][MAXN + 5]; void add(int &x, int y) {
x = (x + y) % MOD;
if(x < 0) x += MOD;
} int Mul(int x, int y) {
return 1LL * x * y % MOD;
} int fpow(int x, int p) {
int ret = 1;
while (p) {
if (p & 1) ret = Mul(ret, x);
x = Mul(x, x);
p >>= 1;
}
return ret;
} int C(ll n, ll m) {
if (n < m) return 0;
int ret = 1;
for (ll x = n, y = 1; y <= m; --x, ++y)
ret = Mul(ret, Mul(x % MOD, inv[y]));
return ret;
} int main() {
cin >> n >> P;
if (n <= P) { // 预处理循环节
int sum = 0;
for (int i = 1; i <= n; ++i) {
sum = (sum * 10 + 1) % P;
cnt[sum]++;
}
all = sum;
}
else {
int sum = 0;
for (int i = 1; i <= P + 1; ++i) {
sum = (sum * 10 + 1) % P;
if (cnt[sum]) {
st = pos[sum], len = i - pos[sum];
break;
}
pos[sum] = i, cnt[sum]++;
}
for (int i = 0; i < P; ++i)
if (cnt[i] && pos[i] >= st) {
cnt[i] = (n - st + 1) / len;
if (pos[i] - st + 1 <= (n - st + 1) % len) cnt[i]++;
if ((pos[i] - st + 1) % len == (n - st + 1) % len) all = i;
}
}
for (int i = 1; i <= 8; ++i) inv[i] = fpow(i, MOD - 2);
f[0][0][all] = 1; // 初始化
for (int r = 0; r < P; ++r) {
for (int i = 0; i <= 8; ++i)
for (int k = 0; k < P; ++k)
for (int j = 0; j + i <= 8; ++j)
if(cnt[r]) add(f[i + j][r + 1][(k + j * r % P) % P], Mul(C(cnt[r] + j - 1, j), f[i][r][k]));
else f[i][r + 1][k] = f[i][r][k]; // 注意这里,要把值继承过来
}
int ans = 0;
for (int i = 0; i <= 8; ++i)
add(ans, f[i][P][0]);
printf("%d\n", ans);
return 0;
}

[SDOI2010]代码拍卖会——DP的更多相关文章

  1. Luogu2481 SDOI2010 代码拍卖会 DP、组合

    传送门 神仙DP 注意到\(N \leq 10^{18}\),不能够直接数位DP,于是考虑形成的\(N\)位数的性质. 因为低位一定不会比高位小,所以所有满足条件的\(N\)位数一定是不超过\(9\) ...

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

    P2481 SDOI2010代码拍卖会 $ solution: $ 这道题调了好久好久,久到都要放弃了.洛谷的第五个点是真的强,简简单单一个1,调了快4个小时! 这道题第一眼怎么都是数位DP,奈何数据 ...

  3. BZOJ 1974: [Sdoi2010]auction 代码拍卖会( dp )

    在1, 11, 111……中选<=8个, + 11..(n个1)拼出所有可能...这些数mod p至多有p中可能, 找出循环的处理一下. 那么dp就很显然了...dp(i, j, k)表示前i种 ...

  4. 【BZOJ-1974】auction代码拍卖会 DP + 排列组合

    1974: [Sdoi2010]auction 代码拍卖会 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 305  Solved: 122[Submit ...

  5. [SDOI2010]代码拍卖会

    题目描述 随着iPig在P++语言上的造诣日益提升,他形成了自己一套完整的代码库.猪王国想参加POI的童鞋们都争先恐后问iPig索要代码库.iPig不想把代码库给所有想要的小猪,只想给其中的一部分既关 ...

  6. bzoj 1974: [Sdoi2010]代码拍卖会

    Description 随着iPig在P++语言上的造诣日益提升,他形成了自己一套完整的代 码库.猪王国想参加POI的童鞋们都争先恐后问iPig索要代码库.iPi g不想把代码库给所有想要的小猪,只想 ...

  7. 洛谷 P2481 [SDOI2010]代码拍卖会

    洛谷 这大概是我真正意义上的第一道黑题吧! 自己想出了一个大概,状态转移方程打错了一点点,最后还是得看题解. 一句话题意:求出有多少个\(n\)位的数,满足各个位置上的数字从左到右不下降,且被\(p\ ...

  8. [BZOJ1974][SDOI2010]代码拍卖会[插板法]

    题意 询问有多少个数位为 \(n\) 的形如 \(11223333444589\) 的数位值不下降的数字在\(\mod p\) 的意义下同余 \(0\). $n\leq 10^{18} ,p\leq ...

  9. luogu P2481 [SDOI2010]代码拍卖会

    luogu 题目中的那个大数一定是若干个1+若干个2+若干个3...+若干个9组成的,显然可以转化成9个\(\underbrace {111...1}_{a_i个1}(0\le a_1\le a_2\ ...

随机推荐

  1. Sightseeing tour 【混合图欧拉回路】

    题目链接:http://poj.org/problem?id=1637 Sightseeing tour Time Limit: 1000MS   Memory Limit: 10000K Total ...

  2. JavaSE基础(六)--Java流程控制语句之条件语句

    Java 条件语句 - if...else 一个 if 语句包含一个布尔表达式和一条或多条语句. 语法 if 语句的语法如下: if(布尔表达式) { //如果布尔表达式为true将执行的语句 } 如 ...

  3. 什么是Java内部类?

    如果大家想了解更多的知识和技术,大家可以 搜索我的公众号:理想二旬不止 (尾部有二维码)或者访问我的 个人技术博客 www.ideal-20.cn 这样阅读起来会更加舒适一些 非常高兴与大家交流,学习 ...

  4. SQLSERVER 查看服务器IP地址的命令

    今天进行负载均衡的测试的时候 想查询一下数据库相关信息 百度了下 找到解决方案为: SELECT SERVERNAME = CONVERT(NVARCHAR(),SERVERPROPERTY('SER ...

  5. Dao设计模式简单实现

    一.什么是Dao设计模式 Dao设计模式封装了操作具体数据库的细节,对业务层提供操作数据库的接口,因此降低了业务层代码与具体数据库之间的耦合,有利于人员分工,增加了程序的可移植性. Dao设计模式中主 ...

  6. 在windows系统下打包linux平台运行的go程序

    在windows系统下打包linux平台运行的go程序 1.先在main.go下打包成.exe可执行程序测试代码是否正确 //cd到main.go目录 go build //打包命令 如果打包成功则表 ...

  7. python 修改文件的创建时间、修改时间、访问时间

    目录 python 修改文件创建.修改.访问时间 方案一 方案二(无法修改文件创建时间) python 修改文件创建.修改.访问时间 突如其来想知道一下 python 如何修改文件的属性(创建.修改. ...

  8. JAVA学习篇--静态代理VS动态代理

    本篇博客的由来,之前我们学习大话设计,就了解了代理模式,但为什么还要说呢? 原因: 1,通过DRP这个项目,了解到了动态代理,认识到我们之前一直使用的都是静态代理,那么动态代理又有什么好处呢?它们二者 ...

  9. Web API 自动生成接口文档

    1.添加NuGet程序包 Microsoft ASP.NET Web API 2.2 Help Page      (这是微软官方的) A Simple Test Client for ASP.NET ...

  10. 1.DOS常用命令

    d:+ 回车:盘符切换,进入D:盘 dir(directory):列出当前目录下的文件及文件夹md(make director):创建目录rd(remove director):删除目录(不能删除非空 ...