@description@

已知一个数(允许前导零)有 n 位(n 为偶数),并知道组成这个数的数字集合(并不一定要把集合内的数用完)。求有多少种可能,使得这个数前半部分的数位和等于后半部分的数位和。

模 998244353。

input

第一行两个整数:n k。表示这个数的位数以及组成这个数的数字集合大小。2 <= n <= 2*10^5,1 <= k <= 10。

第二行 k 个两两不同的整数 d1,d2,...,dk,表示组成这个数的数字集合。0 <= di <= 9。

output

输出可能数,模 998244353。

sample input

4 2

1 8

sample output

6

sample explain

6 种可能分别是:1111,1818,8118,1881,8181,8888。

@solution@

模数暗示 NTT。

如果长度为 n/2,数位和为 s 的数有 k 种,则它对答案的贡献为 \(k^2\)。

问题等价于求,长度为 n/2,数位和为 0,1,2,... 的数有多少种。

我们于是可以 dp。定义状态 \(dp[i][j]\) 表示长度为 i 的数,数位和为 j 的方案数。

定义 \(f\),使得 \(f[d[i]]=1(1\le i \le k)\)。可以得到状态转移方程:

\[dp[i][j] = \sum_{k=0}^{9}dp[i-1][j-k]*f[k]
\]

嗯好的这是一个卷积形式的 dp,我们可以用 NTT 来优化。

可以得到 \(dp[\frac{n}{2}][j] = f^{\frac{n}{2}}[j]\)。

转换成点值形式后直接快速幂再转换回来。

@accepted code@

#include<cstdio>
#include<algorithm>
using namespace std;
const int G = 3;
const int MOD = 998244353;
const int MAXN = 200000*9*2;
int pow_mod(int b, int p) {
int ret = 1;
while( p ) {
if( p & 1 ) ret = 1LL*ret*b%MOD;
b = 1LL*b*b%MOD;
p >>= 1;
}
return ret;
}
void ntt(int A[], int n, int type) {
for(int i=0,j=0;i<n;i++) {
if( i < j ) swap(A[i], A[j]);
for(int l=(n>>1);(j^=l)<l;l>>=1);
}
for(int s=2;s<=n;s<<=1) {
int t = (s >> 1);
int u = (type == 1) ? (pow_mod(G, (MOD-1)/s)) : (pow_mod(G, (MOD-1)-(MOD-1)/s));
for(int i=0;i<n;i+=s) {
for(int p=1,j=0;j<t;j++,p=1LL*p*u%MOD) {
int x = A[i+j], y = 1LL*A[i+j+t]*p%MOD;
A[i+j] = (x + y)%MOD, A[i+j+t] = (x + MOD - y)%MOD;
}
}
}
if( type == -1 ) {
int inv = pow_mod(n, MOD-2);
for(int i=0;i<n;i++)
A[i] = 1LL*A[i]*inv%MOD;
}
}
int f[MAXN + 5];
int main() {
int n, k, d, mx = 0;
scanf("%d%d", &n, &k);
for(int i=1;i<=k;i++) {
scanf("%d", &d);
mx = max(mx, d);
f[d] = 1;
}
int len; for(len = 1;len <= ((n>>1)*mx);len<<=1);
ntt(f, len, 1);
for(int i=0;i<len;i++)
f[i] = pow_mod(f[i], (n>>1));
ntt(f, len, -1);
int ans = 0;
for(int i=0;i<len;i++)
ans = (ans + 1LL*f[i]*f[i]%MOD)%MOD;
printf("%d\n", ans);
}

@details@

连续做了几天多项式毒瘤算法,做一个卷积优化是真的轻松惬意。

我才不会说因为 mx 没有弄初值 RE 了让我懵逼了好久。

我怎么可能会因为傻逼错误做不出来傻逼题。

最新文章

  1. Codeigniter基础
  2. LoadLibrary加载动态库失败的解决办法
  3. The integer promotion.
  4. LeetCode() Merge Intervals 还是有问题,留待,脑袋疼。
  5. 电子商务中:B2C、B2B、C2B、C2C、O2O、P2P
  6. UML学习笔记
  7. loadrunner关联数组后拼凑字符串
  8. java 正则匹配空格字符串 正则表达式截取字符串
  9. UVa 540 Team Queue 【STL】
  10. 【转】c++笔试题
  11. 从cin读入一组词并把它们存入一个vector对象,然后设法把所有词都改写为大写字母。
  12. Android SDK镜像
  13. MFC学习指南大纲
  14. 【转】Chrome保存mhtml网页文件的方法 – 无需任何插件,完美!
  15. 【KMP】Number Sequence
  16. Swift笔记3
  17. hdu 4524 郑厂长系列故事——逃离迷宫 小水题
  18. HBase 二级索引与Join
  19. ORACLE 查询某表中的某个字段的类型,是否为空,是否有默认值等
  20. .NET 术语

热门文章

  1. 10分钟完成 mongodb replSet 部署
  2. 论ul、ol和dl的区别
  3. nginx在win系统下的安装配置与tomcat集成springmvc框架
  4. tomcat的访问日志
  5. 2019.7.29 NOIP模拟测试10 反思总结【T2补全】
  6. oracle基本认识
  7. mybatis官网文档mybatis_doc
  8. LintCode_181 将整数A转换为B
  9. oracle触发器级联删除
  10. go struct 抽象