โครงสร้างข้อมูลและอัลกอริทึม - ต้นไม้
ต้นไม้แสดงถึงโหนดที่เชื่อมต่อกันด้วยขอบ เราจะพูดถึง 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;
}
}
หากต้องการทราบข้อมูลเกี่ยวกับการดำเนินงานของโครงสร้างข้อมูลแบบต้นไม้ค้นหาแบบทวิภาคโปรดคลิกที่นี่