Bài giảng Cấu trúc dữ liệu và giải thuật - Bài 05: Cây cân bằng AVL

pdf 7 trang phuongnguyen 2960
Bạn đang xem tài liệu "Bài giảng Cấu trúc dữ liệu và giải thuật - Bài 05: Cây cân bằng AVL", để tải tài liệu gốc về máy bạn click vào nút DOWNLOAD ở trên

Tài liệu đính kèm:

  • pdfbai_giang_cau_truc_du_lieu_va_giai_thuat_bai_05_cay_can_bang.pdf

Nội dung text: Bài giảng Cấu trúc dữ liệu và giải thuật - Bài 05: Cây cân bằng AVL

  1. CÂY CÂN B ẰNG AVL MC TIÊU Hồn tất bài thực hành này, sinh viên cĩ th ể: Hiểu được các thao tác quay cây (quay trái, quay ph ải) để hiệu chỉnh cây thành cây cân bằng. Cài đặt hồn chỉnh cây cân b ằng AVL. Thời gian thực hành: 120 phút – 360 phút Lưu ý: Sinh viên phải thực hành bài t ập về Cây nhị phân và Cây nhị phân tìm ki ếm trước khi làm bài này. TĨM TT Cây cân bằng AVL là cây nhị phân tìm ki ếm (NPTK) mà tại mỗi đỉnh của cây, độ cao của cây con trái và cây con phải khác nhau khơng quá 1 . Ví dụ 1: cây cân bằng AVL Ví dụ 2: cây khơng cân bằng Khi thêm node mới vào cây AVL cĩ th ể xảy ra các trường hợp mất cân bằng như sau : Mất cân bằng phảiphải (RR) Mất cân bằng phảitrái (RL) Mất cân bằng tráitrái (LL) Mất cân bằng tráiphải (LR) 1 Trang Tài li u hư ng d n th c hành mơn Cu trúc d li u và gi i thu t
  2. Xử lý mất cân bằng bằng cách sử dụng các phép quay cây a. Quay trái b. Quay phải Xử lý cụ thể cho các trường hợp mất cân b ằng như sau: MẤT CÂN BẰNG PHẢI Mất cân bằng phảiphải (RR) Mất cân bằng phảitrái (RL) Quay trái tại node b ị mất cân Quay phải tại node con ph ải của bằng node bị mất cân bằng Quay trái tại node bị mất cân bằng MẤT CÂN BẰNG TRÁI Mất cân bằng tráitrái (LL) Mất cân bằng tráiphải (LR) Quay phải tại node b ị mất cân Quay trái tại node con trái của bằng node bị mất cân bằng Quay phải tại node bị mất cân bằng Giống với cây NPTK , các thao tác trên cây cân b ằng bao gồm: Thêm phần tử vào cây Tìm kiếm 1 phần tử trên cây Duyệt cây Xĩa 1 phần tử trên cây NI DUNG THC HÀNH Cơ bản Sinh viên đọc kỹ phát biểu bài tập và th ực hiện theo hướng dẫn: 2 Tổ chức một cây cân bằng AVL trong đĩ m ỗi node trên cây chứa thơng tin dữ liệu nguyên. Trang Tài li u hư ng d n th c hành mơn Cu trúc d li u và gi i thu t
  3. Người dùng sẽ nhập các giá trị nguyên từ bàn phím. Với mỗi giá trị nguyên được nhập vào, phải tạo cây AVL theo đúng tính chất của nĩ. Nếu người dùng nhập 1 quá trình nhập dữ liệu sẽ kết thúc. Sau đĩ, xuất thơng tin các node trên cây. Khi chương trình kết thúc, tất cả các node trên cây bị xĩa bỏ khỏi bộ nhớ. Phân tích Các node trên cây cân bằng cũng giống như các node trên cây NPTK. Tuy nhiên, do mỗi lần thêm node vào cây chúng ta cần kiểm tra độ cao của node vừa thêm để kiểm sốt tính cân bằng của cây nên cần bổ sung thêm giá trị cho biết sự cân bằng tại node đĩ vào cấu trúc của node. Cụ thể như sau: struct AVLNODE { int key; int bal; // thuc tính cho bit giá tr cân bng // 0: cân bng, 1: lch trái, 2: lch phi NODE* pLeft; NODE* pRight; }; Các thao tác cần cài đặt: xoay trái cây ( RotateLeft ), xoay phải cây ( RotateRight ), thêm 1 node mới vào cây ( InsertNode ), duyệt cây theo ( Traverse ), xĩa tồn bộ node trên cây ( RemoveAll ) Để chương trình mạch lạc và rõ ràng hơn, chúng ta sẽ cài đặt 2 hàm xử lý cân bằng khi cây lệch trái và lệch phải theo bảng phân loại ở trang 2. Như vậy trong chương trình sẽ cĩ thêm 2 hàm BalanceLeft và BalanceRight . Chương trình tham khảo #include struct AVLNODE { int Key; int bal; // thuc tính cho bit giá tr cân bng // 0: cân bng, 1: lch trái, 2: lch phi AVLNODE* pLeft; AVLNODE* pRight; }; AVLNODE* CreateNode( int Data) { AVLNODE* pNode; pNode = new AVLNODE; //Xin cp phát b nh đng đ to mt phn t (node) mi if (pNode == NULL){ return NULL; } pNode>Key = Data; pNode>pLeft = NULL; pNode>pRight = NULL; pNode>bal = 0; //Ghi chú: gii thích ý nghĩa ca thao tác này return pNode; } 3 void LeftRotate(AVLNODE* &P) { Trang Tài liu hưng dn thc hành mơn Cu trúc d liu và gii thut HCMUS 2010
  4. AVLNODE *Q; Q = P>pRight; P>pRight = Q>pLeft; Q>pLeft = P; P = Q; } void RightRotate(AVLNODE* &P) { //Ghi chú: sinh viên t code cho hàm này } void LeftBalance(AVLNODE* &P) { switch (P>pLeft>bal){ case 1: //mt cân bng trái trái RightRotate(P); P>bal = 0; P>pRight>bal = 0; break ; case 2: //Ghi chú: cho bit đây là trưng hp mt cân bng nào? LeftRotate(P>pLeft); RightRotate(P); switch (P>bal){ case 0: P>pLeft>bal= 0; P>pRight>bal= 0; break ; case 1: P>pLeft>bal= 0; P>pRight>bal= 2; break ; case 2: P>pLeft>bal= 1; P>pRight>bal= 0; break ; } P>bal = 0; break ; } } void RightBalance(AVLNODE* &P) { switch (P>pRight>bal){ case 1: //Ghi chú: cho bit đây là trưng hp mt cân bng nào? RightRotate(P>pRight); LeftRotate(P); switch (P>bal){ case 0: P>pLeft>bal= 0; P>pRight>bal= 0; break ; case 1: P>pLeft>bal= 0; P>pRight>bal= 2; break ; case 2: P>pLeft>bal= 1; P>pRight>bal= 0; break ; } P>bal = 0; 4 break ; case 2: //Ghi chú: cho bit đây là trưng hp mt cân bng nào? LeftRotate(P); Trang Tài liu hưng dn thc hành mơn Cu trúc d liu và gii thut HCMUS 2010
  5. P>bal = 0; P>pLeft>bal = 0; break ; } } int InsertNode(AVLNODE* &tree, int x) { int res; if (tree==NULL){ //Ghi chú: cho bit ý nghĩa ca câu lnh này tree = CreateNode(x); if (tree==NULL){ return 1; //thêm ko thành cơng vì thiu b nh } return 2; //thêm thành cơng và làm tăng chiu cao cây } else { if (tree>Key==x){ return 0; //khĩa này đã tn ti trong cây } else if (tree>Key > x){ res = InsertNode(tree>pLeft,x); if (res bal){ //Ghi chú: gii thích ý nghĩa ca câu lnh switch này case 0: tree>bal = 1; return 2; case 1: LeftBalance(tree); return 1; case 2: tree>bal = 0; return 1; } } else { res = InsertNode(tree>pRight,x); if (res bal){ case 0: tree>bal=2; return 2; case 1: tree>bal = 0; return 1; case 2: RightBalance(tree); return 1; } } } } void Traverse(AVLNODE* t) { if (t!=NULL) { Traverse(t>pLeft); 5 printf( "Khoa: %d, can bang: %d\n" , t>Key,t>bal); Traverse(t>pRight); } Trang Tài liu hưng dn thc hành mơn Cu trúc d liu và gii thut HCMUS 2010
  6. } void RemoveAll(AVLNODE* &t) { if (t!=NULL){ RemoveAll(t>pLeft); RemoveAll(t>pRight); delete t; } } int _tmain( int argc, _TCHAR* argv[]) { AVLNODE *tree; //Ghi chu: Ti sao li phi thc hin phép gán phía dưi? tree = NULL; int Data; do { printf( "Nhap vao du lieu, 1 de ket thuc: " ); scanf( "%d" , &Data); if (Data == 1) break ; InsertNode(tree, Data); }while (Data != 1); printf( "\nCay AVL vua tao: \n" ); Traverse(tree); RemoveAll(tree); return 0; } Yêu cầu 1. Biên dịch đoạn chương trình nêu trên. 2. Cho biết kết quả in ra màn hình khi người dùng nhập vào các dữ liệu sau: 1 10 30 35 32 20 8 1 30 40 50 10 5 1 3. Nhận xét trình tự các node được xuất ra màn hình? Giải thích tại sao lại in ra được trình tự như nhận xét? 4. Sinh viên hồn tất hàm RightRotate trong source code. Gợi ý: RightRotate tương tự hàm LeftRotate. 5. Biên dịch lại chương trình sau khi hồn thành câu 3 và cho biết kết quả in ra màn hình khi người dùng nhập vào các dữ liệu sau: 50 20 30 10 5 7 15 35 57 65 55 1 6. Vẽ hình cây AVL được tạo ra từ phần nhập liệu ở câu 5. 7. Hãy ghi chú các thơng tin bằng cách trả lời các câu hỏi ứng với các dịng lệnh cĩ yêu cầu ghi chú (//Ghi chú ) trong các hàm InsertNode, BalanceLeft, BalanceRight, _tmain . 8. Sinh viên cài đặt lại các hàm dùng cho cây nhị phân và cây NPTK để áp dụng cho cây AVL. 6 Trang Tài liu hưng dn thc hành mơn Cu trúc d liu và gii thut HCMUS 2010
  7. Áp dụng – Nâng cao 1. Sinh viên tự cài đặt thêm chức năng cho phép người dùng nhập vào khĩa x và kiểm tra xem khĩa x cĩ nằm trong cây AVL hay khơng. Cho dãy A như sau: 1 3 5 7 9 12 15 17 21 23 25 27 a. Tạo cây AVL từ dãy A. Cho biết số phép so sánh cần thực hiện để tìm phần tử 21 trên cây AVL vừa tạo. b. Tạo cây nhị phân tìm kiếm từ dãy A dùng lại đoạn code tạo cây của bài thực hành trước). Cho biết số phép so sánh cần thực hiện để tìm phần tử 21 trên cây nhị phân tìm kiếm vừa tạo. c. So sánh 2 kết quả trên và rút ra nhận xét? 2. Cài đặt chương trình đọc các số nguyên từ tập tin input.txt (khơng biết trước số lượng số nguyên trên tập tin) và tạo cây AVL từ dữ liệu đọc được 3. Cài đặt cây cân bằng AVL trong đĩ mỗi node trên cây lưu thơng tin sinh viên. 4. Tự tìm hiểu và cài đặt chức năng xĩa một node ra khỏi cây AVL. BÀI TẬP THÊM 1. Viết chương trình cho phép tạo, tra cứu và sửa chữa từ điển AnhViệt (sinh viên liên hệ với GVLT để chép file từ điển AnhViệt) 2. Cài đặt lại các bài tập thêm của cây NPTK bằng cách dùng cây AVL 7 Trang Tài liu hưng dn thc hành mơn Cu trúc d liu và gii thut HCMUS 2010