Solution -「LOCAL」充电
\(\mathcal{Description}\)
给定 \(n,m,p\),求序列 \(\{a_n\}\) 的数量,满足 \((\forall i\in[1,n])(a_i\in[1,m])\land(\forall i\in(1,n])(a_{i-1}\le a_i)\land\left(\sum_{i=1}^na_i10^{n-i}\bmod p=0\right)\),对 \(998244353\) 取模。
\(n\le10^{18}\),\(m\le50\),\(p\le200\)。
\(\mathcal{Solution}\)
肯定要利用 \(10^x\) 在\(\bmod p\) 意义下存在循环节的性质来求解。考场上想到暴力扫前 \(\mathcal O(p)\) 个 DP 状态然后用倍增来求解,不过死掉了 qwq。
转化题意——因为序列单调不减,记 \(1_n=\underbrace{11\cdots1}_{n\text{个}1}\),那么任意一种序列所对应的 \(\sum_{i=1}^na_i10^{n-i}\) 都可以由若干个 \(1_x\) 相加得到。为了保证 \(a_i\ge1\),我们钦定一个 \(1_n\) 计入贡献。而显然 \(1_x\) 在\(\bmod p\) 意义下亦存在循环节,我们可以求出 \(c_i\) 表示有 \(c_i\) 个 \(1_x\bmod p=i~(x\in[1,n])\)。问题就转化成:有 \(p\) 类物品,第 \(i\) 类物品权值为 \(i\),有 \(c_i\) 种,每种有无数个,求选出 \(m-1\) 个物品使得其权值和\(\bmod p=0\) 的方案数。
接下来类似背包 DP,定义 \(f(i,j,k)\) 表示决定了前 \(i\) 类物品,已选了 \(j\) 个,权值和\(\bmod p=k\) 的方案数。转移枚举第 \(i+1\) 类所选个数 \(t\),用隔板法计算方案,有:
\]
特别留意初始状态应为 \(f(-1,0,1_n\bmod p)=1\),因为种类编号从 \(0\) 开始;预先钦定了一个 \(1_n\)。
复杂度 \(\mathcal O(m^2p^2)\)。
\(\mathcal{Code}\)
#include <cstdio>
typedef long long LL;
#define int LL
const int MOD = 998244353, MAXM = 50, MAXP = 200;
LL n, buc[MAXP + 5];
int m, p, visc[MAXP + 5], suf[MAXP + 5], inv[MAXM + 5];
int f[2][MAXM + 5][MAXP + 5];
inline void addeq ( int& a, const int b ) { if ( ( a += b ) >= MOD ) a -= MOD; }
inline void total ( const int fir, const int cnt ) {
bool onc[MAXP + 5] = {};
LL cirs = cnt - visc[fir], cirt = n - cnt + cirs + 1;
for ( int i = fir, stp = 1; stp <= cirs; i = suf[i], ++ stp ) {
onc[i] = true;
buc[i] = cirt / cirs + ( stp <= cirt % cirs );
if ( stp % cirs == cirt % cirs ) f[0][0][i] = 1;//, printf ( "!%lld\n", i );
}
for ( int i = 1 % p, stp = 1; !onc[i] && stp <= n; i = suf[i], ++ stp ) {
buc[i] = 1;
if ( n == stp ) f[0][0][i] = 1;//, printf ( "!%lld\n", i );
}
}
signed main () {
freopen ( "charge.in", "r", stdin );
freopen ( "charge.out", "w", stdout );
scanf ( "%lld %lld %lld", &n, &m, &p ), -- m;
for ( int i = 0; i < p; ++ i ) visc[i] = -1;
for ( int l = 1, sum = 1 % p, pwr = 10 % p; ; pwr = 10 * pwr % p, ++ l ) {
if ( ~visc[sum] ) { total ( sum, l ); break; }
visc[sum] = l, suf[sum] = ( sum + pwr ) % p;
if ( l == n ) { total ( sum, l ); break; }
sum = ( sum + pwr ) % p;
}
inv[1] = 1;
for ( int i = 2; i <= m; ++ i ) {
inv[i] = 1ll * ( MOD - MOD / i ) * inv[MOD % i] % MOD;
}
for ( int i = -1, sta = 0; i < p - 1; ++ i, sta ^= 1 ) {
for ( int j = 0; j <= m; ++ j ) {
for ( int k = 0; k < p; ++ k ) {
f[sta ^ 1][j][k] = 0;
}
}
for ( int k = 0; k < p; ++ k ) {
for ( int t = 0; t <= m; ++ t ) {
// 留意枚举的顺序,本质上是f(i,j,k)向后转移,
// 但这里统一计算了t相等时的组合数,再一起转移。
int c = 1;
for ( LL u = buc[i + 1] + t - 1, v = t; v; -- u, -- v ) {
c = 1ll * c * ( u % MOD ) % MOD * inv[v] % MOD;
}
for ( int j = 0; j + t <= m; ++ j ) {
addeq ( f[sta ^ 1][j + t][( k + ( i + 1 ) * t ) % p],
1ll * f[sta][j][k] * c % MOD );
}
}
}
}
int ans = 0;
for ( int i = 0; i <= m; ++ i ) addeq ( ans, f[p & 1][i][0] );
printf ( "%lld\n", ans );
return 0;
}
\(\mathcal{Details}\)
面向数据编程 de 了好久 bug……DP 初值什么的一定不能想当然呐。
Solution -「LOCAL」充电的更多相关文章
- Solution -「LOCAL」二进制的世界
\(\mathcal{Description}\) OurOJ. 给定序列 \(\{a_n\}\) 和一个二元运算 \(\operatorname{op}\in\{\operatorname{ ...
- Solution -「LOCAL」大括号树
\(\mathcal{Description}\) OurTeam & OurOJ. 给定一棵 \(n\) 个顶点的树,每个顶点标有字符 ( 或 ).将从 \(u\) 到 \(v\) ...
- Solution -「LOCAL」过河
\(\mathcal{Description}\) 一段坐标轴 \([0,L]\),从 \(0\) 出发,每次可以 \(+a\) 或 \(-b\),但不能越出 \([0,L]\).求可达的整点数. ...
- Solution -「LOCAL」Drainage System
\(\mathcal{Description}\) 合并果子,初始果子的权值在 \(1\sim n\) 之间,权值为 \(i\) 的有 \(a_i\) 个.每次可以挑 \(x\in[L,R]\) ...
- Solution -「LOCAL」Burning Flowers
灼之花好评,条条生日快乐(假装现在 8.15)! \(\mathcal{Description}\) 给定一棵以 \(1\) 为根的树,第 \(i\) 个结点有颜色 \(c_i\) 和光亮值 ...
- Solution -「LOCAL」画画图
\(\mathcal{Description}\) OurTeam. 给定一棵 \(n\) 个点的树形随机的带边权树,求所有含奇数条边的路径中位数之和.树形生成方式为随机取不连通两点连边直到全 ...
- Solution -「LOCAL」ZB 平衡树
\(\mathcal{Description}\) OurOJ. 维护一列二元组 \((a,b)\),给定初始 \(n\) 个元素,接下来 \(m\) 次操作: 在某个位置插入一个二元组: 翻 ...
- Solution -「LOCAL」舟游
\(\mathcal{Description}\) \(n\) 中卡牌,每种三张.对于一次 \(m\) 连抽,前 \(m-1\) 次抽到第 \(i\) 种的概率是 \(p_i\),第 \(m\) ...
- Solution -「LOCAL」「cov. 牛客多校 2020 第五场 C」Easy
\(\mathcal{Description}\) Link.(完全一致) 给定 \(n,m,k\),对于两个长度为 \(k\) 的满足 \(\left(\sum_{i=0}^ka_i=n\r ...
随机推荐
- JAVA-JDK1.7-ConCurrentHashMap-测试和验证
概述 上次记录了关于ConCurrentHashMap的原理以及源码,现在我对其进行有关功能的测试,下面就讲解一下我测试的内容和代码.这些测试主要针对JDK1.7版本. GET安全测试 上一篇写过ge ...
- 求n以内最大的k个素数以及它们的和
本题要求计算并输出不超过n的最大的k个素数以及它们的和. 输入格式: 输入在一行中给出n(10≤n≤10000)和k(1≤k≤10)的值. 输出格式: 在一行中按下列格式输出: 素数1+素数2+-+素 ...
- elasticsearch拼写纠错之Term Suggester
一.什么是拼写纠错 拼写纠错就是搜索引擎可以智能的感知用户输入关键字的错误,并使用纠正过的关键字进行搜索展示给用户:拼写纠错是一种改善用户体验的功能: elasticsearch提供了以下不同类型的s ...
- [FatFs 学习] SD卡总结-SPI模式
SD卡为移动设备提供了安全的,大容量存储解决方法.它本身可以通过两种总线模式和MCU进行数据传输,一种是称为SD BUS的4位串行数据模式,另一种就是大家熟知的4线SPI Bus模式.一些廉价,低端的 ...
- manjaro20默认关闭蓝牙
用于节电. https://gist.github.com/0xfe11/d0874b7d31cf649616fa9d816571ab3c 推荐执行 # Stop and disable the bl ...
- Centos 7 安装LAMP以及在Apache上安装positiveSSL。
简介 LAMP(linux , Apache, mysql , php)是集成动态网站经常使用的一套开源软件,实际包含linux操作系统,Apache web服务器,mysql(mariadb 分支) ...
- pytest文档3-测试用例setup和teardown
用例运行级别 模块级(setup_module/teardown_module)开始于模块始末,全局的 函数级(setup_function/teardown_function)只对函数用例生效(不在 ...
- context包
目录 Context包到底是干嘛用的? context原理 什么时候应该使用 Context? 如何创建 Context? 主协程通知有子协程,子协程又有多个子协程 context核心接口 empty ...
- golang中结构体标签在json中的应用
package main import ( "encoding/json" "fmt" "reflect" ) type Movie str ...
- HTML(前端web)
目录 一:HTML前端 1.什么是前端? 2.什么是后端? 3.什么是HTML? 4.HTML不是什么? 5.前端的学习流程 6.BS架构 7.搭建服务器 简易(浏览器访问) 8.浏览器访问报错原因 ...