LOJ2320「清华集训 2017」生成树计数
由于菜鸡的我实在是没学会上升幂下降幂那一套理论,这里用的是完全普通多项式的做法。
要是有大佬愿意给我讲讲上升幂下降幂那一套东西,不胜感激orz!
首先可以想到prufer序列,如果不会的话可以左转百度。
我们把答案写成prufer序列的形式,这样的话他的贡献是固定的,其中$d_i$表示$i$的出现次数。
$ans = (n-2)! \prod_{i=1}^n \frac{a_i^{d_i+1}}{d_i!}$
把原来的柿子带进来
$ans = \sum_{\sum d_i = n-2} (n-2)! \sum_{i=1}^n a_i^{d_i+1} d_i^m\prod_{j=1}\frac{d_j^m}{d_j!}$
把相同的因数提出来
$ans =(n-2)! \prod_{i=1}^n a_i \sum_{\sum d_i = n-2} \sum_{i=1}^n a_i^{d_i} d_i^m\prod_{j=1}\frac{d_j^m}{d_j!}$
考虑后面的
$ans' = \sum_{\sum d_i = n-2} \sum_{i=1}^n a_i^{d_i} d_i^m\prod_{j=1}\frac{d_j^m}{d_j!}$
为了方便写成生成函数形式,把柿子化成这样
$ans' = \sum_{\sum d_i = n-2} \sum_{i=1}^n \frac{a_i^{d_i} d_i^{2m}}{d_i!}\prod_{j=1,j\neq i}\frac{d_j^m}{d_j!} $
设
$A(x)=\sum_{i=0}^n \frac{i^{2m}}{i!} x^i$
$B(x)=\sum_{i=0}^n \frac{i^m}{i!} x^i$
把它们带进去
$ans '=\sum_{i=1}^n A(a_i) \prod_{j=1,j\neq i}^n B(a_j) \\ = \sum_{i=1}^n \frac{A}{B}(a_i) \prod_{j=1}^n B(a_j)$
对于$\prod B(a_i)$ 根据常见套路,写成$exp(\sum ln(B(a_i)))$ 当然$\frac{A}{B}$也是这么处理。
接下来我们就遇到了更常见的,求$\sum_{i=1}^{n} a_i^k$ 也就是求数列幂和。
这个东西可以戳这里:qwq (额 需要密码,如果你需要的话可以戳我QQ)
这里简略的说一下,就是我们有$F(x)=\sum(a_1^i+...+a_n^i)x^i$ 可以化成这样$F(x) = \sum (a_ix+...+a_i^nx^n)$ 进而写成这样$F(x) = \sum \frac{a_i}{1-a_ix}$
我们令$G(x) = \prod (1-a_ix)$ 所以有$F = -ln'(G)$
对于$G$来说,我们可以直接分治FFT,然后F就可以poly_ln来求
其余的就是写一个多项式全家桶就可以了。
啊,多项式真有趣。
//Love and Freedom.
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#define ll long long
#define inf 20021225
#define mdn 998244353
#define G 3
#define N 400100
using namespace std;
int read()
{
int s=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-') f=-; ch=getchar();}
while(ch>=''&&ch<='') s=s*+ch-'',ch=getchar();
return f*s;
}
int fac[N],inv[N];
int Inv(int x){return 1ll*inv[x]*fac[x-]%mdn;}
void upd(int &x,int y){x+=x+y>=mdn?y-mdn:y;}
int ksm(int bs,int mi)
{
int ans=;
while(mi)
{
if(mi&) ans=1ll*ans*bs%mdn;
bs=1ll*bs*bs%mdn; mi>>=;
}
return ans;
}
int r[N];
int init(int n)
{
int l=,lim=;
while(lim<n) lim<<=,l++;
for(int i=;i<lim;i++)
r[i]=(r[i>>]>>)|((i&)<<l-);
return lim;
}
void ntt(int *a,int lim,int f)
{
for(int i=;i<lim;i++) if(r[i]>i)
swap(a[r[i]],a[i]);
for(int k=,mid=;k<=lim;k<<=,mid<<=)
{
int Wn=ksm(G,(mdn-)/k); if(f) Wn=ksm(Wn,mdn-);
for(int w=,i=;i<lim;i+=k,w=) for(int j=;j<mid;j++,w=1ll*w*Wn%mdn)
{
int x=a[i+j],y=1ll*w*a[i+mid+j]%mdn;
a[i+j]=(x+y)%mdn; a[i+mid+j]=(mdn+x-y)%mdn;
}
}
if(f) for(int kinv=ksm(lim,mdn-),i=;i<lim;i++)
a[i]=1ll*a[i]*kinv%mdn;
}
int poly_mul(int *a,int *b,int *c,int n)
{
int lim=init(n<<);
ntt(a,lim,); ntt(b,lim,);
for(int i=;i<lim;i++) c[i]=1ll*a[i]*b[i]%mdn;
ntt(c,lim,); return lim;
}
int tmp[N];
void poly_inv(int *a,int *ans,int n)
{
if(n==){ans[]=ksm(a[],mdn-); return;}
int mid=n+>>; poly_inv(a,ans,mid);
int lim=init(n<<);
for(int i=;i<n;i++) tmp[i]=a[i];
for(int i=n;i<lim;i++) tmp[i]=;
ntt(tmp,lim,); ntt(ans,lim,);
for(int i=;i<lim;i++)
ans[i]=(2ll-1ll*tmp[i]*ans[i]%mdn+mdn)*ans[i]%mdn;
ntt(ans,lim,);
for(int i=n;i<lim;i++) ans[i]=;
}
int tmp2[N];
void poly_ln(int *a,int *ans,int n)
{
poly_inv(a,tmp2,n);
for(int i=;i<n;i++) ans[i]=1ll*a[i+]*(i+)%mdn;
int lim=poly_mul(tmp2,ans,tmp2,n); tmp2[n-]=;
for(int i=;i<n;i++) ans[i]=1ll*tmp2[i-]*Inv(i)%mdn,tmp2[i-]=;
for(int i=n;i<lim;i++) ans[i]=tmp2[i]=;
ans[]=;
}
int tmp3[N],tmp4[N];
void poly_exp(int *a,int *ans,int n)
{
if(n==){ans[]=; return;}
int mid=n+>>; poly_exp(a,ans,mid);
for(int i=;i<(n<<);i++) tmp2[i]=tmp3[i]=;
for(int i=;i<n;i++) tmp4[i]=a[i]; poly_ln(ans,tmp3,n);
int lim=init(n<<);
ntt(tmp3,lim,); ntt(tmp4,lim,); ntt(ans,lim,);
for(int i=;i<lim;i++)
ans[i]=1ll*(tmp4[i]-tmp3[i]++mdn)%mdn*ans[i]%mdn;
ntt(ans,lim,);
for(int i=n;i<lim;i++) ans[i]=tmp4[i]=;
}
int tmp5[N],tmp6[N];
void solve(int *a,int l,int r)
{
if(l==r) return;
int mid=l+r>>;
solve(a,l,mid); solve(a,mid+,r);
int d=r-l+;
for(int i=l;i<=mid;i++) tmp5[i-l+]=a[i];
for(int i=mid+;i<=r;i++) tmp6[i-mid]=a[i];
tmp5[]=tmp6[]=;
int lim=poly_mul(tmp5,tmp6,tmp5,d);
for(int i=;i<d;i++) a[l+i]=tmp5[i+];
for(int i=;i<=lim;i++) tmp5[i]=tmp6[i]=;
}
void prework(int n)
{
n++; fac[]=;
for(int i=;i<=n;i++) fac[i]=1ll*fac[i-]*i%mdn;
inv[n]=ksm(fac[n],mdn-);
for(int i=n;i;i--) inv[i-]=1ll*inv[i]*i%mdn;
}
int a[N],f[N],n,m,g[N],A[N],B[N],in[N],fr[N],ls[N];
int main()
{
n=read(),m=read(); prework(n);
for(int i=;i<=n;i++) a[i]=read(),f[i]=(mdn-a[i])%mdn;
solve(f,,n); f[]=; poly_ln(f,g,n+);
for(int i=;i<=n;i++) f[i]=1ll*g[i+]*(i+)%mdn;
for(int i=n;i;i--) f[i]=(-f[i-]+mdn)%mdn; f[]=n;
for(int i=;i<=n;i++)
{
A[i]=1ll*ksm(i+,*m)*inv[i]%mdn;
B[i]=1ll*ksm(i+,m)*inv[i]%mdn;
}
int lim=init(n<<|);
poly_ln(B,g,n+); poly_inv(B,in,n+);
for(int i=;i<n;i++) g[i]=1ll*g[i]*f[i]%mdn;
for(int i=n;i<lim;i++) g[i]=in[i]=;
poly_exp(g,ls,n);
for(int i=n;i<lim;i++) ls[i]=;
poly_mul(A,in,fr,n);
for(int i=;i<n;i++) fr[i]=1ll*fr[i]*f[i]%mdn;
for(int i=n;i<lim;i++) fr[i]=;
poly_mul(fr,ls,f,n);
int ans=fac[n-];
for(int i=;i<=n;i++) ans=1ll*ans*a[i]%mdn;
ans=1ll*ans*f[n-]%mdn;
printf("%d\n",ans);
return ;
}
LOJ2320「清华集训 2017」生成树计数的更多相关文章
- Loj 2320.「清华集训 2017」生成树计数
Loj 2320.「清华集训 2017」生成树计数 题目描述 在一个 \(s\) 个点的图中,存在 \(s-n\) 条边,使图中形成了 \(n\) 个连通块,第 \(i\) 个连通块中有 \(a_i\ ...
- 【LOJ】#2320. 「清华集训 2017」生成树计数
题解 我,理解题解,用了一天 我,卡常数,又用了一天 到了最后,我才发现,我有个加法取模,写的是while(c >= MOD) c -= MOD 我把while改成if,时间,少了 六倍. 六倍 ...
- Loj #2331. 「清华集训 2017」某位歌姬的故事
Loj #2331. 「清华集训 2017」某位歌姬的故事 IA 是一名会唱歌的女孩子. IOI2018 就要来了,IA 决定给参赛选手们写一首歌,以表达美好的祝愿.这首歌一共有 \(n\) 个音符, ...
- Loj #2324. 「清华集训 2017」小 Y 和二叉树
Loj #2324. 「清华集训 2017」小 Y 和二叉树 小Y是一个心灵手巧的OIer,她有许多二叉树模型. 小Y的二叉树模型中,每个结点都具有一个编号,小Y把她最喜欢的一个二叉树模型挂在了墙上, ...
- Loj #2321. 「清华集训 2017」无限之环
Loj #2321. 「清华集训 2017」无限之环 曾经有一款流行的游戏,叫做 *Infinity Loop***,先来简单的介绍一下这个游戏: 游戏在一个 \(n \times m\) 的网格状棋 ...
- loj #2325. 「清华集训 2017」小Y和恐怖的奴隶主
#2325. 「清华集训 2017」小Y和恐怖的奴隶主 内存限制:256 MiB时间限制:2000 ms标准输入输出 题目类型:传统评测方式:文本比较 题目描述 "A fight? Co ...
- [LOJ#2330]「清华集训 2017」榕树之心
[LOJ#2330]「清华集训 2017」榕树之心 试题描述 深秋.冷风吹散了最后一丝夏日的暑气,也吹落了榕树脚下灌木丛的叶子.相识数年的Evan和Lyra再次回到了小时候见面的茂盛榕树之下.小溪依旧 ...
- [LOJ#2329]「清华集训 2017」我的生命已如风中残烛
[LOJ#2329]「清华集训 2017」我的生命已如风中残烛 试题描述 九条可怜是一个贪玩的女孩子. 这天她在一堵墙钉了 \(n\) 个钉子,第 \(i\) 个钉子的坐标是 \((x_i,y_i)\ ...
- [LOJ#2328]「清华集训 2017」避难所
[LOJ#2328]「清华集训 2017」避难所 试题描述 "B君啊,你当年的伙伴都不在北京了,为什么你还在北京呢?" "大概是因为出了一些事故吧,否则这道题就不叫避难所 ...
随机推荐
- MFC ATL STL概要
MFC-----应用程序框架 ATL-----写COM的利器 STL-----用来写逻辑部分 MFC: MFC的目标是桌面应用,当然也有网络部分但很不充分.MFC是一套APP ...
- 【mysql】错误代码1308 Invalid use of NULL value
错误原因是: 在最初设计表script_run_detail表时,resut_id忘记勾选不是null选项, 导致存储数据后发现result_id有NULL值,而实际上,我不希望这个字段可以存储NUL ...
- K8S 笔记,请坚持
service 通过 selector 来选择指定 label 的 pod. 如何访问 service 呢?集群内部访问,使用 cluster ip:集群外部访问,使用 NodePort. clust ...
- spss中如何处理极端值、错误值
spss中如何处理极端值.错误值 spss中录入数据以后,第一步不是去分析数据,而是要检验数据是不是有录入错误的,是不是有不合常理的数据,今天我们要做一个描述性统计,进而查看哪些数据是不合理的.下面是 ...
- springMVC+Spring+Mybatis+Redis
SPRINGMVC+MYBATIS+SPRING+REDIS 只作参考,以防忘记使用! mybatis的配置文件: <?xml version="1.0" encoding= ...
- DG on Windows 10 S: 执行任意代码
DG on Windows 10 S: 执行任意代码 windows 10 S版本是什么鬼? 众所周知,我们使用的是windows 10企业版 LTSC.更准确一点,CMD运行winver,我的版本是 ...
- 11.metasploit辅助模块----基本Exp----ARP欺骗中间人MITM----WordPress破解
metasploit辅助模块 信息收集 auxiliary scanners 使用metasploitable靶机 桥接 同一局域网 msfconsole nmap -sT 靶机IP nmap -sS ...
- spark MLlib矩阵四则运算,线性代数
1.导包请看我的上一篇博文,maven项目的包 https://www.cnblogs.com/wuzaipei/p/10965680.html 2.denseMatirx 矩阵四则运算如下 版本不同 ...
- jQuery 获取设置图片 src 的路径
1.获取: $('#img')[0].src 2.设置:$('#img').attr("src", data.CodeUrl);
- .net core 学习小结之 PostMan报415
首先415的官方解释是:对于当前请求的方法和所请求的资源,请求中提交的实体并不是服务器中所支持的格式,因此请求被拒绝. 也就是说我所准备的数据格式并不是后台代码使用的数据格式 后台代码如下 using ...