โครงสร้างข้อมูลและอัลกอริทึม - ต้นไม้

ต้นไม้แสดงถึงโหนดที่เชื่อมต่อกันด้วยขอบ เราจะพูดถึง binary tree หรือ binary search tree โดยเฉพาะ

Binary Tree เป็นโครงสร้างข้อมูลพิเศษที่ใช้เพื่อวัตถุประสงค์ในการจัดเก็บข้อมูล ต้นไม้ไบนารีมีเงื่อนไขพิเศษที่แต่ละโหนดสามารถมีลูกได้สูงสุดสองลูก ต้นไม้ไบนารีมีประโยชน์ของทั้งอาร์เรย์ที่เรียงลำดับและรายการที่เชื่อมโยงเนื่องจากการค้นหาทำได้รวดเร็วพอ ๆ กับอาร์เรย์ที่เรียงลำดับและการแทรกหรือการลบจะรวดเร็วพอ ๆ กับในรายการที่เชื่อมโยง

เงื่อนไขสำคัญ

ต่อไปนี้เป็นคำศัพท์ที่สำคัญเกี่ยวกับต้นไม้

  • Path - เส้นทางหมายถึงลำดับของโหนดตามขอบของต้นไม้

  • Root- โหนดที่อยู่ด้านบนสุดของต้นไม้เรียกว่ารูท มีเพียงหนึ่งรูทต่อทรีและหนึ่งเส้นทางจากโหนดรูทไปยังโหนดใด ๆ

  • Parent - โหนดใด ๆ ยกเว้นโหนดรูทมีขอบขึ้นไปหนึ่งโหนดที่เรียกว่าพาเรนต์

  • Child - โหนดที่อยู่ด้านล่างโหนดที่กำหนดซึ่งเชื่อมต่อด้วยขอบลงไปเรียกว่าโหนดลูก

  • Leaf - โหนดที่ไม่มีโหนดลูกเรียกว่าโหนดลีฟ

  • Subtree - ต้นไม้ย่อยแสดงถึงลูกหลานของโหนด

  • Visiting - การเยี่ยมชมหมายถึงการตรวจสอบค่าของโหนดเมื่อการควบคุมอยู่บนโหนด

  • Traversing - การข้ามผ่านหมายถึงการผ่านโหนดตามลำดับที่ระบุ

  • Levels- ระดับของโหนดแสดงถึงการสร้างโหนด ถ้าโหนดรูทอยู่ที่ระดับ 0 โหนดลูกถัดไปจะอยู่ที่ระดับ 1 หลานของโหนดอยู่ที่ระดับ 2 เป็นต้น

  • keys - คีย์แสดงถึงค่าของโหนดตามการดำเนินการค้นหาที่จะดำเนินการสำหรับโหนด

การแสดงต้นไม้ค้นหาแบบไบนารี

โครงสร้างการค้นหาแบบไบนารีแสดงพฤติกรรมพิเศษ ชายด์ด้านซ้ายของโหนดต้องมีค่าน้อยกว่าค่าของพาเรนต์และชายด์ด้านขวาของโหนดต้องมีค่ามากกว่าค่าพาเรนต์

เรากำลังจะติดตั้งโครงสร้างโดยใช้วัตถุโหนดและเชื่อมต่อผ่านการอ้างอิง

โหนดต้นไม้

โค้ดสำหรับเขียนโหนดทรีจะคล้ายกับที่ระบุด้านล่าง มีส่วนข้อมูลและการอ้างอิงโหนดลูกด้านซ้ายและขวา

struct node {
   int data;   
   struct node *leftChild;
   struct node *rightChild;
};

ในแผนภูมิโหนดทั้งหมดใช้โครงสร้างร่วมกัน

การใช้งานพื้นฐาน BST

