题解-Codeforces917D Stranger Trees
Problem
题意概要:一棵 \(n\) 个节点的无向树。问在 \(n\) 个点的完全图中,有多少生成树与原树恰有 \(k\) 条边相同,对于任意 \(k\in[0,n)\) 输出答案,答案取模。
\(2\leq n\leq 100\)
Solution
这题思路新奇啊,智商又能上线了
由于暴力为枚举所有生成树,发现枚举所有生成树的高效算法为矩阵树定理,而且数据范围恰好在矩阵树复杂度接受范围内
由于矩阵树计算的是所有 生成树边权积 之和,考虑给完全图中每一条边边权设为 \(1\),若是树边则设为 \(x\),最后矩阵树消出来的行列式为一个多项式,这个多项式中 \(k\) 次项的系数即 与原树重合恰好 \(k\) 条边的生成树个数
考虑将多项式代入行列式去消不方便,可以采用代入几个数字去算,最后用拉格朗日插值去算系数,由于最后的多项式为 \(n\) 项系数,所以需要用 \(n+1\) 个值去代,不妨设为 \(1...n+1\)
总体时间复杂度 \(O(n^4)\),比容斥Dp慢多了……
拉格朗日差值
刚好正在复习拉格朗日差值,附上大致流程:
求 \(F(x)=\sum a_ix^i\),使得 \(\forall i\in[1,n],F(x_i)=y_i\)
令 \(F(x)=\sum_if_i(x)\),其中\(f_i(x_j)=\begin{cases}y_j,& j=i\\0,& j\not =i\end{cases}\)
由定义可设 \(f_i(x)=c_i\prod_{j\not = i}(x-x_j)=y_i\),可以解出 \(c_i\),反过来得到 \(f_i(x)\),最后求和得到 \(F(x)\)
其中求 \(\prod_{j\not = i}(x-x_j)\) 时,可以先预处理出 \(\prod (x-x_j)\),对于每一个 \(i\) 都将上式除以 \((x-x_i)\),这样二项式乘除法都是 \(O(n^2)\) 的
整个算法复杂度 \(O(n^2)\)
Code
//Codeforces-917D
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
inline void read(int&x){
char c11=getchar();x=0;while(!isdigit(c11))c11=getchar();
while(isdigit(c11))x=x*10+c11-'0',c11=getchar();
}
const int N = 113, p = 1e9+7;
struct Edge{int l,r;}e[N];
int Y[N],n;
inline int qpow(int A,int B){
int res = 1; while(B){
if(B&1) res = (ll)res * A%p;
A = (ll)A * A%p, B >>= 1;
}return res;
}
namespace Matrix_Tree{
int a[N][N];
int Gauss(){
for(int i=1;i<n;++i){
if(!a[i][i])
for(int j=i+1;j<n;++j)
if(a[j][i]){
for(int k=i;k<n;++k)
swap(a[i][k],a[j][k]);
break;
}
for(int j=i+1;j<n;++j)
if(a[j][i]){
int ki = (ll)a[j][i] * qpow(a[i][i],p-2)%p;
for(int k=i;k<n;++k)
a[j][k] = (a[j][k] - (ll)ki * a[i][k]%p +p)%p;
}
}
int res = 1;
for(int i=1;i<n;++i)
res = (ll)res * a[i][i]%p;
return res;
}
int calc(int x){
for(int i=1;i<n;++i)
for(int j=1;j<n;++j)
a[i][j] = 0;
for(int i=1;i<n;++i)
a[i][i] = n - 1;
for(int i=1,l,r;i<n;++i){
l = e[i].l, r = e[i].r;
a[l][r] = a[r][l] = p - x;
a[l][l] = (a[l][l] + x - 1)%p;
a[r][r] = (a[r][r] + x - 1)%p;
}
for(int i=1;i<n;++i)
for(int j=1;j<n;++j)
if(i!=j and !a[i][j])
a[i][j] = p - 1;
return Gauss();
}
}
namespace Lagrange{
int Ans[N],S[N],a[N],tmp[N];
inline int calc(int n,int x){
int res = 0, pw = 1;
for(int i=0;i<=n;++i){
res = (res + (ll)pw * a[i])%p;
pw = (ll)pw * x %p;
}return res;
}
void work(){
++n, S[0] = 1;
for(int i=1;i<=n;++i){
for(int j=i;j;--j)
S[j] = S[j-1];
S[0] = 0;
for(int j=0;j<i;++j)
S[j] = (S[j] - (ll)S[j+1] * i%p +p)%p;
}
for(int i=1;i<=n;++i){
for(int j=0;j<=n;++j) tmp[j] = S[j];
for(int j=n;j;--j){
a[j-1] = tmp[j];
tmp[j-1] = (tmp[j-1] + (ll)i * tmp[j])%p;
}
int v = calc(n-1,i);
v = (ll)qpow(v,p-2) * Y[i]%p;
for(int j=0;j<n;++j)
Ans[j] = (Ans[j] + (ll)v * a[j])%p;
}
for(int i=0;i<n-1;++i)
printf("%d ",Ans[i]);
putchar(10);
}
}
int main(){
read(n);
for(int i=1,x,y;i<n;++i) read(e[i].l),read(e[i].r);
for(int i=1;i<=n+1;++i) Y[i] = Matrix_Tree::calc(i);
Lagrange::work();
return 0;
}
题解-Codeforces917D Stranger Trees的更多相关文章
- Codeforces917D. Stranger Trees
$n \leq 100$的完全图,对每个$0 \leq K \leq n-1$问生成树中与给定的一棵树有$K$条公共边的有多少个,答案$mod \ \ 1e9+7$. 对这种“在整体中求具有某些特性的 ...
- 【CF917D】Stranger Trees 树形DP+Prufer序列
[CF917D]Stranger Trees 题意:给你一棵n个点的树,对于k=1...n,问你有多少有标号的n个点的树,与给出的树有恰好k条边相同? $n\le 100$ 题解:我们先考虑容斥,求出 ...
- CF917D Stranger Trees
CF917D Stranger Trees 题目描述 给定一个树,对于每个\(k=0,1\cdots n-1\),问有多少个生成树与给定树有\(k\)条边重合. 矩阵树定理+高斯消元 我们答案为\(f ...
- 题解 CF917D 【Stranger Trees】
生成树计数问题用矩阵树定理来考虑. 矩阵树定理求得的为\(\sum\limits_T\prod\limits_{e\in T}v_e\),也就是所有生成树的边权积的和. 这题边是不带权的,应用矩阵树定 ...
- codeforces 917D Stranger Trees
题目链接 正解:矩阵树定理+拉格朗日插值. 一下午就搞了这一道题,看鬼畜英文题解看了好久.. 首先这题出题人给了两种做法,感觉容斥+$prufer$序列+$dp$的做法细节有点多所以没看,然而这个做法 ...
- 【CF917D】Stranger Trees
题目 看题解的时候才突然发现\(zky\)讲过这道题啊,我现在怕不是一个老年人了 众所周知矩阵树求得是这个 \[\sum_{T}\prod_{e\in T}w_e\] 而我们现在的这个问题有些鬼畜了, ...
- CF917D. Stranger Trees & TopCoder13369. TreeDistance(变元矩阵树定理+高斯消元)
题目链接 CF917D:https://codeforces.com/problemset/problem/917/D TopCoder13369:https://community.topcoder ...
- Codeforces 917D - Stranger Trees(矩阵树定理/推式子+组合意义)
Codeforces 题目传送门 & 洛谷题目传送门 刚好看到 wjz 在做这题,心想这题之前好像省选前做过,当时觉得是道挺不错的题,为啥没写题解呢?于是就过来补了,由此可见我真是个大鸽子(( ...
- LeetCode题解之Leaf-Similar Trees
1.题目描述 2.问题分析 将叶子节点的值放入vector,然后比较. 3.代码 bool leafSimilar(TreeNode* root1, TreeNode* root2) { vector ...
随机推荐
- SSM(Spring + Springmvc + Mybatis)框架面试题
JAVA SSM框架基础面试题https://blog.csdn.net/qq_39031310/article/details/83050192 SSM(Spring + Springmvc + M ...
- Docker 核心技术之容器
什么是容器 容器(Container) 容器是一种轻量级.可移植.并将应用程序进行的打包的技术,使应用程序可以在几乎任何地方以相同的方式运行 Docker将镜像文件运行起来后,产生的对象就是容器.容器 ...
- 转: 通过WMI获取网卡MAC地址、硬盘序列号、主板序列号、CPU ID、BIOS序列号
最近由于项目的需要,需要在程序中获取机器的硬盘序列号和MAC地址等信息,在C#下,可以很容易的获得这些信息,但是在C++程序中感觉比较麻烦.经过百度,发现很多大虾都是通过WMI来获取这些硬件信息的,网 ...
- jsonp原理详解
1.一个众所周知的问题,Ajax直接请求普通文件存在跨域无权限访问的问题,甭管你是静态页面.动态网页.web服务.WCF,只要是跨域请求,一律不准. 2.不过我们又发现,Web页面上调用js文件时则不 ...
- Django(二)路由系统、视图、模板
大纲 一.内容概要: 二.上节回顾 三.Django 视图–views 1.获取用户多个数据及文件上传 2.FBV 和 CBV 3.装饰器 四.Django模板补充 - Django模板语言循 ...
- PHP——??空合并运算符和?:三元运算符
前言 在上一篇随笔,用三元运算符简单写的一个东西,引发了对他的兴趣,所以打算研究下. PHP7的新特性: https://php.net/manual/zh/migration70.new-featu ...
- centos7升级内核至最新
应用背景: 最近在接触docker,其对内核版本要求较高,就连目前使用的centos7.x默认内核版本为3.10.0-xxx,也是刚好满足其最低要求,故借此机会记录一下升级内核的操作步骤. 测试环境: ...
- [PKUWC2019]Day1 T2 你和虚树的故事
选择k个颜色,使得颜色的虚树有交的方案数 肯定要考虑连通块的贡献. 法一 https://www.cnblogs.com/xzz_233/p/10292983.html 枚举连通块还是不可行的. 枚举 ...
- 超越村后端开发(2:新建models.py+xadmin的引入)
1.新建Model 1.users数据 1.在apps/users/models.py中: from datetime import datetime from django.db import mo ...
- Golang语言的入门开始
一.golang介绍与安装 二.golang-hello world 三.golang的变量 四.golang的类型 五.golang的常量 六.golang的函数(func) 七.golang的包 ...