Codeforces Round #499 (Div. 1) F. Tree

题目链接

\(\rm CodeForces\):https://codeforces.com/contest/1010/problem/F

Solution

设\(v_i\)表示第\(i\)个点的果子数,设\(b_i=v_i-\sum_{x\in son}v_x\),显然依题意要满足\(b_i\geqslant 0\)。

根据差分的性质我们可以得到\(\sum b_i=x\)。

假设我们硬点树上剩下了\(m\)个点,则根据插板法\(b_i\)的方案数为\(\displaystyle\binom{x+m-1}{m-1}\)。

由于\(b\)唯一确定\(v\),所以\(v\)的方案数也是这么多。

那么我们就考虑剩下\(m\)个点的方案数。

考虑一种暴力的\(dp\),设\(f[u][i]\)表示\(x\)子树保留了\(i\)个点(包括\(i\))的方案数,转移显然:

\[f[u][i]=\sum_{k=1}^{i-1}f[l][k]\cdot f[r][i-1-k]
\]

写成生成函数就是:

\[F_u(x)=xF_l(x)F_r(x)+1
\]

若\(u\)只有一个儿子也同理:

\[F_u(x)=xF_{son}(x)+1
\]

若\(u\)为叶子则:

\[F_u(x)=x+1
\]

证明显然。

考虑这个玩意怎么优化,显然如果\(\rm NTT\)优化复杂度\(O(n^2\log n)\),这是我们无法接受的。

我们考虑对这棵树进行轻重链剖分,那么对于一条重链上的每个点我们假设求出了非重儿子的\(F(x)\)。

那么我们对这条重链进行编号,从顶端到叶子为\(1\cdots c\),设\(a_i=xF_{son_i}(x)\)。

那么链顶的答案就是:

\[F_1(x)=xF_{son_1}(x)F_2(x)+1=a_1F_2(x)+1
\]

我们递归的写完所有\(c\)个:

\[F_1(x)=a_1(a_2((\cdots(a_c+1))+1)+1)+1
\]

暴力展开就是:

\[F_1(x)=a_1a_2\cdots a_c+a_1a_2\cdots a_{c-1}+\cdots+a_1+1
\]

这个可以分治\(\rm FFT\)解决,代码大概长这样:

void solve(int lt,int rt,vec &a,vec &b) {
if(lt==rt) return a=b=r[lt],void();
vec al,ar,bl,br;int mid=(lt+rt)>>1;solve(lt,mid,al,bl),solve(mid+1,rt,ar,br);
b=poly::pmul(bl,br);a=poly::padd(poly::pmul(ar,bl),al);
}

其中\(al,ar\)表示答案,\(bl,br\)表示区间乘积,其他的变量可以参考下下面的代码。

那么我们就解决了这个问题,因为总轻边的子树大小之和为\(O(n\log n)\),所以总复杂度为\(O(n\log ^3n)\)。

Code

