Добавил обход в глубину с DLL вместо стека
continuous-integration/drone/push Build is passing
Details
continuous-integration/drone/push Build is passing
Details
parent
a9cdd7c8ad
commit
9b947bf937
|
@ -0,0 +1,101 @@
|
|||
package dll
|
||||
|
||||
type ListInterface interface {
|
||||
Len() int // длина списка
|
||||
Front() *ListItem // первый Item
|
||||
Back() *ListItem // последний Item
|
||||
PushFront(v interface{}) *ListItem // добавить значение в начало
|
||||
PushBack(v interface{}) *ListItem // добавить значение в конец
|
||||
Remove(i *ListItem) // удалить элемент
|
||||
MoveToFront(i *ListItem) // переместить элемент в начало
|
||||
}
|
||||
|
||||
type ListItem struct {
|
||||
Value interface{} // значение
|
||||
Next *ListItem // следующий элемент
|
||||
Prev *ListItem // предыдущий элемент
|
||||
}
|
||||
|
||||
type List struct {
|
||||
Info ListItem
|
||||
len int
|
||||
}
|
||||
|
||||
func NewList() *List {
|
||||
return &List{len: 0}
|
||||
}
|
||||
|
||||
func (l *List) Len() int {
|
||||
return l.len
|
||||
}
|
||||
|
||||
func (l *List) Front() *ListItem {
|
||||
return l.Info.Next
|
||||
}
|
||||
|
||||
func (l *List) Back() *ListItem {
|
||||
return l.Info.Prev
|
||||
}
|
||||
|
||||
func (l *List) PushFront(v interface{}) *ListItem {
|
||||
e := &ListItem{Value: v}
|
||||
if l.len != 0 {
|
||||
e.Prev = l.Info.Next
|
||||
l.Info.Next.Next = e
|
||||
} else {
|
||||
l.Info.Prev = e
|
||||
}
|
||||
l.Info.Next = e
|
||||
l.len++
|
||||
return e
|
||||
}
|
||||
|
||||
func (l *List) PushBack(v interface{}) *ListItem {
|
||||
e := &ListItem{Value: v}
|
||||
if l.len != 0 {
|
||||
e.Next = l.Info.Prev
|
||||
l.Info.Prev.Prev = e
|
||||
} else {
|
||||
l.Info.Next = e
|
||||
}
|
||||
l.Info.Prev = e
|
||||
l.len++
|
||||
return e
|
||||
}
|
||||
|
||||
func (l *List) Remove(i *ListItem) {
|
||||
if i.Prev == nil {
|
||||
i.Prev = &ListItem{}
|
||||
l.Info.Prev = i.Next
|
||||
}
|
||||
if i.Next == nil {
|
||||
i.Next = &ListItem{}
|
||||
l.Info.Next = i.Prev
|
||||
}
|
||||
i.Prev.Next = i.Next
|
||||
i.Next.Prev = i.Prev
|
||||
if l.Len() > 1 {
|
||||
l.Info.Next.Next = nil
|
||||
l.Info.Prev.Prev = nil
|
||||
} else {
|
||||
l.Info = ListItem{}
|
||||
}
|
||||
l.len--
|
||||
}
|
||||
|
||||
func (l *List) MoveToFront(i *ListItem) {
|
||||
if l.Info.Next == i {
|
||||
return
|
||||
}
|
||||
if i.Prev != nil {
|
||||
i.Prev.Next = i.Next
|
||||
i.Next.Prev = i.Prev
|
||||
} else {
|
||||
i.Next.Prev = i.Prev
|
||||
l.Info.Prev = i.Next
|
||||
}
|
||||
i.Prev = l.Front()
|
||||
l.Front().Next = i
|
||||
i.Next = nil
|
||||
l.Info.Next = i
|
||||
}
|
|
@ -0,0 +1,96 @@
|
|||
package dll
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
)
|
||||
|
||||
func TestList(t *testing.T) {
|
||||
t.Run("empty list", func(t *testing.T) {
|
||||
l := NewList()
|
||||
|
||||
require.Equal(t, l.Len(), 0)
|
||||
require.Nil(t, l.Front())
|
||||
require.Nil(t, l.Back())
|
||||
})
|
||||
|
||||
t.Run("complex", func(t *testing.T) {
|
||||
l := NewList()
|
||||
|
||||
l.PushFront(10)
|
||||
require.Equal(t, []int{10}, transpond(l))
|
||||
require.Equal(t, 1, l.Len())
|
||||
|
||||
l.Remove(l.Front())
|
||||
require.Equal(t, []int{}, transpond(l))
|
||||
require.Equal(t, 0, l.Len())
|
||||
|
||||
l.PushFront(10)
|
||||
require.Equal(t, []int{10}, transpond(l))
|
||||
require.Equal(t, 1, l.Len())
|
||||
|
||||
l.PushBack(20)
|
||||
require.Equal(t, []int{20, 10}, transpond(l))
|
||||
require.Equal(t, 2, l.Len())
|
||||
|
||||
l.PushBack(30)
|
||||
require.Equal(t, []int{30, 20, 10}, transpond(l))
|
||||
require.Equal(t, 3, l.Len())
|
||||
|
||||
middle := l.Back().Next // 20
|
||||
l.Remove(middle)
|
||||
require.Equal(t, []int{30, 10}, transpond(l))
|
||||
require.Equal(t, 2, l.Len())
|
||||
|
||||
l.PushFront(20)
|
||||
require.Equal(t, []int{30, 10, 20}, transpond(l))
|
||||
require.Equal(t, 3, l.Len())
|
||||
|
||||
l.Remove(l.Back())
|
||||
require.Equal(t, []int{10, 20}, transpond(l))
|
||||
require.Equal(t, 2, l.Len())
|
||||
|
||||
l.PushBack(30)
|
||||
require.Equal(t, []int{30, 10, 20}, transpond(l))
|
||||
require.Equal(t, 3, l.Len())
|
||||
|
||||
l.Remove(l.Front())
|
||||
require.Equal(t, []int{30, 10}, transpond(l))
|
||||
require.Equal(t, 2, l.Len())
|
||||
|
||||
l.PushFront(20)
|
||||
require.Equal(t, []int{30, 10, 20}, transpond(l))
|
||||
require.Equal(t, 3, l.Len())
|
||||
|
||||
for i, v := range [...]int{40, 50, 60, 70, 80} {
|
||||
if i%2 == 0 {
|
||||
l.PushFront(v)
|
||||
} else {
|
||||
l.PushBack(v)
|
||||
}
|
||||
}
|
||||
require.Equal(t, []int{70, 50, 30, 10, 20, 40, 60, 80}, transpond(l))
|
||||
require.Equal(t, 8, l.Len())
|
||||
|
||||
l.MoveToFront(l.Front())
|
||||
require.Equal(t, []int{70, 50, 30, 10, 20, 40, 60, 80}, transpond(l))
|
||||
require.Equal(t, 8, l.Len())
|
||||
|
||||
l.MoveToFront(l.Back().Next.Next) // 30
|
||||
require.Equal(t, []int{70, 50, 10, 20, 40, 60, 80, 30}, transpond(l))
|
||||
require.Equal(t, 8, l.Len())
|
||||
|
||||
l.MoveToFront(l.Back())
|
||||
require.Equal(t, []int{50, 10, 20, 40, 60, 80, 30, 70}, transpond(l))
|
||||
require.Equal(t, 8, l.Len())
|
||||
})
|
||||
}
|
||||
|
||||
func transpond(l *List) []int {
|
||||
elems := make([]int, 0, l.Len())
|
||||
for i := l.Back(); i != nil; i = i.Next {
|
||||
elems = append(elems, i.Value.(int))
|
||||
}
|
||||
return elems
|
||||
}
|
|
@ -3,6 +3,7 @@ package main
|
|||
import (
|
||||
"encoding/json"
|
||||
"maps"
|
||||
"tests/cut_the_tree/dll"
|
||||
)
|
||||
|
||||
type Node struct {
|
||||
|
@ -27,7 +28,7 @@ func (root Node) WeighingTreeWithRecursion() map[string]uint16 {
|
|||
return res
|
||||
}
|
||||
|
||||
func (root Node) WeighingTreeIdiomaticBFS() map[string]uint16 {
|
||||
func (root Node) WeighingTreeWithStack() map[string]uint16 {
|
||||
weight := getNodeWeight(root.ID, root.Name)
|
||||
counter := map[string]*WeightedNode{root.ID: {
|
||||
weight: weight,
|
||||
|
@ -66,8 +67,45 @@ func (root Node) WeighingTreeIdiomaticBFS() map[string]uint16 {
|
|||
return res
|
||||
}
|
||||
|
||||
func (root Node) WeighingTreeIdiomaticDFS() map[string]uint16 {
|
||||
return nil
|
||||
func (root Node) WeightingTreeWithDLL() map[string]uint16 {
|
||||
weight := getNodeWeight(root.ID, root.Name)
|
||||
counter := map[string]*WeightedNode{root.ID: {
|
||||
weight: weight,
|
||||
}}
|
||||
dllist := dll.NewList()
|
||||
dllist.PushFront(root)
|
||||
|
||||
for dllist.Len() > 0 {
|
||||
current := dllist.Front()
|
||||
dllist.Remove(current)
|
||||
|
||||
nodeWeight := getNodeWeight(current.Value.(Node).ID, current.Value.(Node).Name)
|
||||
counter[current.Value.(Node).ID].weight = nodeWeight
|
||||
// Прибавляем вес всем родительским нодам до корня
|
||||
parentID := counter[current.Value.(Node).ID].parent
|
||||
for {
|
||||
currentNode := counter[parentID]
|
||||
if parentID == "" {
|
||||
break
|
||||
}
|
||||
currentNode.weight = currentNode.weight + nodeWeight
|
||||
parentID = currentNode.parent
|
||||
}
|
||||
|
||||
for _, child := range current.Value.(Node).Children {
|
||||
counter[child.ID] = &WeightedNode{
|
||||
parent: current.Value.(Node).ID,
|
||||
}
|
||||
dllist.PushFront(*child)
|
||||
}
|
||||
}
|
||||
|
||||
var res = make(map[string]uint16)
|
||||
for k, v := range counter {
|
||||
res[k] = v.weight
|
||||
}
|
||||
|
||||
return res
|
||||
}
|
||||
|
||||
func (root Node) DecomposeTree(weights map[string]uint16, limit int) ([]*Node, error) {
|
||||
|
|
|
@ -6,19 +6,20 @@ import (
|
|||
"github.com/stretchr/testify/assert"
|
||||
)
|
||||
|
||||
func BenchmarkNode_WeighingTreeWithRecursion(b *testing.B) {
|
||||
func BenchmarkNode_WeighingSmallOrdinaryTreeWithRecursion(b *testing.B) {
|
||||
for i := 0; i < b.N; i++ {
|
||||
output := map[string]uint16{"node0":1416, "node01":194, "node011":49, "node012":49, "node013":49, "node02":194, "node021":49, "node022":49, "node023":49, "node03":983,"node031":361, "node0311":210, "node03111":53, "node03112":53, "node03113":53, "node0312":51, "node0313":51, "node032":49, "node033":526, "node0331":51, "node0332":375, "node03321":53, "node03322":53, "node03323":218, "node033231":55, "node033232":55, "node033233":55, "node0333":51}
|
||||
|
||||
assert.EqualValues(b, output, testOrdinaryVFS.WeighingTreeWithRecursion())
|
||||
assert.EqualValues(b, testOrdinaryVFS.Weghts, testOrdinaryVFS.Tree.WeighingTreeWithRecursion())
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
func BenchmarkNode_WeighingTreeIdiomaticBFS(b *testing.B) {
|
||||
func BenchmarkNode_WeighingSmallOrdinaryTreeWithStack(b *testing.B) {
|
||||
for i := 0; i < b.N; i++ {
|
||||
output := map[string]uint16{"node0":1416, "node01":194, "node011":49, "node012":49, "node013":49, "node02":194, "node021":49, "node022":49, "node023":49, "node03":983,"node031":361, "node0311":210, "node03111":53, "node03112":53, "node03113":53, "node0312":51, "node0313":51, "node032":49, "node033":526, "node0331":51, "node0332":375, "node03321":53, "node03322":53, "node03323":218, "node033231":55, "node033232":55, "node033233":55, "node0333":51}
|
||||
|
||||
assert.EqualValues(b, output, testOrdinaryVFS.WeighingTreeIdiomaticBFS())
|
||||
assert.EqualValues(b, testOrdinaryVFS.Weghts, testOrdinaryVFS.Tree.WeighingTreeWithStack())
|
||||
}
|
||||
}
|
||||
|
||||
func BenchmarkNode_WeighingSmallOrdinaryTreeWithDLL(b *testing.B) {
|
||||
for i := 0; i < b.N; i++ {
|
||||
assert.EqualValues(b, testOrdinaryVFS.Weghts, testOrdinaryVFS.Tree.WeightingTreeWithDLL())
|
||||
}
|
||||
}
|
||||
|
|
|
@ -12,18 +12,18 @@ func TestNode_WeighingTreeAllAlgo(t *testing.T) {
|
|||
output map[string]uint16
|
||||
}{
|
||||
"Ordinary VFS tree":{
|
||||
input:testOrdinaryVFS,
|
||||
output: map[string]uint16{"node0":1416, "node01":194, "node011":49, "node012":49, "node013":49, "node02":194, "node021":49, "node022":49, "node023":49, "node03":983,"node031":361, "node0311":210, "node03111":53, "node03112":53, "node03113":53, "node0312":51, "node0313":51, "node032":49, "node033":526, "node0331":51, "node0332":375, "node03321":53, "node03322":53, "node03323":218, "node033231":55, "node033232":55, "node033233":55, "node0333":51},
|
||||
input:testOrdinaryVFS.Tree,
|
||||
output: testOrdinaryVFS.Weghts,
|
||||
},
|
||||
"VFS in the form of a pathological tree":{
|
||||
input: testPathologicalTree,
|
||||
output: map[string]uint16{"node0":605, "node01":560, "node012":513, "node0123":464, "node01234":413, "node012345":360, "node0123456":305, "node01234567":248, "node012345678":189, "node0123456789":128, "node0123456789A":65},
|
||||
input: testPathologicalTree.Tree,
|
||||
output: testPathologicalTree.Weghts,
|
||||
},
|
||||
}{
|
||||
t.Run(name,func(t *testing.T) {
|
||||
assert.EqualValues(t, tt.output, tt.input.WeighingTreeWithRecursion())
|
||||
assert.EqualValues(t, tt.output, tt.input.WeighingTreeIdiomaticBFS())
|
||||
//assert.EqualValues(t, tt.output, tt.input.WeighingTreeIdiomaticDFS())
|
||||
assert.EqualValues(t, tt.output, tt.input.WeighingTreeWithStack())
|
||||
assert.EqualValues(t, tt.output, tt.input.WeightingTreeWithDLL())
|
||||
})
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,6 +1,12 @@
|
|||
package main
|
||||
|
||||
var testOrdinaryVFS = Node{
|
||||
type TestTree struct {
|
||||
Tree Node
|
||||
Weghts map[string]uint16
|
||||
}
|
||||
|
||||
var testOrdinaryVFS = TestTree{
|
||||
Tree: Node{
|
||||
ID: "node0",
|
||||
Name: "node0",
|
||||
Children: []*Node{
|
||||
|
@ -148,9 +154,12 @@ var testOrdinaryVFS = Node{
|
|||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
Weghts: map[string]uint16{"node0": 1416, "node01": 194, "node011": 49, "node012": 49, "node013": 49, "node02": 194, "node021": 49, "node022": 49, "node023": 49, "node03": 983, "node031": 361, "node0311": 210, "node03111": 53, "node03112": 53, "node03113": 53, "node0312": 51, "node0313": 51, "node032": 49, "node033": 526, "node0331": 51, "node0332": 375, "node03321": 53, "node03322": 53, "node03323": 218, "node033231": 55, "node033232": 55, "node033233": 55, "node0333": 51},
|
||||
}
|
||||
|
||||
var testPathologicalTree = Node{
|
||||
var testPathologicalTree = TestTree{
|
||||
Tree: Node{
|
||||
ID: "node0",
|
||||
Name: "node0",
|
||||
Children: []*Node{
|
||||
|
@ -213,4 +222,6 @@ var testPathologicalTree = Node{
|
|||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
Weghts: map[string]uint16{"node0": 605, "node01": 560, "node012": 513, "node0123": 464, "node01234": 413, "node012345": 360, "node0123456": 305, "node01234567": 248, "node012345678": 189, "node0123456789": 128, "node0123456789A": 65},
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue