aboutsummaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
-rw-r--r--bruijn.cabal3
-rw-r--r--std/Tree.bruijn46
-rw-r--r--std/Tree/Balanced.bruijn94
-rw-r--r--std/Tree/Rose.bruijn58
4 files changed, 154 insertions, 47 deletions
diff --git a/bruijn.cabal b/bruijn.cabal
index b560caa..ca86a9a 100644
--- a/bruijn.cabal
+++ b/bruijn.cabal
@@ -31,10 +31,11 @@ data-files:
std/Pair.bruijn
std/Result.bruijn
std/String.bruijn
- std/Tree.bruijn
std/Number/Binary.bruijn
std/Number/Ternary.bruijn
std/Number/Unary.bruijn
+ std/Tree/Balanced.bruijn
+ std/Tree/Rose.bruijn
source-repository head
type: git
diff --git a/std/Tree.bruijn b/std/Tree.bruijn
deleted file mode 100644
index 19bdd62..0000000
--- a/std/Tree.bruijn
+++ /dev/null
@@ -1,46 +0,0 @@
-# MIT License, Copyright (c) 2023 Marvin Borner
-# Rose trees based on std/List
-
-:import std/Combinator .
-:import std/List L
-:import std/Logic .
-:import std/Math .
-:import std/Pair .
-
-# a tree node has a label as its head and subtrees as its tail
-
-# constructs a tree with a label and no branches
-leaf [0 : L.empty] ⧗ a → (Tree a)
-
-{:}‣ leaf
-
-# constructs a node with a subnodes
-node [[1 : 0]] ⧗ a → (List (Tree a)) → (Tree a)
-
-{…:…} node
-
-# returns the root label of a tree
-label ^‣ ⧗ (Tree a) → a
-
-^‣ label
-
-# returns the branches of a tree
-branches ~‣ ⧗ (Tree a) → (List (Tree a))
-
-~‣ branches
-
-# returns true if a tree is empty
-empty? [L.empty? ~0] ⧗ (Tree a) → Boolean
-
-∅?‣ empty?
-
-:test (∅?({ 'a' : {:}'b' })) (false)
-:test (∅?({:}'a')) (true)
-
-# applies a function to leaf and the leafs of all branches
-map z [[[rec]]] ⧗ (a → b) → (Tree a) → (Tree b)
- rec { (1 ^0) : (L.map (2 1) ~0) }
-
-…<$>… map
-
-:test (map ^‣ ({ "woo" : ({:}"oof" : (({ "aah" : (({:}"huh" : L.empty)) }) : L.empty)) })) ({ 'w' : ({:}'o' : (({ 'a' : ({:}'h' : L.empty) }) : L.empty)) })
diff --git a/std/Tree/Balanced.bruijn b/std/Tree/Balanced.bruijn
new file mode 100644
index 0000000..fe4a155
--- /dev/null
+++ b/std/Tree/Balanced.bruijn
@@ -0,0 +1,94 @@
+# MIT License, Copyright (c) 2023 Marvin Borner
+# Balanced AVL tree for numerical leafs, inspired by Rudy Matela's implementation
+# TODO: Extend for arbitrary orderable types?
+# TODO: More tests
+
+:import std/Combinator .
+:import std/List L
+:import std/Number .
+:import std/Option .
+:import std/Pair .
+
+# tree ⧗ (height : (lbranch? : (label : rbranch?)))
+
+# error return that can't happen (in theory)
+error Ω
+
+# unwrap tree from option (only use if not empty!)
+unwrap unwrap-or error ⧗ (Option BalancedTree) → BalancedTree
+
+!‣ unwrap
+
+# empty tree
+empty none ⧗ (Option BalancedTree)
+
+# returns height of tree
+height map-or (-1) ^‣ ⧗ (Option BalancedTree) → 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) → Number → (Option BalancedTree) → BalancedTree
+
+# constructs a leaf node
+leaf [node none 0 none] ⧗ Number → BalancedTree
+
+:test (leaf (+42)) (++(-1) : (none : ((+42) : none)))
+
+# returns the label of a tree
+label [^(~(~0))] ⧗ BalancedTree → Number
+
+?‣ label
+
+:test (?(leaf (+42))) ((+42))
+
+# returns the left branch of a tree
+left [^(~0)] ⧗ BalancedTree → (Option BalancedTree)
+
+//‣ left
+
+# returns the right branch of a tree
+right [~(~(~0))] ⧗ BalancedTree → (Option BalancedTree)
+
+\\‣ right
+
+# returns the balancing factor of a tree
+factor map-or (+0) d ⧗ (Option BalancedTree) → Number
+ d [(height //0) - (height \\0)]
+
+:test (factor (some (leaf (+42)))) (++(-1))
+
+rotate-ll [node //(!(//0)) ?(//0) (some (node \\(!(//0)))) ?0 \\0] ⧗ BalancedTree → BalancedTree
+
+rotate-rr [node (some (node //0 ?0 //(!(\\0)))) ?(\\0) \\(!(\\0))] ⧗ BalancedTree → BalancedTree
+
+rotate-lr [rotate-ll (node (some (rotate-rr !(//0))) ?0 \\0)] ⧗ BalancedTree → BalancedTree
+
+rotate-rl [rotate-rr (node //0 ?0 (some (rotate-ll !(\\0))))] ⧗ BalancedTree → BalancedTree
+
+# balance a tree
+balance [go (factor 0)] ⧗ (Option BalancedTree) → (Option BalancedTree)
+ go [(0 >? (+1)) left ((0 <? (-1)) right else)]
+ left some (((factor //(!1)) =? (-1)) (rotate-lr !1) (rotate-ll !1))
+ right some (((factor \\(!1)) =? (+1)) (rotate-rl !1) (rotate-rr !1))
+ else 1
+
+# inserts a number into a tree
+insert z [[[rec]]] ⧗ Number → (Option BalancedTree) → (Option BalancedTree)
+ rec none? 0 (some (leaf 1)) (balance (u 1 ?(!0)))
+ u compare-case eq lt gt
+ eq 0
+ lt some (node (2 1 //(!0)) ?(!0) \\(!0))
+ gt some (node //(!0) ?(!0) (2 1 \\(!0)))
+
+# number of elements in tree (slow)
+size z [[rec]] ⧗ (Option BalancedTree) → Number
+ rec none? 0 case-empty case-full
+ case-full (1 //(!0)) + (1 \\(!0)) + (+1)
+ case-empty (+0)
+
+list! z [[map-or L.empty go 0]] ⧗ (Option BalancedTree) → (List Number)
+ go [L.append (L.append (2 //0) L.{}(?0)) (2 \\0)]
+
+from-list L.foldr insert empty ⧗ (List Number) → (Option BalancedTree)
diff --git a/std/Tree/Rose.bruijn b/std/Tree/Rose.bruijn
new file mode 100644
index 0000000..90abeec
--- /dev/null
+++ b/std/Tree/Rose.bruijn
@@ -0,0 +1,58 @@
+# MIT License, Copyright (c) 2023 Marvin Borner
+# Rose trees based on std/List
+
+:import std/Combinator .
+:import std/List L
+:import std/Logic .
+:import std/Math .
+:import std/Pair .
+
+# a tree node has a label as its head and subtrees as its tail
+
+# constructs a tree with a label and no branches
+leaf [0 : L.empty] ⧗ a → (RoseTree a)
+
+{:}‣ leaf
+
+# constructs a node with subnodes
+node [[1 : 0]] ⧗ a → (List (RoseTree a)) → (RoseTree a)
+
+{…:…} node
+
+# returns the root label of a tree
+label ^‣ ⧗ (RoseTree a) → a
+
+^‣ label
+
+# returns the branches of a tree
+branches ~‣ ⧗ (RoseTree a) → (List (RoseTree a))
+
+~‣ branches
+
+# returns true if a tree is empty
+empty? [L.empty? ~0] ⧗ (RoseTree a) → Boolean
+
+∅?‣ empty?
+
+:test (∅?({ 'a' : (({:}'b') : L.empty) })) (false)
+:test (∅?({:}'a')) (true)
+
+# applies a function to leaf and the leafs of all branches
+map z [[[rec]]] ⧗ (a → b) → (RoseTree a) → (RoseTree b)
+ rec { (1 ^0) : (L.map (2 1) ~0) }
+
+…<$>… map
+
+:test (map ^‣ ({ "woo" : ({:}"oof" : (({ "aah" : (({:}"huh" : L.empty)) }) : L.empty)) })) ({ 'w' : ({:}'o' : (({ 'a' : ({:}'h' : L.empty) }) : L.empty)) })
+
+# maps a function returning list of trees and concatenates
+concat-map L.concat ∘∘ map ⧗ ((RoseTree a) → (List (RoseTree b))) → (List (RoseTree a)) → (List (RoseTree b))
+
+# folds a tree
+# TODO: fix
+fold [[[z [[rec]] 0]]] ⧗ (a → (List b) → b) → (RoseTree a) → b
+ rec ∅?0 case-end case-fold
+ case-fold 4 ^0 (1 ~0)
+ case-end 3
+
+# :test (fold [[∅?0 (+1) (sum 0)]] ({ 'w' : ({:}'o' : (({ 'a' : ({:}'h' : L.empty) }) : L.empty)) })) ((+4))