diff options
author | Luke Shumaker <lukeshu@lukeshu.com> | 2022-06-29 23:04:11 -0600 |
---|---|---|
committer | Luke Shumaker <lukeshu@lukeshu.com> | 2022-06-29 23:04:16 -0600 |
commit | c3de8d7786c68a988711275dbc50e24208bfb6e5 (patch) | |
tree | 8540ba6f9201f6d464884ac736daca2c1abfa053 /pkg | |
parent | ceded5a24a1bde2c9d9d731464e29675e8af3210 (diff) |
wip rbtree
Diffstat (limited to 'pkg')
-rw-r--r-- | pkg/rbtree/rbtree.go | 421 | ||||
-rw-r--r-- | pkg/rbtree/rbtree_test.go | 134 | ||||
-rw-r--r-- | pkg/rbtree/testdata/fuzz/FuzzTree/f9e6421dacf921f7bb25d402bffbfdce114baad0b1c8b9a9189b5a97fda27e41 | 2 |
3 files changed, 557 insertions, 0 deletions
diff --git a/pkg/rbtree/rbtree.go b/pkg/rbtree/rbtree.go new file mode 100644 index 0000000..d9249c8 --- /dev/null +++ b/pkg/rbtree/rbtree.go @@ -0,0 +1,421 @@ +package rbtree + +import ( + "fmt" + + "golang.org/x/exp/constraints" +) + +type Color bool + +const ( + Black = Color(false) + Red = Color(true) +) + +type Node[V any] struct { + Parent, Left, Right *Node[V] + + Color Color + + Value V +} + +func (node *Node[V]) getColor() Color { + if node == nil { + return Black + } + return node.Color +} + +type Tree[K constraints.Ordered, V any] struct { + KeyFn func(V) K + root *Node[V] +} + +func (t *Tree[K, V]) Walk(fn func(*Node[V])) { + t.root.walk(fn) +} + +func (node *Node[V]) walk(fn func(*Node[V])) { + if node == nil { + return + } + node.Left.walk(fn) + fn(node) + node.Right.walk(fn) +} + +// Search the tree for a value that satisfied the given callbackk +// function. A return value of 0 means to to return this value; <0 +// means to go left on the tree (the value is too high), >0 means to +// go right on th etree (the value is too low). +// +// +-----+ +// | v=8 | == 0 : this is it +// +-----+ +// / \ +// / \ +// <0 : go left >0 : go right +// / \ +// +---+ +---+ +// | 7 | | 9 | +// +---+ +---+ +// +// Returns nil if no such value is found. +// +// Search is good for advanced lookup, like when a range of values is +// acceptable. For simple exact-value lookup, use Lookup. +func (t *Tree[K, V]) Search(fn func(V) int) *Node[V] { + ret, _ := t.root.search(fn) + return ret +} + +func (node *Node[V]) search(fn func(V) int) (exact, nearest *Node[V]) { + var prev *Node[V] + for { + if node == nil { + return nil, prev + } + direction := fn(node.Value) + prev = node + switch { + case direction < 0: + node = node.Left + case direction == 0: + return node, nil + case direction > 0: + node = node.Right + } + } +} + +func (t *Tree[K, V]) exactKey(key K) func(V) int { + return func(val V) int { + valKey := t.KeyFn(val) + switch { + case key < valKey: + return -1 + case key > valKey: + return 1 + default: // key == valKey: + return 0 + } + } +} + +// Lookup looks up the value for an exact key. If no such value +// exists, nil is returned. +func (t *Tree[K, V]) Lookup(key K) *Node[V] { + return t.Search(t.exactKey(key)) +} + +// Min returns the minimum value stored in the tree, or nil if the +// tree is empty. +func (t *Tree[K, V]) Min() *Node[V] { + return t.root.min() +} + +func (node *Node[V]) min() *Node[V] { + if node == nil { + return nil + } + for { + if node.Left == nil { + return node + } + node = node.Left + } +} + +// Max returns the maximum value stored in the tree, or nil if the +// tree is empty. +func (t *Tree[K, V]) Max() *Node[V] { + return t.root.max() +} + +func (node *Node[V]) max() *Node[V] { + if node == nil { + return nil + } + for { + if node.Right == nil { + return node + } + node = node.Right + } +} + +func (t *Tree[K, V]) Next(cur *Node[V]) *Node[V] { + return cur.next() +} + +func (cur *Node[V]) next() *Node[V] { + if cur.Right != nil { + return cur.Right.min() + } + child, parent := cur, cur.Parent + for parent != nil && child == parent.Right { + child, parent = parent, parent.Parent + } + return parent +} + +func (t *Tree[K, V]) Prev(cur *Node[V]) *Node[V] { + return cur.prev() +} + +func (cur *Node[V]) prev() *Node[V] { + if cur.Left != nil { + return cur.Left.max() + } + child, parent := cur, cur.Parent + for parent != nil && child == parent.Left { + child, parent = parent, parent.Parent + } + return parent +} + +func (t *Tree[K, V]) parentChild(node *Node[V]) **Node[V] { + switch { + case node.Parent == nil: + return &t.root + case node.Parent.Left == node: + return &node.Parent.Left + case node.Parent.Right == node: + return &node.Parent.Right + default: + panic(fmt.Errorf("node %p is not a child of its parent %p", node, node.Parent)) + } +} + +func (t *Tree[K, V]) leftRotate(x *Node[V]) { + // p p + // | | + // +---+ +---+ + // | x | | y | + // +---+ +---+ + // / \ => / \ + // a +---+ +---+ c + // | y | | x | + // +---+ +---+ + // / \ / \ + // b c a b + + // Define 'p', 'x', 'y', and 'b' per the above diagram. + p := x.Parent + pChild := t.parentChild(x) + y := x.Right + b := y.Left + + // Move things around + + y.Parent = p + *pChild = y + + x.Parent = y + y.Left = x + + if b != nil { + b.Parent = x + } + x.Right = b +} + +func (t *Tree[K, V]) rightRotate(y *Node[V]) { + // | | + // +---+ +---+ + // | y | | x | + // +---+ +---+ + // / \ => / \ + // +---+ c a +---+ + // | x | | y | + // +---+ +---+ + // / \ / \ + // a b b c + + // Define 'p', 'x', 'y', and 'b' per the above diagram. + p := y.Parent + pChild := t.parentChild(y) + x := y.Left + b := x.Right + + // Move things around + + x.Parent = p + *pChild = x + + y.Parent = x + x.Right = y + + if b != nil { + b.Parent = y + } + y.Left = b +} + +func (t *Tree[K, V]) Insert(val V) { + // Naive-insert + + key := t.KeyFn(val) + exact, parent := t.root.search(t.exactKey(key)) + if exact != nil { + exact.Value = val + return + } + + node := &Node[V]{ + Color: Red, + Parent: parent, + Value: val, + } + if parent == nil { + t.root = node + } else if key < t.KeyFn(parent.Value) { + parent.Left = node + } else { + parent.Right = node + } + + // Re-balance + + for node.Parent.getColor() == Red { + if node.Parent == node.Parent.Parent.Left { + uncle := node.Parent.Parent.Right + if uncle.getColor() == Red { + node.Parent.Color = Black + uncle.Color = Black + node.Parent.Parent.Color = Red + node = node.Parent.Parent + } else { + if node == node.Parent.Right { + node = node.Parent + t.leftRotate(node) + } + node.Parent.Color = Black + node.Parent.Parent.Color = Red + t.rightRotate(node.Parent.Parent) + } + } else { + uncle := node.Parent.Parent.Left + if uncle.getColor() == Red { + node.Parent.Color = Black + uncle.Color = Black + node.Parent.Parent.Color = Red + node = node.Parent.Parent + } else { + if node == node.Parent.Left { + node = node.Parent + t.rightRotate(node) + } + node.Parent.Color = Black + node.Parent.Parent.Color = Red + t.leftRotate(node.Parent.Parent) + } + } + } + t.root.Color = Black +} + +func (t *Tree[K, V]) Delete(key K) { + nodeToDelete := t.Lookup(key) + if nodeToDelete == nil { + return + } + + needsFixup := nodeToDelete.getColor() == Black + + var nodeToFixup *Node[V] + switch { + case nodeToDelete.Left == nil: + nodeToFixup = nodeToDelete.Right + if nodeToDelete.Right != nil { + nodeToDelete.Right.Parent = nodeToDelete.Parent + } + *t.parentChild(nodeToDelete) = nodeToDelete.Right + case nodeToDelete.Right == nil: + nodeToFixup = nodeToDelete.Left + if nodeToDelete.Left != nil { + nodeToDelete.Left.Parent = nodeToDelete.Parent + } + *t.parentChild(nodeToDelete) = nodeToDelete.Left + default: + // The node being deleted has a child on both sides, + // so we've go to reshuffle the parents a bit to make + // room for those children. + next := nodeToDelete.next() + needsFixup = next.getColor() == Black + if next.Parent != nodeToDelete { + if next.Right != nil { + next.Right.Parent = next.Parent + } + *t.parentChild(next) = next.Right + + next.Right = nodeToDelete.Right + next.Right.Parent = next + } + next.Parent = nodeToDelete.Parent + *t.parentChild(nodeToDelete) = next + next.Left = nodeToDelete.Left + next.Left.Parent = next + next.Color = nodeToDelete.Color + } + + if needsFixup { + node := nodeToFixup + for node != nil && node != t.root && node.getColor() == Black { + if node == node.Parent.Left { + sibling := node.Parent.Right + if sibling.getColor() == Red { + sibling.Color = Black + node.Parent.Color = Red + t.leftRotate(node.Parent) + sibling = node.Parent.Right + } + if sibling.Left.getColor() == Black && sibling.Right.getColor() == Black { + sibling.Color = Red + node = node.Parent + } else { + if sibling.Right.getColor() == Black { + sibling.Left.Color = Black + sibling.Color = Red + t.rightRotate(sibling) + sibling = node.Parent.Right + } + sibling.Color = node.Parent.Color + node.Parent.Color = Black + sibling.Right.Color = Black + t.leftRotate(node.Parent) + node = t.root + } + } else { + sibling := node.Parent.Left + if sibling.getColor() == Red { + sibling.Color = Black + node.Parent.Color = Red + t.rightRotate(node.Parent) + sibling = node.Parent.Left + } + if sibling.Right.getColor() == Black && sibling.Left.getColor() == Black { + sibling.Color = Red + node = node.Parent + } else { + if sibling.Left.getColor() == Black { + sibling.Right.Color = Black + sibling.Color = Red + t.leftRotate(sibling) + sibling = node.Parent.Left + } + sibling.Color = node.Parent.Color + node.Parent.Color = Black + sibling.Left.Color = Black + t.rightRotate(node.Parent) + node = t.root + } + } + } + if node != nil { + node.Color = Black + } + } +} diff --git a/pkg/rbtree/rbtree_test.go b/pkg/rbtree/rbtree_test.go new file mode 100644 index 0000000..6c49127 --- /dev/null +++ b/pkg/rbtree/rbtree_test.go @@ -0,0 +1,134 @@ +package rbtree + +import ( + "fmt" + "strings" + "testing" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + "golang.org/x/exp/constraints" +) + +func printTree[K constraints.Ordered, V any](t *testing.T, tree *Tree[K, V]) { + fmtBareNode := func(node *Node[V]) string { + if node == nil { + return "nil" + } + if node.Color == Red { + return fmt.Sprintf("R(%v)", node.Value) + } else { + return fmt.Sprintf("B(%v)", node.Value) + } + } + addIndent := func(indent string, lines []string) []string { + ret := make([]string, 0, len(lines)) + for _, line := range lines { + ret = append(ret, indent+line) + } + return ret + } + var fmtNode func(node *Node[V]) []string + fmtNode = func(node *Node[V]) []string { + if node == nil { + return []string{"nil"} + } + ret := addIndent(" ", fmtNode(node.Right)) + ret = append(ret, fmtBareNode(node)) + ret = append(ret, addIndent(" ", fmtNode(node.Left))...) + return ret + } + t.Log("\n" + strings.Join(fmtNode(tree.root), "\n")) +} + +func checkTree[K constraints.Ordered, V any](t *testing.T, tree *Tree[K, V]) { + // 1. Every node is either red or black + + // 2. The root is black. + assert.Equal(t, Black, tree.root.getColor()) + + // 3. Every nil is black. + + // 4. If a node is red, then both its children are black. + tree.Walk(func(node *Node[V]) { + if node.getColor() == Red { + assert.Equal(t, Black, node.Left.getColor()) + assert.Equal(t, Black, node.Right.getColor()) + } + }) + + // 5. For each node, all simple paths from the node to + // descendent leaves contain the same number of black + // nodes. + var walkCnt func(node *Node[V], cnt int, leafFn func(int)) + walkCnt = func(node *Node[V], cnt int, leafFn func(int)) { + if node.getColor() == Black { + cnt++ + } + if node == nil { + leafFn(cnt) + return + } + walkCnt(node.Left, cnt, leafFn) + walkCnt(node.Right, cnt, leafFn) + } + tree.Walk(func(node *Node[V]) { + var cnts []int + walkCnt(node, 0, func(cnt int) { + cnts = append(cnts, cnt) + }) + for i := range cnts { + if cnts[0] != cnts[i] { + if !assert.Truef(t, false, "node %v: not all leafs have same black-count: %v", node.Value, cnts) { + printTree(t, tree) + } + } + } + }) +} + +func FuzzTree(f *testing.F) { + Ins := uint8(0b0100_0000) + Del := uint8(0) + + f.Add([]uint8{}) + f.Add([]uint8{Ins | 5, Del | 5}) + f.Add([]uint8{Ins | 5, Del | 6}) + f.Add([]uint8{Del | 6}) + + f.Add([]uint8{ // CLRS Figure 14.4 + Ins | 1, + Ins | 2, + Ins | 5, + Ins | 7, + Ins | 8, + Ins | 11, + Ins | 14, + Ins | 15, + + Ins | 4, + }) + + f.Fuzz(func(t *testing.T, dat []uint8) { + tree := &Tree[uint8, uint8]{ + KeyFn: func(x uint8) uint8 { return x }, + } + checkTree(t, tree) + for _, b := range dat { + ins := (b & 0b0100_0000) != 0 + val := (b & 0b0011_1111) + if ins { + t.Logf("Insert(%v)", val) + tree.Insert(val) + node := tree.Lookup(val) + require.NotNil(t, node) + assert.Equal(t, val, node.Value) + } else { + t.Logf("Delete(%v)", val) + tree.Delete(val) + assert.Nil(t, tree.Lookup(val)) + } + checkTree(t, tree) + } + }) +} diff --git a/pkg/rbtree/testdata/fuzz/FuzzTree/f9e6421dacf921f7bb25d402bffbfdce114baad0b1c8b9a9189b5a97fda27e41 b/pkg/rbtree/testdata/fuzz/FuzzTree/f9e6421dacf921f7bb25d402bffbfdce114baad0b1c8b9a9189b5a97fda27e41 new file mode 100644 index 0000000..238e44f --- /dev/null +++ b/pkg/rbtree/testdata/fuzz/FuzzTree/f9e6421dacf921f7bb25d402bffbfdce114baad0b1c8b9a9189b5a97fda27e41 @@ -0,0 +1,2 @@ +go test fuzz v1 +[]byte("YZAB\x990") |