vRp.CD2g_test/utils/sorts/topological.go

60 lines
1.9 KiB
Go

package sorts
type topologicalNode[V any] struct {
value V
dependsOn []*topologicalNode[V]
dependents []*topologicalNode[V]
}
// Topological 拓扑排序是一种对有向图进行排序的算法,它可以用来解决一些依赖关系的问题,比如计算字段的依赖关系。拓扑排序会将存在依赖关系的元素进行排序,使得依赖关系的元素总是排在被依赖的元素之前。
// - slice: 需要排序的切片
// - queryIndexHandler: 用于查询切片中每个元素的索引
// - queryDependsHandler: 用于查询切片中每个元素的依赖关系,返回的是一个索引切片,如果没有依赖关系,那么返回空切片
//
// 该函数在存在循环依赖的情况下将会返回 ErrCircularDependencyDetected 错误
func Topological[Index comparable, V any](slice []V, queryIndexHandler func(item V) Index, queryDependsHandler func(item V) []Index) ([]V, error) {
var nodes = make(map[Index]*topologicalNode[V])
for _, item := range slice {
node := &topologicalNode[V]{value: item}
nodes[queryIndexHandler(item)] = node
}
for _, item := range slice {
depends := queryDependsHandler(item)
for _, depend := range depends {
if node, exists := nodes[depend]; exists {
node.dependsOn = append(node.dependsOn, nodes[queryIndexHandler(item)])
node.dependents = append(node.dependents, nodes[queryIndexHandler(item)])
}
}
}
var sorted = make([]V, 0, len(slice))
var visited = make(map[Index]bool)
var visit func(node *topologicalNode[V])
visit = func(node *topologicalNode[V]) {
index := queryIndexHandler(node.value)
if node == nil || visited[index] {
return
}
visited[index] = true
for _, n := range node.dependsOn {
visit(n)
}
sorted = append(sorted, node.value)
}
for _, node := range nodes {
visit(node)
}
if len(sorted) != len(slice) {
return nil, ErrCircularDependencyDetected
}
return sorted, nil
}