E. The Child and Binary Tree

不难写出一个递推式fn=∑i=1ngi∑j=0n−ifjfn−i−jf_n = \sum\limits_{i = 1} ^{n}g_i \sum\limits_{j = 0} ^{n - i}f_jf_{n - i - j}fn​=i=1∑n​gi​j=0∑n−i​fj​fn−i−j​,其中gig_igi​表示ccc中有没有iii这个数。

设F(x)F(x)F(x)为fff的生成函数,G(x)G(x)G(x)为ggg的生成函数,则有F(x)=G(x)F(x)F(x)+1F(x) = G(x)F(x)F(x) + 1F(x)=G(x)F(x)F(x)+1。

因为做卷积[x0]G[x ^ 0]G[x0]G是为000的,所以会没有常数项,得在右边加111。

解这个方程有F(x)=1±1−4G(x)2G(x)=21±1−4G(x)F(x) = \frac{1 \pm \sqrt{1 - 4G(x)}}{2G(x)} = \frac{2}{1 \pm \sqrt{1 - 4G(x)}}F(x)=2G(x)1±1−4G(x)​​=1±1−4G(x)​2​,取x=0x = 0x=0时11±1\frac{1}{1 \pm 1}1±11​,所以负号不可取,F(x)=21+1−4G(x)F(x) = \frac{2}{1 + \sqrt{1 - 4G(x)}}F(x)=1+1−4G(x)​2​。

所以只要多项式开根,多项式求逆就可以得到答案了。

