【最小树形图+带LAZY的可并堆】poj3164

题目大意:最小树形图;

最小树形图:给你一个带权有向图,选出一些边构成一颗有根树,并使得这棵树的边权之和最小;

 

大家都知道最小生成树的算法是prim和Kruskal,但是如果是有向图呢?萎了吧。。。这时候NB的刘朱算法就出现了,太IMBA了!

 

 

下面为基本流程.

最小树形图模型首先给定点集和一个根,若干条带权有向边,求从根出发的一个子图,边数为N-1,能从根到所有节点,并且边权和最小.

首先可以肯定的是,如果这个图从根走一遍后发现不连通,那么肯定无最小树形图.

否则每个仍在图中的点i记对于i点的最小权入边对应的入点为pre[i]

如果没有环,直接将所有cost[pre[i],i]相加记为ans

否则,缩环为点,记环为v,环上的任意点位vv,不在环上的任意点为u,将环上的边权和加入ans,改cost[v,u]=min{cost[vv,u]},cost[u,v]=min(cost[u,vv]-cost[pre[vv],vv])

再继续修改pre[i],并判断还有没有环.

这样改造后,对于任意一个环,最后加上某条入这条环的边,因为边权的改造,必将消去环上一条边,使得最小树形图中边位N-1且连通所有点.

from ly

 

 

这个算法的复杂度最坏是O(NM)的,后来灵机一动,发现可以用可并堆来实现O(logN)的时间查找最小值边权,至于修改操作,我很有创意地发明了个带LAZY标号的可并堆。

具体实现是用的斜堆,这样这道题的复杂度就通过数据结构优化成了O(N2logN)。但是由于LAZY标记的常数和斜堆本身带有2的常数以及我编丑了。。。

所以时间复杂度还是比较高的。

 

不过总的来说还是在POJ的前几名,不知道前面的大牛是靠漂亮的常数还是乱七八糟优化法才刷到那么快的速度。

 

代码:

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值