#include<bits/stdc++.h>
using namespace std; void read(int &x) {
x=0;int f=1;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-f;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';x*=f;
} void print(int x) {
if(x<0) putchar('-'),x=-x;
if(!x) return ;print(x/10),putchar(x%10+48);
}
void write(int x) {if(!x) putchar('0');else print(x);putchar('\n');} #define lf double
#define ll long long #define pii pair<int,int >
#define vec vector<int > #define pb push_back
#define mp make_pair
#define fr first
#define sc second
#define _sz(x) ((int)x.size()) #define FOR(i,l,r) for(int i=l,i##_r=r;i<=i##_r;i++) const int maxn = 1<<19|10;
const int inf = 1e9;
const lf eps = 1e-8;
const int mod = 998244353; int add(int x,int y) {return x+y>=mod?x+y-mod:x+y;}
int del(int x,int y) {return x-y<0?x-y+mod:x-y;}
int mul(int x,int y) {return 1ll*x*y-1ll*x*y/mod*mod;} int qpow(int a,int x) {
int res=1;
for(;x;x>>=1,a=mul(a,a)) if(x&1) res=mul(res,a);
return res;
} namespace poly {
int N,w[maxn],pos[maxn],bit,mxn,t[2][maxn]; void init(int l) {
for(mxn=1;mxn<=l;mxn<<=1) ;
w[0]=1,w[1]=qpow(3,(mod-1)/mxn);
for(int i=2;i<=mxn;i++) w[i]=mul(w[i-1],w[1]);
} void ntt(int *r,int op) {
FOR(i,1,N-1) if(pos[i]>i) swap(r[pos[i]],r[i]);
for(int i=1,d=mxn>>1;i<N;i<<=1,d>>=1)
for(int j=0;j<N;j+=i<<1)
for(int k=0;k<i;k++) {
int x=r[j+k],y=mul(r[i+j+k],w[k*d]);
r[j+k]=add(x,y),r[i+j+k]=del(x,y);
}
if(op==-1) {
reverse(r+1,r+N);int d=qpow(N,mod-2);
for(int i=0;i<N;i++) r[i]=mul(r[i],d);
}
} vec pmul(vec a,vec b) {
if(1ll*_sz(a)*_sz(b)<=5000) {
vec c;c.resize(_sz(a)+_sz(b)-1);
FOR(i,0,_sz(a)-1) FOR(j,0,_sz(b)-1) c[i+j]=add(c[i+j],mul(a[i],b[j]));
return c;
}
for(N=1,bit=0;N<_sz(a)+_sz(b);N<<=1,bit++);
FOR(i,0,N-1) pos[i]=pos[i>>1]>>1|((i&1)<<(bit-1));
FOR(i,0,_sz(a)-1) t[0][i]=a[i];FOR(i,_sz(a),N) t[0][i]=0;
FOR(i,0,_sz(b)-1) t[1][i]=b[i];FOR(i,_sz(b),N) t[1][i]=0;
ntt(t[0],1),ntt(t[1],1);
FOR(i,0,N-1) t[0][i]=mul(t[0][i],t[1][i]);
ntt(t[0],-1);vec c;
FOR(i,0,_sz(a)+_sz(b)-1) c.pb(t[0][i]);
return c;
} vec padd(vec a,vec b) {
if(_sz(a)>_sz(b)) {FOR(i,0,_sz(b)-1) a[i]=add(a[i],b[i]);return a;}
FOR(i,0,_sz(a)-1) b[i]=add(a[i],b[i]);return b;
}
} ll k;
int n,ch[maxn],head[maxn],tot,sz[maxn],F[maxn],cnt;
struct edge{int to,nxt;}e[maxn<<1]; vec f[maxn],r[maxn]; void ins(int u,int v) {e[++tot]=(edge){v,head[u]},head[u]=tot;} void dfs(int x,int fa) {
sz[x]=1;F[x]=fa;
for(int i=head[x],v;i;i=e[i].nxt)
if((v=e[i].to)!=fa) {dfs(v,x);sz[x]+=sz[v];if(sz[ch[x]]<sz[v]) ch[x]=v;}
} void solve(int lt,int rt,vec &a,vec &b) {
if(lt==rt) return a=b=r[lt],void();
vec al,ar,bl,br;int mid=(lt+rt)>>1;solve(lt,mid,al,bl),solve(mid+1,rt,ar,br);
b=poly::pmul(bl,br);a=poly::padd(poly::pmul(ar,bl),al);
} vec dfs2(int x) {
for(int t=x;t;t=ch[t]) {
for(int i=head[t];i;i=e[i].nxt) if(e[i].to!=F[t]&&e[i].to!=ch[t]) f[t]=dfs2(e[i].to);
if(_sz(f[t])<1) f[t].resize(1);f[t][0]++;
f[t].insert(f[t].begin(),0);
}
cnt=0;for(int t=x;t;t=ch[t]) r[++cnt]=f[t];
vec a,b;solve(1,cnt,a,b);return a;
} int main() {
read(n),scanf("%lld",&k);poly::init(n<<1);k%=mod;
for(int i=1,x,y;i<n;i++) read(x),read(y),ins(x,y),ins(y,x);
dfs(1,0);vec res=dfs2(1);int t=1,ans=0;
for(int i=1;i<_sz(res);i++) {
ans=add(ans,mul(res[i],t));
t=mul(t,mul((k+i)%mod,qpow(i,mod-2)));
}write(ans);
return 0;
}