#include <bits/stdc++.h>using namespace std;const int mod = 998244353, inv2 = mod + 1 >> 1;namespace Quadratic_residue {struct Complex {int r, i;Complex(int _r = 0, int _i = 0) : r(_r), i(_i) {}};int I2;Complex operator * (const Complex &a, Complex &b) {return Complex((1ll * a.r * b.r % mod  + 1ll * a.i * b.i % mod * I2 % mod) % mod, (1ll * a.r * b.i % mod + 1ll * a.i * b.r % mod) % mod);}Complex quick_pow(Complex a, int n) {Complex ans = Complex(1, 0);while (n) {if (n & 1) {ans = ans * a;}a = a * a;n >>= 1;}return ans;}int get_residue(int n) {mt19937 e(233);if (n == 0) {return 0;}if(quick_pow(n, (mod - 1) >> 1).r == mod - 1) {return -1;}uniform_int_distribution<int> r(0, mod - 1);int a = r(e);while(quick_pow((1ll * a * a % mod - n + mod) % mod, (mod - 1) >> 1).r == 1) {a = r(e);}I2 = (1ll * a * a % mod - n + mod) % mod;int x = quick_pow(Complex(a, 1), (mod + 1) >> 1).r, y = mod - x;if(x > y) swap(x, y);return x;}
}const int N = 1e6 + 10;int r[N], inv[N], b[N], c[N], d[N], e[N], t[N];int quick_pow(int a, int n) {int ans = 1;while (n) {if (n & 1) {ans = 1ll * a * ans % mod;}a = 1ll * a * a % mod;n >>= 1;}return ans;
}void get_r(int lim) {for (int i = 0; i < lim; i++) {r[i] = (i & 1) * (lim >> 1) + (r[i >> 1] >> 1);}
}void get_inv(int n) {inv[1] = 1;for (int i = 2; i <= n; i++) {inv[i] = 1ll * (mod - mod / i) * inv[mod % i] % mod;}
}void NTT(int *f, int lim, int rev) {for (int i = 0; i < lim; i++) {if (i < r[i]) {swap(f[i], f[r[i]]);}}for (int mid = 1; mid < lim; mid <<= 1) {int wn = quick_pow(3, (mod - 1) / (mid << 1));for (int len = mid << 1, cur = 0; cur < lim; cur += len) {int w = 1;for (int k = 0; k < mid; k++, w = 1ll * w * wn % mod) {int x = f[cur + k], y = 1ll * w * f[cur + mid + k] % mod;f[cur + k] = (x + y) % mod, f[cur + mid + k] = (x - y + mod) % mod;}}}if (rev == -1) {int inv = quick_pow(lim, mod - 2);reverse(f + 1, f + lim);for (int i = 0; i < lim; i++) {f[i] = 1ll * f[i] * inv % mod;}}
}void polyinv(int *f, int *g, int n) {if (n == 1) {g[0] = quick_pow(f[0], mod - 2);return ;}polyinv(f, g, n + 1 >> 1);for (int i = 0; i < n; i++) {t[i] = f[i];}int lim = 1;while (lim < 2 * n) {lim <<= 1;}get_r(lim);NTT(t, lim, 1);NTT(g, lim, 1);for (int i = 0; i < lim; i++) {int cur = (2 - 1ll * g[i] * t[i] % mod + mod) % mod;g[i] = 1ll * g[i] * cur % mod;t[i] = 0;}NTT(g, lim, -1);for (int i = n; i < lim; i++) {g[i] = 0;}
}void polysqrt(int *f, int *g, int n) {if (n == 1) {g[0] = Quadratic_residue::get_residue(f[0]);return ;}polysqrt(f, g, n + 1 >> 1);polyinv(g, b, n);int lim = 1;while (lim < 2 * n) {lim <<= 1;}get_r(lim);for (int i = 0; i < n; i++) {t[i] = f[i];}NTT(g, lim, 1);NTT(b, lim, 1);NTT(t, lim, 1);for (int i = 0; i < lim; i++) {g[i] = (1ll * inv2 * g[i] % mod + 1ll * inv2 * b[i] % mod * t[i] % mod) % mod;b[i] = t[i] = 0;}NTT(g, lim, -1);for (int i = n; i < lim; i++) {g[i] = 0;}
}void derivative(int *a, int *b, int n) {for (int i = 0; i < n; i++) {b[i] = 1ll * a[i + 1] * (i + 1) % mod;}
}void integrate(int *a, int n) {for (int i = n - 1; i >= 1; i--) {a[i] = 1ll * a[i - 1] * inv[i] % mod;}a[0] = 0;
}void polyln(int *f, int *g, int n) {polyinv(f, b, n);derivative(f, g, n);int lim = 1;while (lim < 2 * n) {lim <<= 1;}get_r(lim);NTT(g, lim, 1);NTT(b, lim, 1);for (int i = 0; i < lim; i++) {g[i] = 1ll * g[i] * b[i] % mod;b[i] = 0;}NTT(g, lim, -1);for (int i = n; i < lim; i++) {g[i] = 0;}integrate(g, n);
}void polyexp(int *f, int *g, int n) {if (n == 1) {g[0] = 1;return ;}polyexp(f, g, n + 1 >> 1);int lim = 1;while (lim < 2 * n) {lim <<= 1;}polyln(g, d, n);for (int i = 0; i < n; i++) {t[i] = (f[i] - d[i] + mod) % mod;}t[0] = (t[0] + 1) % mod;get_r(lim);NTT(g, lim, 1);NTT(t, lim, 1);for (int i = 0; i < lim; i++) {g[i] = 1ll * g[i] * t[i] % mod;t[i] = d[i] =  0;}NTT(g, lim, -1);for (int i = n; i < lim; i++) {g[i] = 0;}
}/*b存放多项式逆,c存放多项式开根,d存放多项式对数ln,e存放多项式指数exp,t作为中间转移数组,如果要用到polyinv,得提前调用get_inv(n)先预先得到我们想要得到的逆元范围。
*/int a[N], n, m;int main() {// freopen("in.txt", "r", stdin);// freopen("out.txt", "w", stdout);// ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);scanf("%d %d", &n, &m);for (int i = 1, x; i <= n; i++) {scanf("%d", &x);a[x]++;}for (int i = 0; i <= 100000; i++) {a[i] = (mod - 4 * a[i]) % mod;}a[0] = 1;get_inv(200010);polysqrt(a, c, 200005);for (int i = 0; i < 200005; i++) {a[i] = c[i];c[i] = 0;}a[0] = (a[0] + 1) % mod;polyinv(a, b, 200005);for (int i = 1; i <= m; i++) {printf("%lld\n", 1ll * b[i] * 2 % mod);}return 0;
}

