JAVA/CS 스터디

[CS 스터디 : JAVA] (4) 이진 탐색 트리

Kamea 2022. 8. 11. 12:20

 

이진 탐색 트리(Binary Search Tree, BST)

: 이진트리 + 조건 한개 추가(모든 노드에 대해 왼쪽 노드의 데이터 < 오른쪽 노드의 데이터)

: 모든 노드에 대해 왼쪽 노드의 데이터가 오른쪽 노드의 데이터보다 작음을 만족하는 이진 트리 

 

이진트리를 잘 설명한 이미지

 

: 검색 목적의 자료구조 > 중복된 키를 허용하지 않음  

: 중위 순회 방식 ( 왼 - 루트 - 오)

 

중위 순회 : 작 -> 큰 순서대로 수를 순회한다.

 

 

→ 선언 방식

 

class BinarySearchTree {
    class Node {	// 노드 클래스 : 데이터, 왼쪽 노드, 오른쪽 노드
        int data;	
        Node left, right;
 
        public Node(int item){ // 생성자 초기화
            data = item;
            left = right = null;
        }
    }
    
    Node root; // 루트 노드
 
    BinarySearchTree() { root = null; } // 루트 노드가 비어있는 생성자
    BinarySearchTree(int value) { root = new Node(value); } // 루트 노드를 지정한 생성자
 
    public static void main(String[] args){
        BinarySearchTree tree = new BinarySearchTree(); // 호출
    }
}

 

 

 

(1) 원소 탐색 : 아래의 BST에서 원소 10 찾기 (위치 또는 존재 여부)

 

 

 ❶ 탐색 수와 루트 노드의 데이터 비교 

 ❷ 탐색 수 > 루트 노드 : 오른쪽 트리 탐색

 ❷ 탐색 수 < 루트 노드 : 왼쪽 트리 탐색

 ❸ 계속해서 ❷ 과정을 반복하며 리프노드까지 수를 찾는다. 

 

public Node search(Node root, int element){
    if (root==null || root.data==element) // 종료 조건 : 루트가 비어있거나 키가 루트에 없는 경우
        return root;
 
    if (root.data < element) // 찾는 원소가 루트의 데이터보다 클 경우 
       return search(root.right, element);
 
    return search(root.left, element); // 찾는 원소가 루트의 데이터보다 작을 경우 
}

 

 

 

 

(2) 원소 삽입 : BST에 특정 원소를 삽입함(추가)

 

 

 

❶ 가장 첫번째로 삽입하는 원소라면(BST empty) root 노드로 지칭

❷ 그 다음 원소를 삽입할 때, root 노드보다 작으면 왼쪽 트리, 크면 오른쪽 트리로 포인터를 연결해 추가

❸ 기존 BST 에 새로운 원소를 삽입한다면 search를 통해 자리를 찾아 추가

 

class BinarySearchTree {
    void insert(int element) { root = insertRec(root, element); } // 삽입 메서드
 
    Node insertRec(Node root, int element){
        if (root == null) {		// 트리가 비어있거나 비어있는 루트 노르를 찾았다면 새로운 노드를 루트로 만들고 반환
            root = new Node(element);
            return root;
        }
        else if (element < root.data)	// 그렇지 않으면 데이터를 비교하며 트리 재귀 탐색 
            root.left = insertRec(root.left, element);
        else if (element > root.data)
            root.right = insertRec(root.right, element);
        return root;			
    }

	public static void main(String[] args){
        BinarySearchTree tree = new BinarySearchTree();
 
        /* 생성할 BST 트리 
              50
           /     \
          30      70
         /  \    /  \
       20   40  60   80 */
        tree.insert(50);
        tree.insert(30);
        tree.insert(20);
        tree.insert(40);
        tree.insert(70);
        tree.insert(60);
        tree.insert(80);
    }
}

 

 

 

 

 

(3) 원소 삭제: 특정 원소를 BST에서 삭제 

: 3가지 경우

 ① 자식이 없는 노드를 삭제할 경우(리프 노드)

 ② 자식이 하나 있는 노드를 삭제할 경우

 ③ 자식이 두개 있는 노드를 삭제할 경우

 

 

 ① 자식이 없는 노드를 삭제할 경우(리프 노드) : 찾아서 삭제하면 됨 

 

 

 

 

 ② 자식이 하나 있는 노드를 삭제할 경우 : 찾아서 삭제하고 자식 노드를 본인 자리에 넣어주면 됨 

 

 

 

 

 

 

 ③ 자식이 두개 있는 노드를 삭제할 경우 : 찾아서 삭제하고 해당 노드를 기준으로

(a) 왼쪽 트리의 가장 오른쪽에 있는 노드를 해당 노드 자리로 옮겨주거나 

 

 

 

(b) 오른쪽 트리의 가장 왼쪽에 있는 노드를 해당 자리로 옮겨줌

 

 

 

class BinarySearchTree {
    void deleteKey(int element) { root = deleteRec(root, key); } // 삭제 메서드 
  
    Node deleteRec(Node root, int element)
    {
        if (root == null)	// 종료 조건 : 루트 노드가 비어있을 경우(못 찾음) 루트 반환
            return root;
  
        if (element < root.data)	//	루트 데이터보다 작으면 왼쪽 트리 탐색 
            root.left = deleteRec(root.left, element);
        else if (element > root.data) // 루트 데이터보다 크면 오른쪽 트리 탐색 
            root.right = deleteRec(root.right, element);
        else {			// 같으면, 원소를 찾았으므로 삭제 
			// 1, 2경우 (없거나 자식 하나) : 삭제의 의미는 null로 만들어주기 
            if (root.left == null)
                return root.right;
            else if (root.right == null)
                return root.left;
  
            // 3-(b)의 경우, 오른쪽 트리에서 가장 작은 값(가장 왼쪽 위치)을 찾아서 루트 데이터에 넣어줌
            root.data = minValue(root.right);
  
            // 루트에 넣어준 데이터를 트리에서 찾아서 삭제해줌 
            root.right = deleteRec(root.right, root.data);
        }
        return root;
    }
  
    int minValue(Node root) { // 최솟값을 찾는 메서드
        int minv = root.data;
        while (root.left != null) {	// 왼쪽 트리가 비어있을 때까지 내려가서 찾아줌 
            minv = root.left.data;
            root = root.left;
        }
        return minv;
    }
}

 

 

 

 

🙋 이진탐색트리의 편향성을 개선하기 위한 방법?