Codeforces Round #499 (Div. 1) F. Tree的更多相关文章

  1. Codeforces Round #527 (Div. 3) F. Tree with Maximum Cost 【DFS换根 || 树形dp】

    传送门:http://codeforces.com/contest/1092/problem/F F. Tree with Maximum Cost time limit per test 2 sec ...

  2. Codeforces Round #527 (Div. 3) . F Tree with Maximum Cost

    题目链接 题意:给你一棵树,让你找一个顶点iii,使得这个点的∑dis(i,j)∗a[j]\sum dis(i,j)*a[j]∑dis(i,j)∗a[j]最大.dis(i,j)dis(i,j)dis( ...

  3. Codeforces Round #499 (Div. 2) F. Mars rover_dfs_位运算

    题解: 首先,我们可以用 dfsdfsdfs 在 O(n)O(n)O(n) 的时间复杂度求出初始状态每个点的权值. 不难发现,一个叶子节点权值的取反会导致根节点的权值取反当且仅当从该叶子节点到根节点这 ...

  4. Codeforces Round #499 (Div. 1)

    Codeforces Round #499 (Div. 1) https://codeforces.com/contest/1010 为啥我\(\rm Div.1\)能\(A4\)题还是\(\rm s ...

  5. Codeforces Round #499 (Div. 2)

    Codeforces Round #499 (Div. 2) https://codeforces.com/contest/1011 A #include <bits/stdc++.h> ...

  6. Codeforces Round #499 (Div. 1)部分题解(B,C,D)

    Codeforces Round #499 (Div. 1) 这场本来想和同学一起打\(\rm virtual\ contest\)的,结果有事耽搁了,之后又陆陆续续写了些,就综合起来发一篇题解. B ...

  7. Codeforces Round #485 (Div. 2) F. AND Graph

    Codeforces Round #485 (Div. 2) F. AND Graph 题目连接: http://codeforces.com/contest/987/problem/F Descri ...

  8. Codeforces Round #486 (Div. 3) F. Rain and Umbrellas

    Codeforces Round #486 (Div. 3) F. Rain and Umbrellas 题目连接: http://codeforces.com/group/T0ITBvoeEx/co ...

  9. Codeforces Round #501 (Div. 3) F. Bracket Substring

    题目链接 Codeforces Round #501 (Div. 3) F. Bracket Substring 题解 官方题解 http://codeforces.com/blog/entry/60 ...

随机推荐

  1. Mean Average Precision(mAP),Precision,Recall,Accuracy,F1_score,PR曲线、ROC曲线,AUC值,决定系数R^2 的含义与计算

    背景   之前在研究Object Detection的时候,只是知道Precision这个指标,但是mAP(mean Average Precision)具体是如何计算的,暂时还不知道.最近做OD的任 ...

  2. 【Python】[技术博客] 如何对使用PYQT编写的GUI文件进行单元测试

    如何对使用PYQT编写的GUI文件进行单元测试 想要对PYQT编写的GUI文件进行单元测试,我们主要用到QTest QTest里面包含了一些对窗体的各种控件进行模拟操作的函数,通过QTest对窗体进行 ...

  3. 混合高斯分布与 EM 算法

    极大似然估计在混合高斯分布中遇到的困难 在一般的情况下,对于所得到的样本集,\(X=\left\{x_{1}, \dots, x_{N}\right\}\),我们的目标是最大化似然函数,通过最大化似然 ...

  4. Beef搭建并通过Nginx绑定域名

    Beef和Nginx安装过程这里就不再说明了相关链接:Beef官方安装教程 1. 修改Beef的config.yaml配置文件 xss连接地址改成要绑定的域名 sudo vim /beef/confi ...

  5. url的长度问题

    url最长支持多少字符? 在http协议中,其实并没有对url长度作出限制,往往url的最大长度和用户浏览器和Web服务器有关,不一样的浏览器,能接受的最大长度往往是不一样的,当然,不一样的Web服务 ...

  6. 闭包(python)

    1.闭包的理解 我们可以将闭包理解为一种特殊的函数,这种函数由两个函数的嵌套组成,且称之为外函数和内函数,外函数返回值是内函数的引用,此时就构成了闭包. 2. 闭包的格式 下面用伪代码进行闭包格式的描 ...

  7. sublime px转rem的方法【亲测有效】

    在开发手机网站的时候,我们经常会用到rem来作为单位,但是手动转rem太麻烦了,那么怎么办呢?sublime安装cssrem来进行快捷操作. 安装过程: 第一步:去git上克隆 https://git ...

  8. 【HBase】HBase 单机版安装及使用

    HBase介绍 HBase是一个分布式的.面向列的开源数据库,该技术来源于 Fay Chang 所撰写的Google论文“Bigtable:一个结构化数据的分布式存储系统”.就像Bigtable利用了 ...

  9. Springboot配置连接两个数据库

    背景: 项目中需要从两个不同的数据库查询数据,之前实现方法是:springboot配置连接一个数据源,另一个使用jdbc代码连接. 为了改进,现在使用SpringBoot配置连接两个数据源 实现效果: ...

  10. jstree:重新加载数据集,刷新树

    true:表示获得一个已经存在的jstree实例 $('#tree').jstree(true).destroy();// 清除树节点 // 重新设置树的JSON数据集 $('#tree').jstr ...