BZOJ 3625: [Codeforces Round #250]小朋友和二叉树

时间:2021-09-25 00:48:25

3625: [Codeforces Round #250]小朋友和二叉树

Time Limit: 40 Sec  Memory Limit: 256 MB
Submit: 304  Solved: 130
[Submit][Status][Discuss]

Description

我们的小朋友很喜欢计算机科学,而且尤其喜欢二叉树。
考虑一个含有n个互异正整数的序列c[1],c[2],...,c[n]。如果一棵带点权的有根二叉树满足其所有顶点的权值都在集合{c[1],c[2],...,c[n]}中,我们的小朋友就会将其称作神犇的。并且他认为,一棵带点权的树的权值,是其所有顶点权值的总和。
给出一个整数m,你能对于任意的s(1<=s<=m)计算出权值为s的神犇二叉树的个数吗?请参照样例以更好的理解什么样的两棵二叉树会被视为不同的。
我们只需要知道答案关于998244353(7*17*2^23+1,一个质数)取模后的值。

Input

第一行有2个整数 n,m(1<=n<=10^5; 1<=m<=10^5)。
第二行有n个用空格隔开的互异的整数 c[1],c[2],...,c[n](1<=c[i]<=10^5)。

Output

输出m行,每行有一个整数。第i行应当含有权值恰为i的神犇二叉树的总数。请输出答案关于998244353(=7*17*2^23+1,一个质数)取模后的结果。

Sample Input

样例一:
2 3
1 2
样例二:
3 10
9 4 3
样例三:
5 10
13 10 6 4 15

Sample Output

样例一:
1
3
9
样例二:
0
0
1
1
0
2
4
2
6
15
样例三:
0
0
0
1
0
1
0
2
0
5

HINT

对于第一个样例,有9个权值恰好为3的神犇二叉树:

BZOJ 3625: [Codeforces Round #250]小朋友和二叉树

Source

[Submit][Status][Discuss]

据说是道生成函数+多项式计算水题,所以机房里只有我这样的蒟蒻才不会做。

设F(x)中的i次方项系数表示权值和为i的二叉树的个数。(这个就是最终答案函数)

设C(x)中的i次方项系数表示权值i是否出现在可选集合内。(这其实就是C集合的bitset表示)

然后得到$F=CF^2+1$,其中C为枚举根节点权值,两个F分别是左右子树,+1代表可能是一棵空树。

化简得到$CF^2-F+1=0$,根据一元二次方程求根公式,这个多项式方程的根就是$\frac{1\pm\sqrt{1-4C}}{2C}$。

然后嘞,就支持一下多项式开方,多项式求逆好了,其中需要用到NTT和两个倍增算法,然而蒟蒻的我并不会,只好向Monster_Yi大佬和LincHpin大爷虚心请教。

 #include <cstdio>
#include <cstring> template <class T>
inline void read(T &x)
{
x = ;
char c = getchar();
while (c < )c = getchar();
while (c > )x = x* + c - , c = getchar();
} template <class T>
inline void swap(T &a, T &b)
{
T c;
c = a;
a = b;
b = c;
} const int P = ;
const int D = ;
const int N = ; inline int pow(int a, int b)
{
int r = ; while (b)
{
if (b & )
r = 1LL * r * a % P; b >>= , a = 1LL * a * a % P;
} return r;
} inline void ntt(int *a, int len, int type)
{
for (int i = , t = ; i < len; ++i)
{
if (i > t)swap(a[i], a[t]);
for (int j = (len >> ); (t ^= j) < j; j >>= );
} for (int h = ; h <= len; h <<= )
{
int wn = pow(, (P - ) / h); for (int i = ; i < len; i += h)
{
int wk = ; for (int j = ; j < (h >> ); ++j, wk = 1LL * wk * wn % P)
{
int t = 1LL * wk * a[i + j + (h >> )] % P; a[i + j + (h >> )] = (a[i + j] - t + P) % P;
a[i + j] = (a[i + j] + t) % P;
}
}
} if (type == -)
{
for (int i = ; i < (len >> ); ++i)
swap(a[i], a[len - i]); int inv = pow(len, P - ); for (int i = ; i < len; ++i)
a[i] = 1LL * a[i] * inv % P;
}
} void inv(int *a, int *b, int len)
{
if (len == )
b[] = pow(a[], P - );
else
{
static int t[N]; inv(a, b, len >> ); memcpy(t, a, len * sizeof(int));
memset(t + len, , len * sizeof(int)); ntt(t, len << , );
ntt(b, len << , ); for (int i = ; i < len << ; ++i)
b[i] = 1LL * b[i] * ( - 1LL * t[i] * b[i] % P + P) % P; ntt(b, len << , -); memset(b + len, , len * sizeof(int));
}
} void sqrt(int *a, int *b, int len)
{
if (len == )
b[] = ;
else
{
static int c[N], d[N]; sqrt(a, b, len >> ); memset(d, , len * sizeof(int));
memset(d + len, , len * sizeof(int)); inv(b, d, len); memcpy(c, a, len * sizeof(int));
memset(c + len, , len * sizeof(int)); ntt(c, len << , );
ntt(b, len << , );
ntt(d, len << , ); for (int i = ; i < len << ; ++i)
b[i] = (1LL * c[i] * d[i] % P + b[i]) % P * D % P; ntt(b, len << , -); memset(b + len, , len * sizeof(int));
}
} int n, m, len; int a[N], b[N]; signed main(void)
{
read(n);
read(m); a[] = ; for (len = ; len <= m; len <<= ); for (int i = , x; i <= n; ++i)
if (read(x), x <= m)a[x] = P - ; sqrt(a, b, len); memcpy(a, b, len * sizeof(int)), ++a[];
memset(b, , len * sizeof(int)), inv(a, b, len);
memcpy(a, b, len * sizeof(int)); for (int i = ; i <= m; ++i)
printf("%d\n", (a[i] << ) % P);
}

@Author: YouSiki

BZOJ 3625: [Codeforces Round #250]小朋友和二叉树的更多相关文章

  1. BZOJ 3625 &lbrack;Codeforces Round &num;250&rsqb;小朋友和二叉树 ——NTT 多项式求逆 多项式开根

    生成函数又有奇妙的性质. $F(x)=C(x)*F(x)*F(x)+1$ 然后大力解方程,得到一个带根号的式子. 多项式开根有解只与常数项有关. 发现两个解只有一个是成立的. 然后多项式开根.求逆. ...

  2. bzoj 3625&colon; &lbrack;Codeforces Round &num;250&rsqb;小朋友和二叉树【NTT&plus;多项式开根求逆】

    参考:https://www.cnblogs.com/2016gdgzoi509/p/8999460.html 列出生成函数方程,g(x)是价值x的个数 \[ f(x)=g(x)*f^2(x)+1 \ ...

  3. BZOJ3625&colon; &lbrack;Codeforces Round &num;250&rsqb;小朋友和二叉树

    Description 我们的小朋友很喜欢计算机科学,而且尤其喜欢二叉树.考虑一个含有n个互异正整数的序列c[1],c[2],...,c[n].如果一棵带点权的有根二叉树满足其所有顶点的权值都在集合{ ...

  4. BZOJ3625 &lbrack;Codeforces Round &num;250&rsqb;小朋友和二叉树(生成函数&plus;多项式开根)

    设f(n)为权值为n的神犇二叉树个数.考虑如何递推求这个东西. 套路地枚举根节点的左右子树.则f(n)=Σf(i)f(n-i-cj),cj即根的权值.卷积的形式,cj也可以通过卷上一个多项式枚举.可以 ...

  5. &lbrack;Codeforces Round &num;250&rsqb;小朋友和二叉树

    题目描述: bzoj luogu 题解: 生成函数ntt. 显然这种二叉树应该暴力薅掉树根然后分裂成两棵子树. 所以$f(x)= \sum_{i \in c} \sum _{j=0}^{x-c} f( ...

  6. &lbrack;BZOJ3625&rsqb;&lbrack;Codeforces Round &num;250&rsqb;小朋友和二叉树 多项式开根&plus;求逆

    https://www.lydsy.com/JudgeOnline/problem.php?id=3625 愉快地列式子.设\(F[i]\)表示权值为\(i\) 的子树的方案数,\(A[i]\)为\( ...

  7. &lbrack;BZOJ 3625&rsqb; &lbrack;Codeforces 438E&rsqb; 小朋友的二叉树 &lpar;DP&plus;生成函数&plus;多项式开根&plus;多项式求逆&rpar;

    [BZOJ 3625] [Codeforces 438E] 小朋友的二叉树 (DP+生成函数+多项式开根+多项式求逆) 题面 一棵二叉树的所有点的点权都是给定的集合中的一个数. 让你求出1到m中所有权 ...

  8. Codeforces Round&num;250 D&period; The Child and Zoo&lpar;并差集)

    题目链接:http://codeforces.com/problemset/problem/437/D 思路:并差集应用,先对所有的边从大到小排序,然后枚举边的时候,如果某条边的两个顶点不在同一个集合 ...

  9. Codeforces Round &num;250 &lpar;Div&period; 2&rpar;A(英语学习)

    链接:http://codeforces.com/contest/437/problem/A A. The Child and Homework time limit per test 1 secon ...

随机推荐

  1. kmdjs api reference

    总览 kmdjs的主要就两个API:kmdjs.config和define kmdjs.config kmdjs.config是用于项目整体配置,一般的配置如下所示: kmdjs.config({ n ...

  2. docker搭建ros-indigo-arm交叉编译环境

    ROS运行环境:ARM ubuntu14.04 + ROS indigo在arm环境下编译ros应用程序,速度极慢,无法忍受,尝试在x86机器上搭建docker+ros交叉编译环境. 交叉编译环境的搭 ...

  3. UEFI与MBR区别

     EFI与MBR启动的区别 大硬盘和WIN8系统,让我们从传统的BIOS+MBR模式升级到UEFI+GPT模式,现在购买的主流电脑,都是预装WIN8系统,为了更好的支持2TB硬盘 ,更快速的启动win ...

  4. UIViewCotroller 的生命周期函数

    Viewcontroller 的所有生命周期函数 重写时 一定要先写 父类 方法 就是(super  +生命周期函数) LoadView ViewDidLoad ViewDidUnload: 在iOS ...

  5. C语言的可变参数

    可变参数给编程带来了很大的方便,在享受它带来的方便的同时,很有必要了解一下其实现方式,在了解编程语言的同时,也可以扩展编程的思路. 可变参数需要用到3个宏函数和一个类型,他们都定义在<stdar ...

  6. &lpar;莱昂氏unix源代码分析导读-49&rpar; 字符缓冲区

    by cszhao1980 同块设备一样,对字符设备的输入输出也是通过缓冲区来进行的.使用缓冲区有个额外 的好处,即以缓冲区为界,函数可分为高低两个层次.低层函数负责与实际设备交互, 而高层函数只与缓 ...

  7. php 系列

    1.给 跑在windows 环境下的php, 安装redis 拓展.(installing Redis & Redis extension in PHP on XAMPP on windows ...

  8. 人生苦短,我用Python

    Life is short, You need Python. 工作中常常要用到脚本来完成许多重复性的工作,刚开始是查数据库的时候,也曾用shell 来写脚本,但终于还是觉得shell太艰涩, 一行命 ...

  9. 韩顺平教学资源java、oracle、linux

    http://blog.itpub.net/28688617/viewspace-766392/

  10. bzoj2856&colon; &lbrack;ceoi2012&rsqb;Printed Circuit Board

    Description 给出一个N个顶点的简单多边形,对于每个顶点,假如它和原点连成的线段只在这个顶点处和多边形相交,就称为满足要求的顶点.你的任务是输出所有满足要求的顶点编号. Input 第一行一 ...