การดำเนินการพื้นฐานที่สามารถทำได้บนโครงสร้างข้อมูลต้นไม้ค้นหาแบบไบนารีมีดังต่อไปนี้ -

  • Insert - แทรกองค์ประกอบในต้นไม้ / สร้างต้นไม้

  • Search - ค้นหาองค์ประกอบในต้นไม้

  • Preorder Traversal - เดินลัดเลาะไปตามต้นไม้ในลักษณะสั่งจองล่วงหน้า

  • Inorder Traversal - เดินลัดเลาะไปตามต้นไม้ตามลำดับ

  • Postorder Traversal - สำรวจต้นไม้ในลักษณะโพสต์ออร์เดอร์

เราจะเรียนรู้การสร้าง (แทรกลงใน) โครงสร้างต้นไม้และค้นหารายการข้อมูลในต้นไม้ในบทนี้ เราจะเรียนรู้เกี่ยวกับวิธีการข้ามต้นไม้ในบทที่จะมาถึง

แทรกการทำงาน

การแทรกครั้งแรกจะสร้างต้นไม้ หลังจากนั้นเมื่อใดก็ตามที่จะแทรกองค์ประกอบอันดับแรกให้หาตำแหน่งที่เหมาะสม เริ่มค้นหาจากโหนดรูทจากนั้นหากข้อมูลมีค่าน้อยกว่าค่าคีย์ให้ค้นหาตำแหน่งว่างในทรีย่อยด้านซ้ายและแทรกข้อมูล มิฉะนั้นให้ค้นหาตำแหน่งว่างในทรีย่อยด้านขวาและแทรกข้อมูล

อัลกอริทึม

If root is NULL 
   then create root node
return

If root exists then
   compare the data with node.data
   
   while until insertion position is located

      If data is greater than node.data
         goto right subtree
      else
         goto left subtree

   endwhile 
   
   insert data
	
end If

การนำไปใช้

การใช้งานฟังก์ชันแทรกควรมีลักษณะดังนี้ -

void insert(int data) {
   struct node *tempNode = (struct node*) malloc(sizeof(struct node));
   struct node *current;
   struct node *parent;

   tempNode->data = data;
   tempNode->leftChild = NULL;
   tempNode->rightChild = NULL;

   //if tree is empty, create root node
   if(root == NULL) {
      root = tempNode;
   } else {
      current = root;
      parent  = NULL;

      while(1) {                
         parent = current;

         //go to left of the tree
         if(data < parent->data) {
            current = current->leftChild;                
            
            //insert to the left
            if(current == NULL) {
               parent->leftChild = tempNode;
               return;
            }
         }
			
         //go to right of the tree
         else {
            current = current->rightChild;
            
            //insert to the right
            if(current == NULL) {
               parent->rightChild = tempNode;
               return;
            }
         }
      }            
   }
}

การดำเนินการค้นหา

เมื่อใดก็ตามที่ต้องการค้นหาองค์ประกอบให้เริ่มค้นหาจากโหนดรูทจากนั้นหากข้อมูลมีค่าน้อยกว่าค่าคีย์ให้ค้นหาองค์ประกอบในแผนผังย่อยด้านซ้าย มิฉะนั้นให้ค้นหาองค์ประกอบในทรีย่อยด้านขวา ทำตามอัลกอริทึมเดียวกันสำหรับแต่ละโหนด

อัลกอริทึม

If root.data is equal to search.data
   return root
else
   while data not found

      If data is greater than node.data
         goto right subtree
      else
         goto left subtree
         
      If data found
         return node
   endwhile 
   
   return data not found
   
end if

การใช้อัลกอริทึมนี้ควรมีลักษณะดังนี้

struct node* search(int data) {
   struct node *current = root;
   printf("Visiting elements: ");

   while(current->data != data) {
      if(current != NULL)
      printf("%d ",current->data); 
      
      //go to left tree

      if(current->data > data) {
         current = current->leftChild;
      }
      //else go to right tree
      else {                
         current = current->rightChild;
      }

      //not found
      if(current == NULL) {
         return NULL;
      }

      return current;
   }  
}

หากต้องการทราบข้อมูลเกี่ยวกับการดำเนินงานของโครงสร้างข้อมูลแบบต้นไม้ค้นหาแบบทวิภาคโปรดคลิกที่นี่