【BZOJ4637】期望 Kruskal+矩阵树定理
【BZOJ4637】期望
Description
Input
Output
一行一个整数,即满足总道路长度最小的情况下,设计方案的美学值期望。要求保留5位小数
Sample Input
1 2 3 4
Sample Output
题解:傻题细节多啊~
我们先进行Kruskal求距离值的最小生成树,如果有多条边权值相同,则我们将它们放到一起处理。我们再把加入后会被分到同一个连通块中的边放到一起,并把连通块离散化缩成点。因为期望是可加的,所以我们可以枚举其中的每条边,这条边出现的概率就是总的缩点后的图的生成树数目 分之 保证这条边在内时剩余图的生成树数目。拿矩阵树定理算一下就好了,用long double即可过。
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <vector>
#include <cmath>
using namespace std;
const int maxn=10010;
const int maxm=200010;
typedef long double db;
const db eps=1e-6;
struct edge
{
int a,b,c,d;
}p[maxm];
int n,m,tot,tp,now;
int f[maxn],bel[maxn],g[65],vis[maxn],st[40],pa[40],pb[40],qa[40],qb[40],ref[65];
int qs[65][40],qt[65],ps[65][40],pt[65];
db v[40][40],ans;
bool cmp(const edge &a,const edge &b)
{
return a.c<b.c;
}
inline int find(int x)
{
return (f[x]==x)?x:(f[x]=find(f[x]));
}
inline int gind(int x)
{
return (g[x]==x)?x:(g[x]=gind(g[x]));
}
inline int rd()
{
int ret=0,f=1; char gc=getchar();
while(gc<'0'||gc>'9') {if(gc=='-') f=-f; gc=getchar();}
while(gc>='0'&&gc<='9') ret=ret*10+(gc^'0'),gc=getchar();
return ret*f;
}
inline double gauss(int N)
{
db ret=1;
int i,j,k;
for(i=1;i<N;i++)
{
for(j=k=i;j<N;j++) if(fabs(v[j][i])>fabs(v[k][i])) k=j;
if(k!=i) for(ret=-ret,j=1;j<N;j++) swap(v[i][j],v[k][j]);
if(fabs(v[i][i])<1e-6) return 0;
for(j=1;j<N;j++) if(j!=i&&fabs(v[j][i])>1e-6)
{
db tmp=v[j][i]/v[i][i];
for(k=1;k<N;k++) v[j][k]-=v[i][k]*tmp;
}
ret=ret*v[i][i];
}
return ret;
}
inline void calc(int x)
{
int i,j,a,b;
for(i=1;i<=pt[x];i++) ref[ps[x][i]]=i;
for(i=1;i<=qt[x];i++) qa[i]=ref[pa[qs[x][i]]],qb[i]=ref[pb[qs[x][i]]];
memset(v,0,sizeof(v));
for(i=1;i<=qt[x];i++) a=qa[i],b=qb[i],v[a][a]++,v[b][b]++,v[a][b]--,v[b][a]--;
double tmp=gauss(pt[x]);
for(i=1;i<=qt[x];i++)
{
memset(v,0,sizeof(v));
if(qa[i]>qb[i]) swap(qa[i],qb[i]);
for(j=1;j<=qt[x];j++) if(i!=j)
{
a=qa[j],b=qb[j];
if(a==qb[i]) a=qa[i];
if(a>qb[i]) a--;
if(b==qb[i]) b=qa[i];
if(b>qb[i]) b--;
v[a][a]++,v[b][b]++,v[a][b]--,v[b][a]--;
}
ans+=gauss(pt[x]-1)*p[st[qs[x][i]]].d/tmp;
}
qt[x]=pt[x]=0;
}
inline void solve()
{
int i,a,b;
now++,tot=0;
for(i=1;i<=tp;i++)
{
if(vis[find(p[st[i]].a)]!=now) vis[f[p[st[i]].a]]=now,bel[f[p[st[i]].a]]=++tot;
if(vis[find(p[st[i]].b)]!=now) vis[f[p[st[i]].b]]=now,bel[f[p[st[i]].b]]=++tot;
pa[i]=bel[f[p[st[i]].a]],pb[i]=bel[f[p[st[i]].b]];
}
for(i=1;i<=tot;i++) g[i]=i;
for(i=1;i<=tp;i++)
{
a=gind(pa[i]),b=gind(pb[i]);
if(a!=b) g[a]=b;
}
for(i=1;i<=tp;i++) a=gind(pa[i]),qs[a][++qt[a]]=i;
for(i=1;i<=tot;i++) a=gind(i),ps[a][++pt[a]]=i;
for(i=1;i<=tot;i++) if(gind(i)==i) calc(i);
for(i=1;i<=tp;i++)
{
a=find(p[st[i]].a),b=find(p[st[i]].b);
if(a!=b) f[a]=b;
}
tp=0;
}
int main()
{
//freopen("bz4637.in","r",stdin);
n=rd(),m=rd();
int i,a,b,pre=0;
for(i=1;i<=m;i++) p[i].a=rd(),p[i].b=rd(),p[i].c=rd(),p[i].d=rd();
sort(p+1,p+m+1,cmp);
for(i=1;i<=n;i++) f[i]=i;
for(i=1;i<=m;i++)
{
if(p[i].c>pre&&pre) solve();
a=find(p[i].a),b=find(p[i].b);
if(a==b) continue;
st[++tp]=i,pre=p[i].c;
}
solve();
printf("%.5Lf",ans);
return 0;
}//3 3 1 2 1 4 1 3 1 6 2 3 1 8
【BZOJ4637】期望 Kruskal+矩阵树定理的更多相关文章
- bzoj1016 [JSOI2008]最小生成树计数——Kruskal+矩阵树定理
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1016 从 Kruskal 算法的过程来考虑产生多种方案的原因,就是边权相同的边有一样的功能, ...
- 矩阵树定理&BEST定理学习笔记
终于学到这个了,本来准备省选前学来着的? 前置知识:矩阵行列式 矩阵树定理 矩阵树定理说的大概就是这样一件事:对于一张无向图 \(G\),我们记 \(D\) 为其度数矩阵,满足 \(D_{i,i}=\ ...
- [spoj104][Highways] (生成树计数+矩阵树定理+高斯消元)
In some countries building highways takes a lot of time... Maybe that's because there are many possi ...
- BZOJ 4766: 文艺计算姬 [矩阵树定理 快速乘]
传送门 题意: 给定一个一边点数为n,另一边点数为m,共有n*m条边的带标号完全二分图$K_{n,m}$ 求生成树个数 1 <= n,m,p <= 10^18 显然不能暴力上矩阵树定理 看 ...
- bzoj 4596 [Shoi2016]黑暗前的幻想乡 矩阵树定理+容斥
4596: [Shoi2016]黑暗前的幻想乡 Time Limit: 20 Sec Memory Limit: 256 MBSubmit: 559 Solved: 325[Submit][Sta ...
- 【LOJ#6072】苹果树(矩阵树定理,折半搜索,容斥)
[LOJ#6072]苹果树(矩阵树定理,折半搜索,容斥) 题面 LOJ 题解 emmmm,这题似乎猫讲过一次... 显然先\(meet-in-the-middle\)搜索一下对于每个有用的苹果数量,满 ...
- 2019.01.02 bzoj2467: [中山市选2010]生成树(矩阵树定理)
传送门 矩阵树定理模板题. 题意简述:自己看题面吧太简单懒得写了 直接构建出这4n4n4n个点然后按照题面连边之后跑矩阵树即可. 代码: #include<bits/stdc++.h> # ...
- [CF917D]Stranger Trees[矩阵树定理+解线性方程组]
题意 给你 \(n\) 个点的无向完全图,指定一棵树 \(S\),问有多少棵生成树和这棵树的公共边数量为 \(k\in[0,n-1]\) \(n\leq 100\) 分析 考虑矩阵树定理,把对应的树边 ...
- 【bzoj4596】[Shoi2016]黑暗前的幻想乡 容斥原理+矩阵树定理
题目描述 给出 $n$ 个点和 $n-1$ 种颜色,每种颜色有若干条边.求这张图多少棵每种颜色的边都出现过的生成树,答案对 $10^9+7$ 取模. 输入 第一行包含一个正整数 N(N<=17) ...
随机推荐
- App架构师实践指南五之性能优化二
App架构师实践指南五之性能优化二 2018年07月30日 13:08:44 nicolelili1 阅读数:214 从UI和CPU方面来说App流畅体验优化,核心为流畅度/卡顿性能优化. 1.基 ...
- Android定位&地图&导航——基于百度地图,实现自定义图标绘制并点击时弹出泡泡
一.问题描述 上一次我们使用百度地图实现基本的定位功能,接下来我们继续实现搜索和定位,并使用LocationOverlay绘制定位位置,同时展示如何使用自定义图标绘制并点击时弹出泡泡 如图所示: 二. ...
- VMware DHCP Service服务无法启动问题的解决
我的电脑出现VMware DHCP Service和VMware NAT Service两个服务无法启动的问题: 打开VMware主界面,菜单->编辑->虚拟网络编辑器: 勾选上“将主机虚 ...
- MySql之游标的使用
一:游标的使用场合 游标只能用于存储过程和函数中. 游标存储了检索语句的结果集,然后在存储过程和函数中可以通过游标来迭代访问结果集中的记录. 二:创建游标 CREATE PROCEDURE 存储过程名 ...
- 设置log rotation避免tomcat catalina.out文件增长过大
创建logrotate配置文件 $ vi /etc/logrotate.d/tomcat 添加以下内容: /opt/tomcat/logs/catalina.out { copytruncate da ...
- request.GetResponse()超时的解决办法
var request = (HttpWebRequest)WebRequest.Create(url); request.Timeout = Timeout.Infinite; request.Ke ...
- (Java编程思想)Thinking in Java
1. 为什么突然想去研读<Thinking in Java>? 最近终于下定决心撸了一本<Thinking in Java>第四版,虽然在此之前我就久闻这本书的大名,但一直未曾 ...
- iOS多个storyboard间跳转
Stroyboard 可以被看作一个管理View画面的集合.也就是说一个iOS专案裡面并没有限制只能有一个Storyboard.所以在你的APP专案中,你可以把功能相近的View放到同一个APP之中, ...
- [ci]jenkins-slave-ssh docker容器化-用户名密码
jenkins-slave-ssh docker容器化 架构 参考:https://www.youtube.com/watch?v=OxrBCt1JLuQ https://github.com/Dav ...
- SourceInsight: sourceInsight4.0 修改默认字体
快捷键 Alt + Y