# MIT License, Copyright (c) 2023 Marvin Borner # Balanced AVL tree, inspired by Rudy Matela's implementation # TODO: Analyze performance bottlenecks :import std/Combinator . :import std/List L :import std/Logic . :import std/Option . :import std/Pair . :import std/Number . # error return that can't happen (in theory) error Ω # unwraps tree from option (only use if not empty!) unwrap unwrap-or error ⧗ (Option (BalancedTree a)) → (BalancedTree a) !‣ unwrap # empty tree empty none ⧗ (Option (BalancedTree a)) # returns height of tree height map-or (-1) ^‣ ⧗ (Option (BalancedTree a)) → Number :test (height empty) ((-1)) :test (height (some ((+5) : ((+42) : (none : none))))) ((+5)) # constructs a tree with a label and no branches node [[[(max (height 0) ++(height 2)) : (2 : (1 : 0))]]] ⧗ (Option (BalancedTree a)) → a → (Option (BalancedTree a)) → (BalancedTree a) # constructs a leaf node leaf [node none 0 none] ⧗ a → (BalancedTree a) :test (leaf (+42)) (++(-1) : (none : ((+42) : none))) # returns the label of a tree label [^(~(~0))] ⧗ (BalancedTree a) → a ?‣ label :test (?(leaf (+42))) ((+42)) # returns the left branch of a tree left [^(~0)] ⧗ (BalancedTree a) → (Option (BalancedTree a)) //‣ left :test (//(node none (+0) none)) (none) :test (//(node (some (leaf (+3))) (+0) none)) (some (leaf (+3))) # returns the right branch of a tree right [~(~(~0))] ⧗ (BalancedTree a) → (Option (BalancedTree a)) \\‣ right :test (\\(node none (+0) none)) (none) :test (\\(node none (+0) (some (leaf (+3))))) (some (leaf (+3))) # returns the balancing factor of a tree factor map-or (+0) d ⧗ (Option (BalancedTree a)) → Number d [(height //0) - (height \\0)] :test (factor (some (leaf (+42)))) (++(-1)) rotate-ll [node //(!(//0)) ?(!(//0)) (some (node \\(!(//0)) ?0 \\0))] ⧗ (BalancedTree a) → (BalancedTree a) rotate-rr [node (some (node //0 ?0 //(!(\\0)))) ?(!(\\0)) \\(!(\\0))] ⧗ (BalancedTree a) → (BalancedTree a) rotate-lr [rotate-ll (node (some (rotate-rr !(//0))) ?0 \\0)] ⧗ (BalancedTree a) → (BalancedTree a) rotate-rl [rotate-rr (node //0 ?0 (some (rotate-ll !(\\0))))] ⧗ (BalancedTree a) → (BalancedTree a) # balances a tree balance [go (factor 0)] ⧗ (Option (BalancedTree a)) → (Option (BalancedTree a)) go [=?0 else (0 >? (+1) left (0 has? (+42) empty) (false) # returns the value in a tree # could have more information with a clever comparison function find [z [[[rec]]]] ⧗ (Compare a) → a → (Option (BalancedTree a)) → (Option a) rec none? 0 0 (3 eq gt lt 1 ?(!0)) eq some ?(!0) gt 2 1 \\(!0) lt 2 1 //(!0) :test (find (+42) empty) (none) :test (find (+42) (insert (+42) empty)) (some (+42)) # number of elements in tree (slow) size z [[rec]] ⧗ (Option (BalancedTree a)) → Number rec none? 0 case-empty case-full case-full ++((1 //(!0)) + (1 \\(!0))) case-empty (+0) # converts a tree to a list tree→list z [[map-or L.empty go 0]] ⧗ (Option (BalancedTree a)) → (List a) go [L.append (L.append (2 //0) L.{}(?0)) (2 \\0)] # converts a list to a tree list→tree [L.foldr (insert 0) empty] ⧗ (Compare a) → (List a) → (Option (BalancedTree a))