mirror of
				https://github.com/gonum/gonum.git
				synced 2025-10-26 16:50:28 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			60 lines
		
	
	
		
			1.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			60 lines
		
	
	
		
			1.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright ©2015 The Gonum Authors. All rights reserved.
 | |
| // Use of this source code is governed by a BSD-style
 | |
| // license that can be found in the LICENSE file.
 | |
| 
 | |
| package path
 | |
| 
 | |
| import "gonum.org/v1/gonum/graph"
 | |
| 
 | |
| // FloydWarshall returns a shortest-path tree for the graph g or false indicating
 | |
| // that a negative cycle exists in the graph. If the graph does not implement
 | |
| // graph.Weighter, UniformCost is used.
 | |
| //
 | |
| // The time complexity of FloydWarshall is O(|V|^3).
 | |
| func FloydWarshall(g graph.Graph) (paths AllShortest, ok bool) {
 | |
| 	var weight Weighting
 | |
| 	if wg, ok := g.(graph.Weighted); ok {
 | |
| 		weight = wg.Weight
 | |
| 	} else {
 | |
| 		weight = UniformCost(g)
 | |
| 	}
 | |
| 
 | |
| 	nodes := g.Nodes()
 | |
| 	paths = newAllShortest(nodes, true)
 | |
| 	for i, u := range nodes {
 | |
| 		paths.dist.Set(i, i, 0)
 | |
| 		for _, v := range g.From(u) {
 | |
| 			j := paths.indexOf[v.ID()]
 | |
| 			w, ok := weight(u, v)
 | |
| 			if !ok {
 | |
| 				panic("floyd-warshall: unexpected invalid weight")
 | |
| 			}
 | |
| 			paths.set(i, j, w, j)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for k := range nodes {
 | |
| 		for i := range nodes {
 | |
| 			for j := range nodes {
 | |
| 				ij := paths.dist.At(i, j)
 | |
| 				joint := paths.dist.At(i, k) + paths.dist.At(k, j)
 | |
| 				if ij > joint {
 | |
| 					paths.set(i, j, joint, paths.at(i, k)...)
 | |
| 				} else if ij-joint == 0 {
 | |
| 					paths.add(i, j, paths.at(i, k)...)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	ok = true
 | |
| 	for i := range nodes {
 | |
| 		if paths.dist.At(i, i) < 0 {
 | |
| 			ok = false
 | |
| 			break
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return paths, ok
 | |
| }
 | 
