-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtree.erl
65 lines (54 loc) · 2.06 KB
/
tree.erl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
-module(tree).
-export([
empty/0,
insert/3,
lookup/2,
has_value/2
]).
%% create empty tree
empty() -> {node, 'nil'}.
%% insert given Key:Value pair into a Tree, or create Root tree if given tree is empty.
insert(Key, Val, {node, 'nil'}) ->
{node, {Key, Val, {node, 'nil'}, {node, 'nil'}}};
insert(NewKey, NewVal, {node, {Key, Val, Smaller, Larger}}) when NewKey < Key ->
{node, {Key, Val, insert(NewKey, NewVal, Smaller), Larger}};
insert(NewKey, NewVal, {node, {Key, Val, Smaller, Larger}}) when NewKey > Key ->
{node, {Key, Val, Smaller, insert(NewKey, NewVal, Larger)}};
insert(Key, Val, {node, {Key, _, Smaller, Larger}}) ->
{node, {Key, Val, Smaller, Larger}}.
%% note here that the function returns a completely new tree.
%% this is typical of functional languages having only single assignment.
%% while this can be seen as inefficient, most of the underlying
%% structures of two versions of a tree sometimes happen to be
%% the same and are thus shared, copied by the VM only when needed.
%% look for key in a Tree
lookup(_, {node, 'nil'}) ->
undefined;
lookup(Key, {node, {Key, Val, _, _}}) ->
{ok, Val};
lookup(Key, {node, {NodeKey, _, Smaller, _}}) when Key < NodeKey ->
lookup(Key, Smaller);
lookup(Key, {node, {_, _, _, Larger}}) ->
lookup(Key, Larger).
%% looks for a given value 'Val' in the tree.
has_value(Val, Tree) ->
try has_value1(Val, Tree) of
false -> false
catch
true -> true
end.
has_value1(_, {node, 'nil'}) ->
false;
has_value1(Val, {node, {_, Val, _, _}}) ->
throw(true); % non-local return
has_value1(Val, {node, {_, _, Left, Right}}) ->
has_value1(Val, Left),
has_value1(Val, Right).
% usage examples
% T1 = tree:insert("Jim Woodland", "jim.woodland@gmail.com", tree:empty()).
% T2 = tree:insert("Mark Anderson", "i.am.a@hotmail.com", T1).
% Addresses = tree:insert("Anita Bath", "abath@someuni.edu",
% tree:insert("Kevin Robert", "myfairy@yahoo.com",
% tree:insert("Wilson Longbrow", "longwil@gmail.com", T2))).
% tree:lookup("Anita Bath", Addresses).
% tree:has_value("longwil@gmail.com", Addresses).