【CF438E】小朋友和二叉树

Description

​ 我们的小朋友很喜欢计算机科学,而且尤其喜欢二叉树。

​ 考虑一个含有\(n\)个互异正整数的序列\(c_1,c_2,\dots,c_n\)。如果一棵带点权的有根二叉树满足其所有顶点的权值都在集合\(\{c_1,c_2,\dots,c_n\}\)中,我们的小朋友就会将其称作神犇的。并且他认为,一棵带点权的树的权值,是其所有顶点权值的总和。

​ 给出一个整数\(m\),你能对于任意的\(s(1≤s≤m)\)计算出权值为\(s\)的神犇二叉树的个数吗?请参照样例以更好的理解什么样的两棵二叉树会被视为不同的。

​ 我们只需要知道答案关于\(998244353\)取模后的值。

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\)取模后的结果。


考虑\(DP\),设\(f_i\)代表权值为\(i\)的二叉树的个数,\(C_i\)代表是否存在权值为\(i\)的节点

\[f_n=\sum_{i=1}^nC_i\sum_{j=0}^{n-i}f_jf_{n-i-j}
\]

\[f_0=1
\]

然后我们发现长得很像卷积,但是没法好好卷自己。

于是构造一波生成函数,直接表示为系数就行了

\[F=C*F*F+1
\]

关于这个\(+1\),我的理解是,加了一个常数项为\(1\)的多项式表示\(f_0=1\)

然后解一下二次方程,得到

\[F=\frac{1 \pm \sqrt{1-4C}}{2C}
\]

然后讨论一下发现需要取+

再次化简

\[F=\frac{2}{1+\sqrt{1-4C}}
\]

套用多项式求逆+开根就可以了


Code:

#include <cstdio>
#include <cctype>
#include <algorithm>
const int N=(1<<20)+10;
const int mod=998244353,Gi=332748118,i2=499122177;
int read()
{
int x=0;char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) {x=x*10+c-'0';c=getchar();}
return x;
}
#define add(x,y) ((x+y)%mod)
#define mul(x,y) (1ll*(x)*(y)%mod)
int qp(int d,int k){int f=1;while(k){if(k&1)f=mul(f,d);d=mul(d,d),k>>=1;}return f;}
int C[N],sq[2][N],b[2][N],A[N],B[N],turn[N];
void NTT(int *a,int len,int typ)
{
for(int i=1;i<len;i++) if(i<turn[i]) std::swap(a[i],a[turn[i]]);
for(int le=1;le<len;le<<=1)
{
int wn=qp(typ?3:Gi,(mod-1)/(le<<1));
for(int p=0;p<len;p+=le<<1)
{
int w=1;
for(int i=p;i<p+le;i++,w=mul(w,wn))
{
int tx=a[i],ty=mul(w,a[i+le]);
a[i]=add(tx,ty);
a[i+le]=add(tx,mod-ty);
}
}
}
if(!typ)
{
int inv=qp(len,mod-2);
for(int i=0;i<len;i++) a[i]=mul(a[i],inv);
}
}
void polymul(int *a,int *b,int len)
{
int L=-1;for(int i=1;i<len;i<<=1) ++L;
for(int i=0;i<len;i++) turn[i]=turn[i>>1]>>1|(i&1)<<L,A[i]=B[i]=0;
for(int i=0;i<len>>1;i++) A[i]=a[i],B[i]=b[i];
NTT(A,len,1),NTT(B,len,1);
for(int i=0;i<len;i++) A[i]=mul(A[i],B[i]);
NTT(A,len,0);
for(int i=0;i<len;i++) a[i]=A[i];
}
void polyinv(int *a,int n)
{
int len=2,cur=0;
b[cur][0]=qp(a[0],mod-2);
while(len<=(n<<2))
{
cur^=1;
for(int i=0;i<len>>1;i++) b[cur][i]=add(b[cur^1][i],b[cur^1][i]);
polymul(b[cur^1],b[cur^1],len);
polymul(b[cur^1],a,len);
for(int i=0;i<len;i++) b[cur][i]=add(b[cur][i],mod-b[cur^1][i]);
len<<=1;
}
for(int i=0;i<n;i++) a[i]=b[cur][i];
}
void polysqrt(int *a,int n)
{
int len=2,cur=0;
sq[cur][0]=1;
while(len<=(n<<2))
{
cur^=1;
for(int i=0;i<len>>1;i++) sq[cur][i]=mul(sq[cur^1][i],i2);
for(int i=0;i<len>>1;i++) sq[cur^1][i]=add(sq[cur^1][i],sq[cur^1][i]);
polyinv(sq[cur^1],len);
polymul(sq[cur^1],a,len);
for(int i=0;i<len;i++) sq[cur][i]=add(sq[cur][i],sq[cur^1][i]);
len<<=1;
}
for(int i=0;i<n;i++) a[i]=sq[cur][i];
}
int main()
{
int n=read(),m=read();
for(int i=1;i<=n;i++) C[read()]=1;
++m;C[0]=1;
for(int i=1;i<m;i++) C[i]=(mod-(C[i]<<2))%mod;
polysqrt(C,m);
C[0]=add(C[0],1);
polyinv(C,m);
for(int i=0;i<m;i++) C[i]=add(C[i],C[i]);
for(int i=1;i<m;i++) printf("%d\n",C[i]);
return 0;
}

2018.12.17

