I am trying to adapt Brian's Fold for Bianary Trees (http://lorgonblog.wordpress.com/2008/04/06/catamorphisms-part-two/) to apply for Multiway trees.
Summarizing from Brian's Blog:
Data structure:
type Tree<'a> =
| Node of (*data*)'a * (*left*)Tree<'a> * (*right*)Tree<'a>
| Leaf
let tree7 = Node(4, Node(2, Node(1, Leaf, Leaf), Node(3, Leaf, Leaf)),
Node(6, Node(5, Leaf, Leaf), Node(7, Leaf, Leaf)))
Binary Tree Fold function
let FoldTree nodeF leafV tree =
let rec Loop t cont =
match t with
| Node(x,left,right) -> Loop left (fun lacc ->
Loop right (fun racc ->
cont (nodeF x lacc racc)))
| Leaf -> cont leafV
Loop tree (fun x -> x)
examples
let SumNodes = FoldTree (fun x l r -> x + l + r) 0 tree7
let Tree6to0 = FoldTree (fun x l r -> Node((if x=6 then 0 else x), l, r)) Leaf tree7
Multiway Tree version [not (fully) working]:
Data Structure
type MultiTree = | MNode of int * list<MultiTree>
let Mtree7 = MNode(4, [MNode(2, [MNode(1,[]); MNode(3, [])]);
MNode(6, [MNode(5, []); MNode(7, [])])])
Fold function
let MFoldTree nodeF leafV tree =
let rec Loop tree cont =
match tree with
| MNode(x,sub)::tail -> Loop (sub@tail) (fun acc -> cont(nodeF x acc))
| [] -> cont leafV
Loop [tree] (fun x -> x)
Example 1 Returns 28 - seems to work
let MSumNodes = MFoldTree (fun x acc -> x + acc) 0 Mtree7
Example 2
Doesn't Run
let MTree6to0 = MFoldTree (fun x acc -> MNode((if x=6 then 0 else x), [acc])) Mtree7
Initially I thought the MFoldTree
needed a map.something
somewhere but I got it to work with the @
operator instead.
Any help on the second example and or correcting what I've done in the MFoldTree
function would be great!
Cheers
dusiod
Another solution could be
really, we can treat tree as a lineal struct (to fold it).
Use case
Filter is the same with normal fold (because
mfold
is a normal fold):That function could be generic (as
List.fold
,Array.fold
, ... could be generics)."but the intention of the second is to return the whole tree modified so that any nodes which had the value 6 for example now have value 0"
But that is not a
fold
computation, is amap
!You can do easilly (treating, again, as a lineal struct)
Use case
Again, I suggest to do it for each possible list container (
Seq
,List
,Array
, ...), it enable to user select best strategy in context.Notes:
The trick is that you need to pass one additional function to fold.
In Brian's version, the fold function just takes
nodeF
that is called with the value in the node and the two values produced from the left and right sub-trees.That is not sufficient for multiway trees. Here, we need a function
nodeF
that is called with the value in the node and the result produced by aggregating all values of the sub-trees. But you also need a function - saycombineF
that combines value produced from multiple sub-trees of a node.Your fold function is a good start - you just need to add one more recursive call to process
tail
:Summing is easy, because we just use the
+
operator for both functions:Filtering the tree is more tricky. The
nodeF
function will get the element in the node and a list of child nodes (that is the result of aggregation) and produces a single node. ThecombineF
function will get the result from a first node (that is aMultiTree
value) and a list of children produced from the remaining nodes. The initial value produced from an empty tree is an empty list: