Completed
Push — development ( 835d72...49dbbd )
by Dylan David
01:41
created

BinaryTree::insert()   C

Complexity

Conditions 8
Paths 8

Size

Total Lines 36
Code Lines 18

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
dl 0
loc 36
rs 5.3846
c 0
b 0
f 0
cc 8
eloc 18
nc 8
nop 1
1
<?php
2
3
namespace Ptypes;
4
5
use Ptypes\TreeNode;
6
use Ptypes\Exceptions\UnexpectedType;
7
use Ptypes\Exceptions\InvalidArgument;
8
9
class BinaryTree
10
{
11
	public const IN_ORDER = 0;
12
	public const PRE_ORDER = 1;
13
	public const POST_ORDER = 2;
14
	
15
	public $root;
16
	
17
	public function __construct($root=null)
18
	{
19
		$this->root = $root;
20
	}
21
	
22
	public function insert($node) //$node is a treenode
23
	{
24
		$this->validate_parameter($node);
25
		
26
		if($this->root == null)
27
		{
28
			$this->root = $node;
29
			return $this;
30
		}
31
		
32
		$n = $this->root;
33
		while(1)
34
		{	
35
			if($node->value < $n->value)
36
			{
37
				if($n->left == null)
38
				{
39
					$n->left = $node;
40
					return $this;
41
				}
42
				
43
				$n = $n->left;
44
			}
45
			else if($node->value > $n->value)
46
			{
47
				if($n->right == null)
48
				{
49
					$n->right = $node;
50
					return $this;
51
				}
52
				
53
				$n = $n->right;
54
			}
55
			else if($node->value == $n->value) //node is already in the tree, we do not create duplicates!
56
			{
57
				return $this;
58
			}
59
		}
60
	}
61
	
62
	public function search($value)
63
	{
64
		if(gettype($value) != "integer" && gettype($value) != "int" && gettype($value) != "double")
65
		{
66
			throw new InvalidArgument("Expected a number, got: " . gettype($value) . " !\n");
67
		}
68
		
69
		return $this->recursive_search($value, $this->root);
70
	}
71
	
72
	private function recursive_search($value, $node)
73
	{
74
		if($node == null)
75
		{
76
			return null;
77
		}
78
		
79
		$eval = $value - $node->value;
80
		if($eval == 0)
81
		{
82
			return $node;
83
		}
84
		
85
		if($eval < 0)
86
		{
87
			return $this->recursive_search($value, $node->left);
88
		}
89
		else if($eval > 0)
90
		{
91
			return $this->recursive_search($value, $node->right);
92
		}
93
	}
94
	
95
	public function delete($value)
0 ignored issues
show
Unused Code introduced by
The parameter $value is not used and could be removed. ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-unused  annotation

95
	public function delete(/** @scrutinizer ignore-unused */ $value)

This check looks for parameters that have been defined for a function or method, but which are not used in the method body.

Loading history...
96
	{
97
		//TODO
98
	}
99
	
100
	public function is_complete()
101
	{
102
		//TODO
103
	}
104
	
105
	public function traverse($order)
106
	{
107
		switch($order)
108
		{
109
			case self::IN_ORDER:
110
				$results = array();
111
				$this->in_order_traversal($this->root, $results);
112
				return $results;
113
				
114
			case self::PRE_ORDER:
115
				$results = array();
116
				$this->pre_order_traversal($this->root, $results);
117
				return $results;
118
				
119
			case self::POST_ORDER:
120
				$results = array();
121
				$this->post_order_traversal($this->root, $results);
122
				return $results;
123
				break;
0 ignored issues
show
Unused Code introduced by
break is not strictly necessary here and could be removed.

The break statement is not necessary if it is preceded for example by a return statement:

switch ($x) {
    case 1:
        return 'foo';
        break; // This break is not necessary and can be left off.
}

If you would like to keep this construct to be consistent with other case statements, you can safely mark this issue as a false-positive.

Loading history...
124
				
125
			default:
126
				throw new InvalidArgument("Invalid order: " . $order . "! Valid orders: IN_ORDER (BinaryTree::IN_ORDER), PRE_ORDER (BinaryTree::PRE_ORDER), and POST_ORDER (BinaryTree::POST_ORDER).\n");
127
		}
128
	}
129
	
130
	private function in_order_traversal($node, &$results) //it isimportant that reference to results stays a reference (php pointer) !
131
	{
132
		if ($node == null)
133
		{
134
			return null;
135
		}
136
		
137
		$this->in_order_traversal($node->left, $results);
138
		
139
		array_push($results, $node->value);
140
		
141
		$this->in_order_traversal($node->right, $results);
142
	}
143
	
144
	private function pre_order_traversal($node, &$results)
145
	{
146
		if ($node == null)
147
		{
148
			return null;
149
		}
150
		
151
		array_push($results, $node->value);
152
		
153
		$this->pre_order_traversal($node->left, $results);
154
		$this->pre_order_traversal($node->right, $results);
155
	}
156
	
157
	private function post_order_traversal($node, &$results)
158
	{
159
		if ($node == null)
160
		{
161
			return null;
162
		}
163
		
164
		$this->post_order_traversal($node->left, $results);
165
		$this->post_order_traversal($node->right, $results);
166
		
167
		array_push($results, $node->value);
168
	}
169
	
170
	private function validate_parameter($node)
171
	{
172
		if(gettype($node) != "object")
173
		{
174
			throw new UnexpectedType("Expected a Ptypes\TreeNode, got: " . gettype($node));
175
		}
176
		
177
		if(get_class($node) != "Ptypes\TreeNode")
178
		{
179
			throw new UnexpectedType("Expected a Ptypes\TreeNode, got: " . get_class($node));
180
		}
181
	}
182
}