树的解构 题解
bsoj7076,没找到出处…
树的解构
Mivik 喜欢 Eprom 的解构俱乐部,于是他想解构一棵树。
Mivik 找到了一棵以
为根的有 个结点的有根外向树。Mivik 会进行 次操作,每次 Mivik 都会从未删掉的边中等概率选择一条边将其删去。记这条边为 ,则删去这条边的代价是删边时 的子树大小(包括 自己);删去这条边后 为根的子树会形成一棵新的以 为根的有根树。 Mivik 想知道,他进行这
次操作后期望的代价总和是多少。由于 Mivik 不喜欢太大的数,你只需要输出期望的值对 取模的结果。 对于所有测试点,满足
。保证给出的有根树合法。
一道并不是特别难但没有切掉的期望题。
考场写了链过后就一直在想怎么拆树,无果而终。然后想到解决排列问题的一个常见的思路是考虑所有排列方案的贡献和,最后乘个
对整体dp不好做,那就转而思考每个点对答案的贡献。
考虑从树上某个深度(到根需经过的边数)为
对于其中某条位置为 意外的约掉了好多玩意儿呢
线性求逆元并前缀和即可
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<ctime>
#include<cstdlib>
#include<queue>
#include<algorithm>
#include<map>
#include<set>
#include<vector>
using namespace std;
typedef long long ll;
(){
ll Rdint ans=0;bool fh=0;char c=getchar();
while(c<'0'||c>'9'){if(c=='-') fh=1; c=getchar();}
while(c>='0'&&c<='9') ans=ans*10+c-'0',c=getchar();
if(fh) ans=-ans;
return ans;
}
typedef vector<ll> Vec;
typedef vector<ll>::iterator IVec;
#define foreach(it,v) for(IVec it=v.begin();it!=v.end();it++)
const ll MOD=1E9+7;
#define _ %MOD
(ll x){
ll PModif(x<0) return x+MOD;
else if(x>=MOD) return x-=MOD;
else return x;
}
const ll MXN=2E6+5;
;Vec chd[MXN];
ll N
[MXN];
ll depvoid InfoDFS(ll u,ll depth){
[u]=depth;
depforeach(it,chd[u]) InfoDFS((*it),depth+1);
}
[MXN],sInv[MXN];
ll invvoid SpawnInv(){
[1]=1;for(ll i=2;i<=N;i++) inv[i]=PMod(-(MOD/i)*inv[MOD%i]_);
inv[1]=inv[1];for(ll i=2;i<=N;i++) sInv[i]=PMod(sInv[i-1]+inv[i]);
sInv}
void Solve(){
();
SpawnInv(1,0);
InfoDFS=0;
ll ansfor(ll u=1;u<=N;u++) ans=PMod(ans+sInv[dep[u]]);
("%lld",ans);
printf}
int main(){
//freopen("deconstruct.in","r",stdin);
//freopen("deconstruct.out","w",stdout);
=Rd();
Nfor(ll i=2;i<=N;i++) chd[Rd()].push_back(i);
();
Solvereturn 0;
}