E. The Child and Binary Tree(生成函数 + 多项式)相关推荐

  1. CF438E The Child and Binary Tree 生成函数、多项式开根

    传送门 设生成函数\(C(x) = \sum\limits_{i=0}^\infty [\exists c_j = i]x^i\),答案数组为\(f_1 , f_2 , ..., f_m\),\(F( ...

  2. CF438E The Child and Binary Tree(有意思的生成函数 + 多项式求逆 + 多项式开方)

    整理的算法模板合集: ACM模板 点我看多项式全家桶(●^◡_◡◡​^●) CF438E The Child and Binary Tree 简单的黑题 首先我们发现模数为99824435399824 ...

  3. [bzoj3625][Codeforces 250 E]The Child and Binary Tree(生成函数+多项式运算+FFT)

    3625: [Codeforces Round #250]小朋友和二叉树 Time Limit: 40 Sec  Memory Limit: 256 MB Submit: 650  Solved: 2 ...

  4. CF438E:The Child and Binary Tree(生成函数)

    解析: 设计 fif_ifi​ 表示权值为 iii 的方案数,f0=1f_0=1f0​=1. 枚举根节点权值,可以写出转移: fn=∑gk∑ififn−k−i=∑i+j+k=nfifjgkf_n=\s ...

  5. CF438E-The Child and Binary Tree【生成函数】

    正题 题目链接:https://www.luogu.com.cn/problem/CF438E 题目大意 每个节点有nnn个权值可以选择,对于1∼m1\sim m1∼m中的每个数字kkk,求权值和为k ...

  6. bzoj 3625(CF 438E)The Child and Binary Tree——多项式开方

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3625 http://codeforces.com/contest/438/problem/E ...

  7. PAT(甲级)2019年春季考试 7-4 Structure of a Binary Tree

    目录 整体思路 犯的错误 代码 整体思路 1.先根据后序和中序序列建树,老生常谈,记得返回root 2.对树进行BFS,在这个过程中用hash的方式记录下每个值对应的父节点.左孩子.右孩子的值,记录下 ...

  8. LeetCode 102. Binary Tree Level Order Traversal--递归,迭代-Python,Java解法

    题目地址: Given a binary tree, return the level order traversal of its nodes' values. (ie, from left to ...

  9. PAT甲级——1102 Invert a Binary Tree (层序遍历+中序遍历)

    本文同步发布在CSDN:https://blog.csdn.net/weixin_44385565/article/details/90577042 1102 Invert a Binary Tree ...

最新文章

  1. Dotfuscator代码混淆工具的使用
  2. LightSpeed 的Left Join Bug解决方案
  3. Python从题目中学习:random() module
  4. 认识哈希函数(散列函数)
  5. Python 常见优化技巧,让你的程序溜溜的跑起来!
  6. php prism,漂亮的代码语法高亮库:Prism.js
  7. windows上hadoop安装(cygwin等)
  8. Pytest之pytest.assume用例中断言1失败会继续执行后续代码断言2
  9. js多种方法:返回上一页
  10. 计算机控制专业代码,专业代码080605.doc
  11. TensorFlow2.0:高阶操作
  12. mongoimport csv文件
  13. 无锁循环缓冲区的实现c语言,C++ 无锁环形缓冲区实现
  14. 如何用代码表白——matlab绘制玫瑰、爱心和I LOVE YOU
  15. 图论的应用 计算机,图论的应用计算机技术与科学毕业论文.doc
  16. 移位运算符 java_java.移位运算符
  17. 计算机无法关闭密码保护,Win7密码保护共享关闭不了怎么办?密码保护共享关不掉的解决方法...
  18. Android会议室管理app
  19. linux环境下php安装sqlsrv扩展连接mssql
  20. ThinkPHP 2

热门文章

  1. 2560x1600分辨率高吗_做设计还弄不清分辨率和像素之间的关系,来了解下他们是怎么换算...
  2. 释放内存软件_原来苹果手机这样清理内存,可以释放大量空间,真是太好用了...
  3. 女生的拳头有多厉害?
  4. 你所阅读的,决定你是什么样的人
  5. mysql对null排序_mysql中null值的排序问题分析_MySQL
  6. mybatis-plus 会自动增加 order by_python自动撸支付宝基金答题红包
  7. python生成器yield原理_Python generator生成器和yield表达式详解
  8. ts定义数组类型_ts基本数据类型
  9. java读取图片缩略方法_java 图片缩略图的两种方法
  10. 汉字为什么能流传至今_汉字能流传至今,比毛不易还不易,它的同龄字统统都死掉了...