2021-04-19:手写代码:最小生成树算法之Kruskal。
2021-04-19:手写代码:最小生成树算法之Kruskal。
福大大 答案2021-04-19:
并查集。边从小到大,找最小边,无环。
代码用golang编写。代码如下:
package main
import (
"fmt"
"sort"
)
func main() {
graph := &Graph{}
graph.nodes = make(map[int]*Node)
graph.nodes[0] = &Node{}
graph.nodes[1] = &Node{}
graph.nodes[2] = &Node{}
graph.edges = make(map[*Edge]struct{})
graph.edges[&Edge{weight: 22, from: graph.nodes[0], to: graph.nodes[1]}] = struct{}{}
graph.edges[&Edge{weight: 33, from: graph.nodes[1], to: graph.nodes[2]}] = struct{}{}
graph.edges[&Edge{weight: 11, from: graph.nodes[2], to: graph.nodes[0]}] = struct{}{}
ret := kruskalMST(graph)
fmt.Println("结果:")
for a, _ := range ret {
fmt.Println(a.weight)
}
}
type Edge struct {
weight int
from *Node
to *Node
}
// 点结构的描述
type Node struct {
value int
in int
out int
nexts []*Node
edges []*Edge
}
type Graph struct {
nodes map[int]*Node
edges map[*Edge]struct{}
}
func printPriorityQueue(priorityQueue []*Edge) {
for _, edge := range priorityQueue {
fmt.Println(edge.weight)
}
}
func kruskalMST(graph *Graph) map[*Edge]struct{} {
unionFind := &UnionFind{}
unionFind.makeSets(graph.nodes)
// 从小的边到大的边,依次弹出,小根堆!
priorityQueue := make([]*Edge, 0)
for edge, _ := range graph.edges {
priorityQueue = append(priorityQueue, edge)
}
fmt.Println("排序前:")
printPriorityQueue(priorityQueue)
//排序
sort.SliceStable(priorityQueue, func(i int, j int) bool {
return priorityQueue[i].weight > priorityQueue[j].weight
})
fmt.Println("--------")
fmt.Println("排序后:")
printPriorityQueue(priorityQueue)
fmt.Println("--------")
result := make(map[*Edge]struct{})
for len(priorityQueue) > 0 { // M 条边
edge := priorityQueue[len(priorityQueue)-1]
priorityQueue = priorityQueue[0 : len(priorityQueue)-1]
if !unionFind.isSameSet(edge.from, edge.to) { // O(1)
result[edge] = struct{}{}
unionFind.union(edge.from, edge.to)
}
}
return result
}
type UnionFind struct {
// key 某一个节点, value key节点往上的节点
fatherMap map[*Node]*Node
// key 某一个集合的代表节点, value key所在集合的节点个数
sizeMap map[*Node]int
}
func (this *UnionFind) makeSets(nodes map[int]*Node) {
this.fatherMap = make(map[*Node]*Node)
this.sizeMap = make(map[*Node]int)
for _, node := range nodes {
this.fatherMap[node] = node
this.sizeMap[node] = 1
}
}
func (this *UnionFind) findFather(n *Node) *Node {
path := make([]*Node, 0)
for n != this.fatherMap[n] {
path = append(path, n)
n = this.fatherMap[n]
}
for len(path) > 0 {
this.fatherMap[path[len(path)-1]] = n
path = path[0 : len(path)-1]
}
return n
}
func (this *UnionFind) isSameSet(a *Node, b *Node) bool {
return this.findFather(a) == this.findFather(b)
}
func (this *UnionFind) union(a *Node, b *Node) {
if a == nil || b == nil {
return
}
aDai := this.findFather(a)
bDai := this.findFather(b)
if aDai != bDai {
aSetSize := this.sizeMap[aDai]
bSetSize := this.sizeMap[bDai]
if aSetSize <= bSetSize {
this.fatherMap[aDai] = bDai
this.sizeMap[bDai] = aSetSize + bSetSize
delete(this.sizeMap, aDai)
} else {
this.fatherMap[bDai] = aDai
this.sizeMap[aDai] = aSetSize + bSetSize
delete(this.sizeMap, bDai)
}
}
}
执行结果如下:

2021-04-19:手写代码:最小生成树算法之Kruskal。的更多相关文章
- .netER的未来路,关于基础是否重要和应该自己手写代码吗?
http://www.cnblogs.com/onepiece_wang/p/5558341.html#!comments 引用"基础知识的学习,一开始可能是背书,但是在后续若干年的工作过程 ...
- ClownFish:比手写代码还快的通用数据访问层
http://www.cnblogs.com/fish-li/archive/2012/07/17/ClownFish.html 阅读目录 开始 ClownFish是什么? 比手写代码还快的执行速度 ...
- 手写代码自动实现自动布局,即Auto Layout的使用
手写代码自动实现自动布局,即Auto Layout的使用,有需要的朋友可以参考下. 这里要注意几点: 对子视图的约束,若是基于父视图,要通过父视图去添加约束. 对子视图进行自动布局调整,首先对UIVi ...
- 如果选择构建ui界面方式,手写代码,xib和StoryBoard间的博弈
代码手写UI这种方法经常被学院派的极客或者依赖多人合作的大型项目大规模使用. 大型多人合作项目使用代码构建UI,主要是看中纯代码在版本管理时的优势,检查追踪改动以及进行代码合并相对容易一些. 另外,代 ...
- 手写代码UI,xib和StoryBoard间的的优劣比较
在UI制作方面,逐渐分化三种主要流派:使用代码手写UI:使用单个xib文件组织viewController或者view:使用StoryBoard来通过单个或很少的几个文件构建UI.三种方式各有优劣,也 ...
- UI到底应该用xib/storyboard完成,还是用手写代码来完成?
UI到底应该用xib/storyboard完成,还是用手写代码来完成? 文章来源:http://blog.csdn.net/libaineu2004/article/details/45488665 ...
- 2019前端面试系列——JS高频手写代码题
实现 new 方法 /* * 1.创建一个空对象 * 2.链接到原型 * 3.绑定this值 * 4.返回新对象 */ // 第一种实现 function createNew() { let obj ...
- Appium初始化设置:手写代码连接手机、appium-desktop连接手机
一.包名获取的三种方式 1)找开发要2)mac使用命令:adb logcat | grep START win使用命令:adb logcat | findstr START 或者可以尝试使用第3条命令 ...
- gcd手写代码及STL中的使用方法
一.手写代码 inline int gcd(int x,int y){ if(y==0) return x; else return(gcd(y,x%y)); } 二.STL中的使用方法 注:在STL ...
- SpringCloud-Ribbon负载均衡机制、手写轮询算法
Ribbon 内置的负载均衡规则 在 com.netflix.loadbalancer 包下有一个接口 IRule,它可以根据特定的算法从服务列表中选取一个要访问的服务,默认使用的是「轮询机制」 Ro ...
随机推荐
- 一天吃透Git面试八股文
什么是Git? Git是一个版本控制系统,用于跟踪计算机文件的变化.Git是一个跟踪计算机文件变化的版本控制系统,用于帮助协调一个项目中几个人的工作,同时跟踪一段时间的进展.换句话说,我们可以说它是一 ...
- Thingsboard3.2.2本地部署
Thingboard3.2.2本地安装编译详细教程!!! 一:拉取源码. 创建一个空的文件夹 在此处使用git拉取源码. git clone https://github.com/thingsboar ...
- 深入理解 Python 虚拟机:整型(int)的实现原理及源码剖析
深入理解 Python 虚拟机:整型(int)的实现原理及源码剖析 在本篇文章当中主要给大家介绍在 cpython 内部是如何实现整型数据 int 的,主要是分析 int 类型的表示方式,分析 int ...
- 说一下在写Android APP时遇到的具体问题
问题一:总是显示出no such table错误 不要担心这个问题(该担心还是得担心一下的哈),以我出错多次的经验来看,只需要在运行APP之后,回到虚拟机的主界面,然后找到相应的APP虚拟软件,将他删 ...
- Java面试——架构设计与分布式
更多内容,移步 IT-BLOG 一.用 Java 自己实现一个 LRU LRU(Least Recently Used:最近最少使用):简单的说,就是保证基本的 Cache容量,如果超过容量则必须丢掉 ...
- Windows7蓝牙音响连接成功,但是无法播放音乐,没有声音
本人使用的蓝牙是博通94360CD的无线网卡集成的,在Windows7系统,成功安装蓝牙驱动,但是无论连接什么蓝牙音响设备,都可以连接成功,但是在音频管理却没有蓝牙音响的设备,自然就没有声音.后来找到 ...
- new做了哪些事情,手写一个new
1)创建一个空对象,将构造函数中的this指向这个空对象 2)将空对象的__proto__指向构造函数的prototype原型 3)执行构造函数里面的代码,为这个空对象添加属性和方法 4)返回一个新的 ...
- JAVA异步编程之Callbacks与Futures模型
JAVA异步编程之Callbacks与Futures模型 一:Callbacks模型 该模型的异步方法,在异步任务完成之后调用,主线程没有异步线程的结果.经典模型如Swing's EventLis ...
- Spring(Bean详解)
GoF之工厂模式 GoF是指二十三种设计模式 GoF23种设计模式可分为三大类: 创建型(5个):解决对象创建问题. 单例模式 工厂方法模式 抽象工厂模式 建造者模式 原型模式 结构型(7个):一些类 ...
- [Maven]探究settings.xml
1 settings.xml的配置模块 1-1 localRepository The path to the local repository maven will use to store art ...