Đề tài Cấu trúc dữ liệu động

Dùng đề lưu trữ những ñối tượng dữ liệu ñược sử dụng không có nhu cầu thay ñổi và kích thước, số lượng . . . • ðược khai báo tường minh • Tồn tại trong phạm vi khái báo • Kích thước không thay ñổi trong suốt quá trình sống

pdf80 trang | Chia sẻ: tlsuongmuoi | Lượt xem: 2048 | Lượt tải: 0download
Bạn đang xem trước 20 trang tài liệu Đề tài Cấu trúc dữ liệu động, để xem tài liệu hoàn chỉnh bạn click vào nút DOWNLOAD ở trên
13.1. Kiểu dữ liệu con trỏ 3.2. Danh sách liên kết (link list) 3.3. Danh sách liên kết ñơn 3.4. Sắp xếp danh sách 3.5. Các cấu trúc ñặc biệt của danh sách liên kết ñơn 3.5.1. Stack 3.5.2. Hàng ñợi (Queue) 3.6. Bài tập Chương 3: CẤU TRÚC DỮ LIỆU ðỘNG Khoa KTCN Trường ðH KTKT B.Dương© Dương Thành Phết-www.thayphet.net 23.1. Kiểu Dữ Liệu Con Trỏ © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.1.1. Biến không ñộng 3.1.2. Kiểu con trỏ 3.1.3. Biến ñộng 3 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Dùng ñề lưu trữ những ñối tượng dữ liệu ñược sử dụng không có nhu cầu thay ñổi và kích thước, số lượng . . . • ðược khai báo tường minh • Tồn tại trong phạm vi khái báo • Kích thước không thay ñổi trong suốt quá trình sống Ví dụ: int a; char b[10]; 3.1.1. Biến không ñộng 4 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Kiểu con trỏ là kiểu cơ sở dùng lưu ñịa chỉ của một ñối tượng dữ liệu khác. Biến thuộc kiểu con trỏ là biến mà giá trị của nó là ñịa chỉ một vùng nhớ của một biến hoặc là giá trị Null. Tùy vào loại con trỏ gần (near pointer) hay con trỏ xa (far pointer) mà kiểu dữ liệu con trỏ có các kích thước khác nhau: + Con trỏ gần: 2 bytes + Con trỏ xa: 4 bytes 3.1.2. Kiểu con trỏ 5 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Cú pháp ñịnh nghĩa một kiểu con trỏ typedef *; Ví dụ: typedef int *intpointer; inpointer p; Hay int *p; Các thao tác: - Khi 1 biến con trỏ p lưu ñịa chỉ của ñối tượng x, ta nói “p trỏ x” - Gán ñịa chỉ của biến cho con trỏ p: p=& -Truy xuất nội dung của ñối tượng do p trỏ ñến *p 6 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương c. Ví dụ: void main() { int a,b,*pa,*pb; a=2; b=3; cout<<"\nGia tri cua bien a="<<a; cout<<"\nGia tri cua bien b="<<b; pa=&a; pb=&b; cout<<"\nDia chi cua o nho con tro pa tro toi="<<pa; cout<<"\nDia chi cua o nho con tro pb tro toi="<<pb; cout<<"\nNoi dung cua o nho con tro pa tro toi="<<*pa; cout<<"\nNoi dung cua o nho con tro pb tro toi="<<*pb; *pa=20; /* Thay doi giá tr cua *pa*/ *pb=20; /* Thay doi giá tri cua *pb*/ cout<<"\nGia tri moi cua bien a="<<a; cout<<"\nGia tri moi cua bien b="<<b; } 7 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Trong trường hợp, tại thời ñiểm biên dịch không thể xác ñịnh trước kích thước chính xác của ñối tượng dữ liệu(do chúng phụ thuộc vào ngữ cảnh). Các ñối tượng dữ liệu này ñược khai báo như biến ñộng. Biến ñộng là những biến thỏa: 3.1.3. Biến ñộng  Không ñược khai báo tường minh.  ðược cấp phát/giải phóng bộ nhớ khi yêu cầu.  Các biến này không theo qui tắc phạm vi.  Vùng nhớ của biến ñược cấp phát trong Heap.  Kích thước thay ñổi trong quá trình sống. 8 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Các thao tác trên biến ñộng: Tạo biến ñộng và cho con trỏ p trỏ ñến: Các hàm cấp phát bộ nhớ: void* malloc(size); // trả về con trỏ chỉ ñến một vùng // nhớ size byte vừa ñược cấp phát. void* calloc(n,size);// trả về con trỏ chỉ ñến một vùng // nhớ vừa ñược cấp phát gồm n //phần tử,mỗi phần tử có kích //thước size byte new // hàm cấp phát bộ nhớ trong C++ 9 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Hủy một biến ñộng do p chỉ ñến: Hàm free(p): Huỷ vùng nhớ cấp phát bởi hàm malloc do p trỏ tới Hàm delete p: huỷ vùng nhớ cấp phát bởi hàm new do p trỏ tới 10 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Ví dụ: //Trong C int* p1, p2; //Cấp phát vùng nhớ cho 1 biến ñộng kiểu int p1 = (int*)malloc(sizeof(int)); //ðặt giá trị 5 cho biến ñộng p1 p1* = 5; //Cấp phát biến ñộng kiểu mảng 10 p.tử kiểu int p2 = (int*)calloc(10, sizeof(int)); //ðặt giá trị 0 cho phần tử thứ 4 của mảng p2 (p2+3)* = 0; free(p1); free(p2); //Trong C++ int* p; p=new int; delete p; 11 3.2. Danh Sách Liên Kết © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.2.1. Ðịnh nghĩa 3.2.2. Các hình thức tổ chức danh sách 12 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.2.1. Ðịnh nghĩa Kiểu danh sách Tx gồm các phần tử thuộc kiểu T ñược ñịnh nghĩa là: Tx = trong ñó:  Vx = {tập hợp có thứ tự các phần tử kiểu T ñược móc nối với nhau theo trình tự tuyến tính};  Ox = {Tạo danh sách; Tìm 1 phần tử trong danh sách; Chèn một phần tử vào danh sách; Huỷ một phần tử khỏi danh sách ; Liệt kê danh sách, Sắp xếp danh sách ...} 13 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Ví du: Hồ sơ các học sinh của một trường ñược tổ chức thành danh sách gồm nhiều hồ sơ của từng học sinh; số lượng học sinh trong trường có thể thay ñổi do vậy cần có các thao tác thêm, hủy một hồ sơ; ñể phục vụ công tác giáo vụ cần thực hiện các thao tác tìm hồ sơ của một học sinh, in danh sách hồ sơ ... 14 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.2.2. Các hình thức tổ chức danh sách  Mối liên hệ giữa các phần tử ñược thể hiện ngầm:  Mỗi phần tử trong danh sách ñược ñặc trưng bằng chỉ số.  Cặp phần tử xi, xi+1 ñược xác ñịnh là kế cận  Với hình thức tổ chức này, các phần tử của danh sách phải lưu trữ liên tiếp trong bộ nhớ, công thức xác ñịnh ñịa chỉ phần tử thứ I là. Cách biểu diễn này cho phép truy xuất ngẫu nhiên, ñơn giản và nhanh chóng ñến một phần tử bất kỳ trong danh sách, nhưng lại hạn chế về mặt sử dụng bộ nhớ bị lãng phí. 15 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Mối liên hệ giữa các phần tử thể hiện tường minh:  Mỗi phần tử ngoài các thông tin còn chứa một liên kết (ñịa chỉ) ñến phần tử kế trong danh sách nên còn ñược gọi là danh sách móc nối.  Với hình thức này các phần tử trong danh sách không cần phải lưu trữ kế cận trong bộ nhớ nên khắc phục ñược các khuyết ñiểm của hình thức tổ chức mảng, nhưng việc truy xuất ñến một phần tử ñòi hỏi phải thực hiện truy xuất qua một số phần tử khác. 16 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Có các kiểu tổ chức liên kết giữa các phần tử.  Danh sách liên kết ñơn  Danh sách liên kết kép  Danh sách liên kết vòng 17 3.3. Danh Sách Liên Kết ðơn © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.3.1. Tổ chức danh sách ñơn theo cách cấp phát liên kết 3.3.2. Các thao tác cơ bản trên danh sách ñơn 18 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.3.1. Tổ chức danh sách ñơn theo cách cấp phát liên kết Mỗi phần tử là một cấu trúc chứa 2 thông tin : - Thành phần dữ liệu: Lưu trữ các thông tin về bản thân phần tử . - Thành phần mối liên kết: lưu trữ ñịa chỉ của phần tử kế tiếp trong danh sách, hoặc lưu trữ giá trị NULL nếu là phần tử cuối danh s ách. Ta có ñịnh nghĩa tổng quát struct tNode { DataType key; tNode* pNext; }; 19 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Ví dụ: Ðịnh nghĩa danh sách ñơn lưu trữ hồ sơ SV struct tSinhVien { char Ten[30]; int MaSV; }; typedef struct tSinhVien Sinhvien; struct SinhvienNode { Sinhvien Info; SinhvienNode* pNext; }; typedef struct SinvienNode SVNode; 20 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Nếu biết ñược ñịa chỉ của phần tử ñầu tiên trong danh sách ñơn thì có thể dựa vào thông tin pNext của nó ñể truy xuất ñến phần tử thứ 2, thứ 3... ðể quản lý một xâu ñơn chỉ cần biết ñịa chỉ phần tử ñầu xâu. Con trỏ Head dùng ñể lưu trữ ñịa chỉ phần tử ñầu xâu Khai báo: NODE *pHead; Ðể tiện lợi, có thể sử dụng thêm một con trỏ pTail giữ ñịa chỉ phần tử cuối xâu. Khai báo: NODE *pTail; 21 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.3.2. Các thao tác cơ bản trên danh sách ñơn 22 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Các ñịnh nghĩa, Khởi tạo danh sách LK ðơn: struct tNode{ int key; tNode *pnext; }; typedef struct tNode *Node; void Khoitao(List &L){ L.pHead=L.pTail=NULL; } struct tList{ node *pHead; node *pTail; }; typedef struct tList List; 23 © Dương Thành Phết-www.thayphet.net Khoa CNTT Trường Cð CNTT TP.HCM Các hàm con tạo Nút và Huỷ nút : Node * Taonut(int x){ Node *p; p=new Node; p->key=x; p->pNext=NULL; return p; } void Huy(List &L){ Node *p; while(L.pHead) { p=L.pHead; L.pHead=L.pHead->pNext; delete p; } } 24 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương  Chèn một phần tử vào ñầu danh sách Thu t toán : Bắt ñầu: Nếu Danh sách rỗng Thì B11 : Head = new_elelment; B12 : Tail = Head; Ngược lại B21 : new_ele ->pNext = Head; B22 : Head = new_ele ; void Themdau(List &L, Node *p){ if(L.pHead==NULL) L.pHead=L.pTail=p; else { p->pNext=L.pHead; L.pHead=p; } } 25 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương void Nhap(List &L) { int x; Node *p; Khoitao(L); do { cout<<"Nhap gia tri vao danh sach (Nhap 0 ket thuc): "; cin>>x; if(x==0) break; p=Taonut(x); Themdau(L,p); }while(true); } 26 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương void Xuat(List L){ Node *p=L.pHead; while(p) { coutKey<<" "; p=p->pNext; } } void main() { List L; Nhap(L); cout<<"\nDanh sach vua nhap: "; Xuat(l); Huy(L); } 27 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương  Chèn một phần tử vào cuối danh sách Thu t toán : Bắt ñầu : Nếu Danh sách rỗng Thì B11 : Head = new_elelment; B12 : Tail = Head; Ngược lại B21 : Tail ->pNext = new_ele; B22 : Tail = new_ele ; void Themcuoi(List &L, Node *p){ if(L.pHead==NULL) L.pHead=L.pTail=p; else { L.pTail->pNext=p; L.pTail=p; } } 28 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương  Chèn một phần tử vào sau phần tử q Thu t toán : Bắt ñầu : Nếu ( q != NULL) thì B1: new_ele -> pNext = q->pNext; B2: q->pNext = new_ele void Thempsauq(List &L, Node *p, Node *q){ if(q->pNext!=NULL) p->pNext=q->pNext; q->pNext=p; } 29 © Dương Thành Phết-www.thayphet.net Khoa CNTT Trường Cð CNTT TP.HCM  Tìm một phần tử trong danh sách ñơn Thuật toán : Áp dụng thuật toán tìm tuyến tính ñể xác ñịnh phần tử trong xâu có khoá k: Bước 1: p = Head; //Cho p trỏ ñến phần tử ñầu danh sách Bước 2: Trong khi (p != NULL) và (p->key != k ) p:=p->Next;// Cho p trỏ tới phần tử kế Bước 3: Nếu p != NULL thì p trỏ tới phần tử cần tìm Ngược lại không có phần tử cần tìm. Node *Search(LIST l, int k){ Node *p; p = l.pHead; while((p!= NULL)&&(p->key != k)) p = p->pNext; return p; } 30 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương NODE *Search(LIST l, Data k) { NODE *p; p = l.pHead; while((p!= NULL)&&(p->Info != x)) p = p->pNext; return p; } 31 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương  Hủy một phần tử ñầu danh sách ñơn Thu t toán : Bắt ñầu: Nếu (Head != NULL) thì B1: p = Head; B2: B21 : Head = Head->pNext; // tách p ra khỏi xâu B22 : free(p); // Hủy biến ñộng do p trỏ ñến B3: Nếu Head=NULL thì Tail = NULL; //Xâu rỗng. 32 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Data RemoveHead(LIST &l) { NODE *p; Data x = NULLDATA; if ( l.pHead != NULL) { p = l.pHead; x = p->Info; l.pHead = l.pHead->pNext; delete p; if(l.pHead == NULL) l.pTail = NULL; } return x; } 33 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương  Hủy một phần tử ñứng sau phần tử q Thu t toán : Bắt ñầu: Nếu (q!= NULL) thì B1: p = q->Next; // p là phần tử cần hủy B2: Nếu (p != NULL) thì // q không phải là cuối xâu B21 : q->Next = p->Next; // tách p ra khỏi xâu B22 : free(p); // Hủy biến ñộng do p trỏ ñến 34 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương void RemoveAfter (LIST &l, NODE *q) { NODE *p; if ( q != NULL) { p = q ->pNext ; if ( p != NULL) { if(p == l.pTail) l.pTail = q; q->pNext = p->pNext; delete p; } } else RemoveHead(l); } 35 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương  Hủy một phần tử có khóa k Thu t toán : Bước 1: Tìm phần tử p có khóa k và phần tử q ñứng trước nó Bước 2: Nếu (p!= NULL) thì // tìm thấy k Hủy p ra khỏi xâu tương tự hủy phần tử sau q; Ngược lại Báo không có k; 36 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương int RemoveNode(LIST &l, Data k) { NODE *p = l.pHead; NODE *q = NULL; while( p != NULL) { if(p->Info == k) break; q = p; p = p->pNext; } if(p == NULL) return 0; //Không tìm thấy k if(q != NULL) { if(p == l.pTail) l.pTail = q; q->pNext = p->pNext; delete p; }else{ l.pHead = p->pNext; if(l.pHead == NULL) l.pTail = NULL; } return 1; } 37 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương  Duyệt danh sách Duyệt danh sách là thao tác thường ñược thực hiện khi có nhu cầu xử lý các phần tử của danh sách như: - Ðếm các phần tử của danh sách, - Tìm tất cả các phần tử thoả ñiều kiện, - Huỷ toàn bộ danh sách (và giải phóng bộ nhớ) Thu t toán : Bước 1: p = Head; //Cho p trỏ ñến phần tử ñầu DS Bước 2: Trong khi (Danh sách chưa hết) thực hiện B21 : Xử lý phần tử p; B22 : p:=p->pNext; // Cho p trỏ tới phần tử kế 38 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương void ProcessList (LIST &l) { NODE *p; p = l.pHead; while (p!= NULL) { ProcessNode(p); // xử lý cụ thể tùy ứng dụng p = p->pNext; } } 39 3.4. Sắp Xếp Danh Sách © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.4.1. Các cách tiếp cận 3.4.2. Một số phương pháp sắp xếp trên danh sách 40 3.4.1. Các cách tiếp cận © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương DS có thứ tự là DS mà các phần tử ñược sắp xếp theo một thứ tự nào ñó dựa trên trường khoá. Ðể sắp xếp một danh sách, có 2 phương án: Phương án 1: Hoán vị nội dung các phần tử trong danh sách  Sử dụng các thuật toán sắp xếp như trên mảng, dựa trên việc hoán vị nội dung của các phần tử  Phương pháp này ñòi hỏi sử dụng vùng nhớ trung gian, số lần hoán vị có thể lên ñến bậc n2. Làm cho thao tác sắp xếp chậm không tận dụng ñược các ưu ñiểm của DSLK . 41 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Ví dụ : Cài ñặt thuật toán sắp xếp Chọn trực tiếp void ListSelectionSort (LIST &l) { NODE *min,*p,*q; p = l.pHead; int tam; while(p != l.pTail) { q = p->pNext; min = p; while(q != NULL){ if(q->KeyKey ) min = q; q= q->pNext; } tam=min->Key; min->Key=p->Key; p->Key=tam; p = p->pNext; } } 42 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Phương án 2: Thay ñổi các mối liên kết (trên vùng Next) Thay ñổi trình tự móc nối của các phần tử sao cho tạo lập nên ñược thứ tự mong muốn, Tuy nhiên thao tác trên các mọc nối thường sẽ phức tạp hơn là thao tác trực tiếp trên dữ liệu. Bước1: Khởi tạo danh sách mới Result là rỗng; Bước2: Tìm trong danh sách cũ l phần tử nhỏ nhất; Bước3: Tách min khỏi danh sách l; Bước4: Chèn min vào cuối danh sách Result; Bước5: Lặp lại bước 2 khi chưa hết danh sách Head; 43 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương void ListSelectionSort2 (LIST &l){ LIST lRes; NODE *min; // chỉ ñến phần tử có giá trị nhỏ nhất NODE *p,*q, minprev; lRes.pHead = lRes.pTail = NULL; // khởi tạo lRes while(l.pHead != NULL){ p = l.pHead; q = p->pNext; min = p; minprev = NULL; while(q != NULL){ if(q->InfoInfo ) { min = q; minprev = p } p = q; q = q->pNext; } if(minprev != NULL) minprev->pNext = min->pNext; else l.pHead = min->pNext; min->pNext = NULL; AddTail(lRes, min); } l = lRes; } 44 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.4.2. Một số phương pháp sắp xếp trên danh sách  Thuật toán Quick Sort Bước 1: Chọn X là phần tử ñầu DS L làm phần tử cầm canh. Loại X ra khỏi L. Bước 2: Tách DS L ra làm 2 DS L1 (gồm các phần tử nhỏ hơn hay bằng X) và L2 (gồm các phần tử lớn hơn X). Bước 3: Nếu L1 != NULL thì Quick Sort (L1). Bước 4: Nếu L2 != NULL thì Quick Sort (L2). Bước 5: Nối L1, X, và L2 theo trình tự ta có DS L ñược sắp xếp. 45 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Ví dụ Chọn X = 4 làm phần tử cầm canh và tách L thành L1, L2: Sắp xếp L1: 46 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Sắp xếp L2: Chọn X = 6 làm phần tử cầm canh và tách L2 thanh L21, L22 Nối L12, X2, L22 thành L2: Nối L1, X, L2 thành L: 47 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương void ListQSort(LIST & l){ NODE *p, *X; // X chỉ ñến phần tử cầm canh LIST l1, l2; if(l.pHead == l.pTail) return;//ñã có thứ tự l1.pHead == l1.pTail = NULL; //khởi tạo l2.pHead == l2.pTail = NULL; X = l.pHead; l.pHead = X->pNext; while(l.pHead != NULL) //Tách l thành l1, l2;{ p = l.pHead; l.pHead = p->pNext; p->pNext = NULL; if (p->Info Info) AddTail(l1, p); else AddTail(l2, p); } ListQSort(l1); ListQSort(l2); //Gọi ñệ qui ñể sorl1, sort l2 //Nối l1, X và l2 lại thành l ñã sắp xếp. if(l1.pHead != NULL) { l.pHead = l1.pHead; l1.pTail->pNext = X; } else l.pHead = X; X->pNext = l2; if(l2.pHead != NULL) l.pTail = l2.pTail; else l.pTail = X; } 48 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Ngoài ra còn một số thuật toán khác như Merge Sort, Radix Sort 49 3.5. Các Cấu Trúc ðặc Biệt Của Danh Sách Liên Kết ðơn © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.5.1. Stack 3.5.2. Hàng ñợi (Queue) 50 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Stack là một vật chứa (container) làm việc theo cơ chế LIFO (Last In First Out) "Vào sau ra trước". Thao tác thêm 1 ñối tượng vào stack gọi là "Push". Thao tác lấy 1 ñối tượng ra khỏi stack gọi là "Pop". Stack có nhiều ứng dụng: khử ñệ qui, tổ chức lưu vết các quá trình tìm kiếm theo chiều sâu, quay lui, vét cạn, ứng dụng trong tính toán biểu thức, ñồ thị . . . 3.5.1. Stack Hình ảnh stack Push Pop 51 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương CTDL Stack hỗ trợ các thao tác:  Push(o): Thêm ñối tượng o vào ñầu stack  Pop(): Lấy ñối tượng ở ñầu stack ra khỏi stack và trả về giá trị của nó. Nếu stack rỗng thì lỗi sẽ xảy ra.  isEmpty(): Kiểm tra xem stack có rỗng không.  isFull():Kiểm tra xem stack có ñầy không.  StackTop(): Trả về giá trị của phần tử nằm ở ñầu stack. Nếu stack rỗng thì lỗi sẽ xảy ra. 52 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Ðể biểu diễn Stack: Dùng mảng 1 chiều hoặc danh sách liên kết. Mảng 1 chiều Danh sách liên kết ñơn - Viết chương trình dễ dàng, nhanh chóng - Bị hạn chế do số lượng phần tử cố ñịnh - Tốn chi phí tái cấp phát và sao chép vùng nhớ nếu sử dụng mảng ñộng Phức tạp khi triển khai chương trình Không bị cố ñịnh về số phần tử, phụ thuộc vào bộ nhớ 53 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương // Giả sử Stack chứa các phần tử kiểu nguyên // Khai báo cấu trúc Stack typedef struct STACK { int* StkArray; // mảng chứa các phần tử int StkMax; // số phần tử tối ña int StkTop; // vị trí ñỉnh Stack }; //Thao tác “Khởi tạo Stack rỗng” int InitStack(STACK& s, int MaxItems) { s.StkArray = new int[MaxItems]; if (s.StkArray == NULL) return 0; // Không cấp phát ñược bộ nhớ s.StkMax = MaxItems; s.StkTop = -1; // chưa có phần tử nào trong Stack return 1; // khởi tạo thành công } Ngăn xếp sử dụng mảng 54 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác “Kiểm tra Stack rỗng” int IsEmpty(const STACK &s) { if (s.StkTop==-1) return 1; // Stack rỗng return 0; // Stack không rỗng } //Thao tác “Kiểm tra Stack ñầy” int IsFull(const STACK &s) { if (s.StkTop==s.StkMax-1) return 1; // Stack ñầy return 0; // Stack chưa ñầy } 55 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác “Push”: thêm một phần tử vào ñỉnh Stack int Push (STACK& s, int newitem) { if (IsFull(s)) return 0; // stack ñầy, không thể thêm s.StkTop++; s.StkArray[s.StkTop] = newitem; return 1; // thêm thành công } //Thao tác “Pop”: lấy ra 1 phần tử từ ñỉnh Stack int Pop(STACK& s, int& outitem) { if (IsEmpty(s)) return 0; // Stack rỗng, không lấy ra ñược outitem = s.StkArray[s.StkTop]; s.StkTop--; return 1; // lấy ra thành công } 56 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //truy xuất 1 phần tử ở ñỉnh Stack, không làm thay ñổi Stack int StackTop(const STACK s, int& outitem) { if (IsEmpty(s)) return 0; // Stack rỗng, không lấy ra ñược outitem = s.StkArray[s.StkTop]; return 1; // lấy ra thành công } 57 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương // Giả sử Stack chứa các phần tử kiểu nguyên // Khai báo cấu trúc một phần tử trong stack typedef struct tagSTACK_NODE { int Data; tagSTACK_NODE *pNext; } STACK_NODE; // Khai báo cấu trúc stack typedef struct STACK { int StkCount; STACK_NODE *StkTop; }; Ngăn xếp sử dụng DSLK 58 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác “Khởi tạo stack rỗng”: void InitStack(STACK& s) { s.StkTop = NULL; s.StkCount = 0; } //Thao tác “Kiểm tra stack rỗng” int IsEmpty(const STACK& s) { if (s.StkTop == NULL) return 1; // stack rỗng return 0; // stack không rỗng } 59 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác “Kiểm tra stack ñầy” int IsFull (const STACK s) { // thử tạo mới một phần tử STACK_NODE* temp = new STACK_NODE; // nếu không tạo ñược stack ñầy if (temp == NULL) return 1; // stack ñầy delete temp; return 0; // stack chưa ñầy } 60 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác “Push”: thêm 1 phần tử vào ñỉnh stack int Push(STACK &s, int newitem) { if (IsFull(s)) return 0; // Stack ñầy, không thêm vào ñược STACK_NODE *pNew = new STACK_NODE; pNew->Data = newitem; pNew->pNext = s.StkTop; s.StkTop = pNew; s.StkCount++; return 1; // Thêm thành công } 61 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác “Pop”: lấy ra 1 phần tử từ ñỉnh stack int Pop(STACK &s, int& outitem) { if (IsEmpty(s)) return 0; // Stack rỗng, không lấy ra ñược // lưu lại con trỏ ñến ptử ñầu STACK_NODE *temp = s.StkTop; outitem = s.StkTop->Data; s.StkTop = s.StkTop->pNext; delete temp; s.StkCount--; return 1; // Lấy ra thành công } 62 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác “StackTop”: truy xuất 1 phần tử ở ñỉnh stack int StackTop(STACK &s, int& outitem) { if (IsEmpty(s)) return 0; // Stack rỗng, không lấy ra ñược outitem = s.StkTop->Data; return 1; // Lấy ra thành công } 63 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương 3.5.2. Hàng ñợi (Queue)  Hàng ñợi là một vật chứa (container) các ñối tượng  Làm việc theo cơ chế FIFO (First In First Out) "Vào trước ra trước". Phòng vé Hình ảnh queue 64 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương CTDL Queue hỗ trợ các thao tác: InitQueue: Khởi tạo hàng ñợi rỗng IsEmpty: Kiểm tra hàng ñợi rỗng ? IsFull: Kiểm tra hàng ñợi ñầy ? EnQueue: Thêm 1 phần tử vào cuối hàng ñợi, có thể làm hàng ñợi ñầy DeQueue: Lấy ra 1 phần tử từ ñầu Queue, có thể làm Queue rỗng QueueFront: Kiểm tra phần tử ñầu Queue  QueueRear: Kiểm tra phần tử cuối Queue 65 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Ðể biểu diễn Queue: Dùng mảng 1 chiều hoặc danh sách liên kết. Mảng 1 chiều Danh sách liên kết ñơn - Viết chương trình dễ dàng, nhanh chóng - Bị hạn chế do số lượng phần tử cố ñịnh - Tốn chi phí tái cấp phát và sao chép vùng nhớ nếu sử dụng mảng ñộng Phức tạp khi triển khai chương trình Không bị cố ñịnh về số phần tử, phụ thuộc vào bộ nhớ 66 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương  Mảng (QArray) ñể chứa các phần tử.  Số nguyên (QMax)ñể lưu số phần tử tối ña  2 số nguyên (Qfront/QRear) ñể xác ñịnh vị trí ñầu/cuối  Số nguyên (QNumItems) ñể lưu số phần tử hiện có Hàng ñợi sử dụng mảng 0 1 2 3 4 5 6 Qarray 37 22 15 3 QMax = 7 QNumItems = 4 QFront = 1 QRear = 4 67 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương // Giả sử Queue chứa các phần tử kiểu nguyên (int) // Khai báo cấu trúc Queue typedef struct QUEUE { int* QArray; int QMax; int QNumItems; int QFront; int QRear; }; 68 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Khi thêm nhiều phần tử sẽ xảy ra hiện tượng “tràn giả” Giải pháp? Nối dài mảng (mảng ñộng) hay sử dụng một mảng vô cùng lớn? 0 1 2 3 4 5 6 Qarray 37 22 15 3 7 9 QMax = 7 QNumItems = 4 QFront = 1 QRear = 7 69 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác “Khởi tạo Queue rỗng”: int InitQueue(QUEUE& q, int MaxItem) { q.QArray = new int[MaxItem]; if (q.QArray == NULL) return 0; // không ñủ bộ nhớ q.QMax = MaxItem; q.QNumItems = 0; q.QFront = q.QRear = -1; return 1; // thành công } 70 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác “Kiểm tra Queue rỗng”: int IsEmpty(QUEUE q) { if (q.QNumItems == 0) return 1; // rỗng return 0; // không rỗng } //Thao tác “Kiểm tra Queue ñầy”: int IsFull(QUEUE q) { if (q.QMax == q.QNumItems) return 1; // ñầy return 0; // không ñầy } 71 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác EnQueue: thêm 1 phần tử vào cuối Queue int EnQueue(QUEUE &q, int newitem) { if (IsFull(q)) return 0; // Queue ñầy, không thêm vào ñược q.QRear++; if (q.QRear==q.QMax) // “tràn giả” q.QRear = 0; // Quay trở về ñầu mảng q.QArray[q.QRear] = newitem; // thêm phần tử vào cuối Queue if (q.QNumItems==0) q.QFront = 0; q.QNumItems++; return 1; // Thêm thành công } 72 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác DeQueue: lấy ra 1 phần tử ở ñầu Queue int DeQueue(QUEUE &q, int& itemout) { if (IsEmpty(q)) return 0; // Queue rỗng, không lấy ra ñược itemout = q.QArray[q.QFront]; // lấy phần tử ñầu ra q.QFront++; q.QNumItems--; if (q.QFront==q.QMax) // nếu ñi hết mảng … q.QFront = 0; // … quay trở về ñầu mảng if (q.QNumItems==0) // nếu lấy ra phần tử cuối cùng q.QFront = q.QRear = -1; // khởi tạo lại Queue return 1; // Lấy ra thành công } 73 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thao tác QueueFront: Kiểm tra phần tử ở ñầu Queue int QueueFront(const QUEUE &q, int& itemout) { if (IsEmpty(q)) return 0; // Queue rỗng, không kiểm tra // lấy phần tử ñầu ra itemout = q.QArray[q.QFront]; return 1; } //Thao tác QueueRear: Kiểm tra phần tử ở cuối Queue int QueueRear(const QUEUE &q, int& itemout) { if (IsEmpty(q)) return 0; // Queue rỗng, không kiểm tra // lấy phần tử cuối ra itemout = q.QArray[q.QRear]; return 1; } 74 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Khai báo cấu trúc typedef struct tagNODE { int data; tagNODE* pNext; } NODE, *PNODE; typedef struct tagQUEUE { int NumItems; PNODE pFront, pRear; } QUEUE; Hàng ñợi sử dụng DSLK 75 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương Các thao tác cơ bản int InitQueue(QUEUE& q); int IsEmpty(const QUEUE& q); int IsFull(const QUEUE& q); int EnQueue(QUEUE &q, int newitem); int DeQueue(QUEUE &q, int& itemout); int QueueFront(const QUEUE &q, int& itemout); int QueueRear(const QUEUE &q, int& itemout); 76 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Khởi tạo Queue rỗng int InitQueue(QUEUE& q) { q.NumItems = 0; q.pFront = q.pRear = NULL; return 1; } 77 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Kiểm tra Queue rỗng int IsEmpty(const QUEUE& q) { return (q.NumItems==0); } //Kiểm tra Queue ñầy int IsFull(const QUEUE& q) { PNODE tmp = new NODE; if (tmp==NULL) return 1; delete tmp; return 0; } 78 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Thêm 1 phần tử vào cuối Queue int EnQueue(QUEUE &q, int newitem) { if (IsFull(q)==1) return 0; PNODE p = new NODE; p->data = newitem; p->pNext = NULL; if (q.pFront==NULL && q.pRear==NULL) q.pFront = q.pRear = p; else { q.pRear->pNext = p; q.pRear = p; } q.NumItems++; return 1; } 79 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Lấy ra 1 phần tử ở ñầu Queue int DeQueue(QUEUE &q, int& itemout) { if (IsEmpty(q)==1) return 0; PNODE p = q.pFront; q.pFront = p->pNext; itemout = p->data; q.NumItems--; delete p; if (q.NumItems==0) InitQueue(q); return 1; } 80 © Dương Thành Phết-www.thayphet.net Khoa KTCN Trường ðH KTKT B.Dương //Kiểm tra 1 phần tử ở ñầu Queue int QueueFront(const QUEUE &q, int& itemout) { if (IsEmpty(q)==1) return 0; itemout = q.pFront->data; return 1; } //Kiểm tra 1 phần tử ở cuối Queue int QueueRear(const QUEUE &q, int& itemout) { if (IsEmpty(q)==1) return 0; itemout = q.pRear->data; return 1; }

Các file đính kèm theo tài liệu này:

  • pdf04_chuong3_7991.pdf
Tài liệu liên quan