【CF438E】小朋友和二叉树 解题报告的更多相关文章

  1. 【九度OJ】题目1113:二叉树 解题报告

    [九度OJ]题目1113:二叉树 解题报告 标签(空格分隔): 九度OJ http://ac.jobdu.com/problem.php?pid=1113 题目描述: 如上所示,由正整数1,2,3-- ...

  2. 【剑指Offer】从上往下打印二叉树 解题报告(Python)

    [剑指Offer]从上往下打印二叉树 解题报告(Python) 标签(空格分隔): 剑指Offer 题目地址:https://www.nowcoder.com/ta/coding-interviews ...

  3. 【剑指Offer】序列化二叉树 解题报告(Python)

    [剑指Offer]序列化二叉树 解题报告(Python) 标签(空格分隔): 剑指Offer 题目地址:https://www.nowcoder.com/ta/coding-interviews 题目 ...

  4. 【剑指Offer】按之字形顺序打印二叉树 解题报告(Python)

    [剑指Offer]按之字形顺序打印二叉树 解题报告(Python) 标签(空格分隔): 剑指Offer 题目地址:https://www.nowcoder.com/ta/coding-intervie ...

  5. [BZOJ3625][CF438E]小朋友和二叉树

    题面 Description 我们的小朋友很喜欢计算机科学,而且尤其喜欢二叉树. 考虑一个含有\(n\)个互异正整数的序列\(c_1,c_2,\ldots,c_n\).如果一棵带点权的有根二叉树满足其 ...

  6. 【剑指Offer】07. 重建二叉树 解题报告(Java & Python & C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 个人微信公众号:负雪明烛 目录 题目描述 解题方法 基本方法:线性查找根节点的位置 方法优 ...

  7. [CF438E] 小朋友和二叉树

    Description 给定一个整数集合 \(c\),对于每个 \(i\in[1,m]\),求有多少种不同的带点权的二叉树使得这棵树点权和为 \(i\) 并且顶点的点权全部在集合 \(c\) 中.\( ...

  8. [BZOJ3625][CF438E]小朋友和二叉树 (多项式开根,求逆)

    题面 题解 设多项式的第a项为权值和为a的二叉树个数,多项式的第a项表示是否为真,即 则,所以F是三个多项式的卷积,其中包括自己: ,1是F的常数项,即. 我们发现这是一个一元二次方程,可以求出,因为 ...

  9. BZOJ3625 CF438E 小朋友与二叉树

    心态崩了 不放传送门了 辣鸡bz 还是正经一点写一下题解= = 就是显然我们可以把权值写成生成函数形式g(0/1序列)来表示权值是否出现 然后f来表示总的方案数 可以列出 分别枚举左右子树和空树的情况 ...

随机推荐

  1. python函数式编程,性能,测试,编码规范

    这篇文章主要是对我收集的一些文章的摘要.因为已经有很多比我有才华的人写出了大量关于如何成为优秀Python程序员的好文章. 我的总结主要集中在四个基本题目上:函数式编程,性能,测试,编码规范.如果一个 ...

  2. 《Pro SQL Server Internals, 2nd edition》的CHAPTER 1 Data Storage Internals中的Data Pages and Data Rows(翻译)

    数据页和数据行 数据库中的空间被划分为逻辑8KB的页面.这些页面是以0开始的连续编号,并且可以通过指定文件ID和页号来引用它们.页面编号都是连续的,这样当SQL Server增长数据库文件时,从文件中 ...

  3. 【翻译】HOG, Histogram of Oriented Gradients / 方向梯度直方图 介绍

    本文翻译自 SATYA MALLICK 的 "Histogram of Oriented Gradients" 原文链接: https://www.learnopencv.com/ ...

  4. Netty源码分析第3章(客户端接入流程)---->第5节: 监听读事件

    Netty源码分析第三章: 客户端接入流程 第五节: 监听读事件 我们回到AbstractUnsafe的register0()方法: private void register0(ChannelPro ...

  5. PHP中 post方法 与 get方法 的区别

    1.Get 方法通过 URL 请求来传递用户的数据,将表单内各字段名称与其内容,以成对的字符串连接,置于 action 属性所指程序的 url 后,如[url]http://www.domain.co ...

  6. 基于LiFi可见光通信技术的研究及应用转化调查

    这个仅是本人的部分调研结果,有同行做可见光研究的可以联系交流,QQ:391349683 

  7. 第10讲:利用SQL语言实现关系代数操作

    一.实现并.交.差运算 1. 基本语法形式:子查询 [union [all] | intersect [all] | except [all] 子查询] ①意义:将关系代数中的∪.∩.- 分别用uni ...

  8. Scrum立会报告+燃尽图(十二月七日总第三十八次):功能测试

    此作业要求参见:https://edu.cnblogs.com/campus/nenu/2018fall/homework/2284 项目地址:https://git.coding.net/zhang ...

  9. Daily Scrum (2015/10/23)

    这天晚上PM和我一起细算下来这周的确做了不少事儿.由于这天是周五,有的组员今晚有外出活动,有的组员忙了一周想休息一下.所以PM与我讨论提出今晚就布置些阅读的任务,给组员们一些自由分配的时间: 成员 今 ...

  10. 信息安全系统设计基础_exp1

    北京电子科技学院(BESTI) 实     验    报     告 课程:信息安全系统设计基础 班级:1353 姓名:吴子怡.郑伟 学号:20135313.20135322 指导教师: 娄嘉鹏 实验 ...