İkili arama ağacı
İkili arama ağacı, verileri organize etmek için kullanılan bir çeşit ikili ağaçtır. İkili ağaçtan temel farkı, verilerin sıralanmış bir şekilde tutulmasıdır, bu sayede ikili arama algoritmasının kullanılmasına imkân verir.
İkili arama ağaçlarında; her düğümün sağ çocuğu kendisinden büyük, sol çocuğu ise kendisinden küçüktür. Bu ağaçlarda çalışan arama algoritması önce kökten başlar, eğer aranan eleman kökten büyükse sağ çocuğa, kökten küçükse sol çocuğa ilerler. Böylece, eleman bulunana yapraklara kadar yinelemeli bir şekilde ilerleme sağlanır.
İşlemler
İkili arama ağacı üzerinde; arama, ekleme, silme işlemleri yapılabilir.
Arama
Kökten başlanarak arama işlemi yapılır. Eğer aranan eleman o anki düğümden büyükse sağa, küçükse sola ilerlenir. Bu algoritma bir döngü içinde ya da özyinelemeli şekilde uygulanabilir. Python kodu şu şekildedir:
def search_recursively(key, node):
if node is None or node.key == key:
return node
if key < node.key:
return search_recursively(key, node.left)
# key > node.key
return search_recursively(key, node.right)
def search_iteratively(key, node):
current_node = node
while current_node is not None:
if key == current_node.key:
return current_node
elif key < current_node.key:
current_node = current_node.left
else: # key > current_node.key:
current_node = current_node.right
return current_node
Ekleme
Eleman ekleme işlemi her zaman yapraklara ya da tek çocuğu olan düğümlere yapılır. Bunun için öncelikle eklenecek elemanın konumu bulunmalıdır, konum bulma işlemi arama kısmında olduğu gibi büyükse sağa, küçükse sola ilerle şeklinde yapılır. Bu algoritma da bir döngü içinde ya da özyinelemeli şekilde uygulanabilir. C++ kodu şu şekildedir:
Node* insert(Node*& root, int key, int value) {
if (!root)
root = new Node(key, value);
else if (key < root->key)
root->left = insert(root->left, key, value);
else // key >= root->key
root->right = insert(root->right, key, value);
return root;
Silme
Silme işlemi üç başlık altında incelenebilir. Eğer,
- Silinecek düğümün çocuğu yoksa: basitçe düğüm silinir.
- Silinecek düğümün tek çocuğu varsa: düğüm silinir ve tek çocuk silinen düğümün yerine taşınır.
- Silinecek düğümün iki çocuğu varsa: silinecek düğüme D diyelim, D'yi silmeden önce D'den büyük en küçük eleman bulunur (inorder successor) buna da E diyelim. E ile D yer değiştirilir ve ardından D silinir.
Bir elemandan büyük en küçük eleman, o elemanın sağ alt ağacının en solundaki elemandır ve bunu bulmak için ekstra metot yazmak gerekir. Bu yüzden, silme algoritması ekleme ve arama algoritmalarına göre daha uzundur. Python kodu şu şekildedir:
1def find_min(self): # Gets minimum node in a subtree
2 current_node = self
3 while current_node.left_child:
4 current_node = current_node.left_child
5 return current_node
6
7def replace_node_in_parent(self, new_value=None):
8 if self.parent:
9 if self == self.parent.left_child:
10 self.parent.left_child = new_value
11 else:
12 self.parent.right_child = new_value
13 if new_value:
14 new_value.parent = self.parent
15
16def binary_tree_delete(self, key):
17 if key < self.key:
18 self.left_child.binary_tree_delete(key)
19 elif key > self.key:
20 self.right_child.binary_tree_delete(key)
21 else: # delete the key here
22 if self.left_child and self.right_child: # if both children are present
23 successor = self.right_child.find_min()
24 self.key = successor.key
25 successor.binary_tree_delete(successor.key)
26 elif self.left_child: # if the node has only a *left* child
27 self.replace_node_in_parent(self.left_child)
28 elif self.right_child: # if the node has only a *right* child
29 self.replace_node_in_parent(self.right_child)
30 else: # this node has no children
31 self.replace_node_in_parent(None)
Dengeli ikili arama ağaçları
Bir ağaçtaki tüm düğümlerin sağ alt ağaçları ve sol alt ağaçları arasındaki yükseklik farkı en fazla 1 ise, o ağaç dengeli olarak tanımlanır. Ağaçların dengeli olması onların yüksekliğini azaltarak ağaç üzerinde çalışan algoritmaların performansını arttırır. Örneğin boş bir ikili arama ağacına 1, 2, 3, 4, 5, 6, 7 elemanlarını sırayla eklediğimizi düşünelim. Bu durumda elemanlar hep sağ tarafa ekleneceği için ağaç esasen bir bağlı listeye dönüşür. Halbuki, aynı elemanlar, yüksekliği 3 olan bir ağaca da yerleştirilebilirlerdi. Bu durum, dengeli ikili arama ağaçlarının icadına yol açmıştır. AVL ağacı, 2-3-4 ağacı ve kırmızı-siyah ağaçlar dengeli ağaçlara örnektir.