Lập trình hướng đối tượng

Giới thiệu

 Các khái niệm cơ bản

 Lớp và đối tượng

 Kỹ thuật thừa kế và tính đa hình

 Thiết kế chương trình hướng đối tượng

 Khuôn mẫu (template)

 Luồng nhập xuất

 Các mẫu thiết kế hướng đối tượng

pdf418 trang | Chia sẻ: Mr Hưng | Lượt xem: 802 | Lượt tải: 0download
Bạn đang xem trước 20 trang nội dung tài liệu Lập trình hướng đối tượng, để xem tài liệu hoàn chỉnh bạn click vào nút DOWNLOAD ở trên
NULL;} List(Item a) { last = new Link(a, NULL); last- >next = last; } ~List() { CleanUp(); } bool Empty() const {return last == NULL;} bool IsMember(Item x) const; int Count() const; void View() const; }; 49 of 111 Không dùng kế thừa private class Stack { List l; public: Stack():l() {} bool Push(Item x) {l.Insert(x); return true;} bool Pop(Item *px); bool Empty() const {return l.Empty();} }; inline bool Stack::Pop(Item *px) { Item *p = l.GetFirst(); if (!p) return false; *px = *p; return true; } 50 of 111 Không dùng kế thừa private class Set { List l; public: Set():l(){} void Add(Item x) {if (!l.IsMember(x)) Insert(x);} bool Empty() const {return l.Empty();} bool IsMember(Item x) const {return l.IsMember(x);} int Count() const {return l.Count();} void View() const {l.View();} }; 51 of 111 Không dùng kế thừa private class HinhTron { double r; Diem tam; public: HinhTron(double tx, double ty, double rr): tam(tx,ty),r(rr){} void Ve(int color) const; void TinhTien(double dx, double dy) const; }; 52 of 111 Không dùng kế thừa private Diem dd[] = {Diem(100,100), Diem(200,200), Diem(200,50)}; class DaGiac { MangDiem md; public: DaGiac():md(3,3,dd){} DaGiac(int sd, Diem *pd):md(sd,sd,pd){} DaGiac(const DaGiac &d):md(d.md){} DaGiac& operator = (const DaGiac &d); void Ve(int color) const; void Quay(Diem Tam, double goc); void TinhTien(double dx, double dy) const; Diem TrongTam() const; //... }; 53 of 111 Phương thức thiết lập và huỷ bỏ • Phương thức thiết lập và huỷ bỏ là các hàm thành phần đặc biệt dùng để tự động khởi động đối tượng khi nó được tạo ra và tự động dọn dẹp đối tượng khi nó bị hủy đi. • Một đối tượng thuộc lớp con có chứa các thành phần dữ liệu của các lớp cơ sở. Có thể xem lớp con có các thành phần ngầm định ứng với các lớp cơ sở. Vì vậy khi một đố tượng thuộc lớp con được tạo ra, các thành phần cơ sở cũng được tạo ra, nghĩa là phương thức thiết lập của các lớp cơ sở phải được gọi. • Trình biên dịch tự động gọi phương thức thiết lập của các lớp cơ sở cho các đối tượng (cơ sở) nhúng vào đối tượng đạng được tạo ra. • Đối với phương thức thiết lập của một lớp con, công việc đầu tiên là gọi phương thức thiết lập của các lớp cơ sở. 54 of 111 Phương thức thiết lập và huỷ bỏ • Nếu mọi phương thức thiết lập của lớp cơ sở đều đòi hỏi phải cung cấp tham số thì lớp con bắt buộc phải có phương thức thiết lập để cung cấp các tham số đó. class Diem { double x,y; public: Diem(double x, double y):x(xx),y(yy){} //... }; class HinhTron: private Diem { double r; public: void Ve(int color) const; }; HinhTron t; // SAI 55 of 111 Cung cấp tham số cho phương thức thiết lập của lớp cha • Trong trường hợp đó, lớp con bắt buộc phải có phương thức thiết lập để cung cấp tham số cho phương thức thiết lập của lớp cơ sở. Cú pháp để gọi phương thức thiết lập của lớp cơ sở tương tự như cú pháp thiết lập đối tượng thành phần, bản thân tên lớp cơ sở được quan điểm như đối tượng thành phần nhúng vào lớp con. class HinhTron: private Diem { double r; public: HinhTron(double tx, double ty, double rr):Diem(tx,ty),r(rr){} void Ve(int color) const; void TinhTien(double dx, double dy) const; }; HinhTron t(200,200,50); // Dung 56 of 111 Phương thức thiết lập và huỷ bỏ class Nguoi { protected: char *HoTen; int NamSinh; public: Nguoi(char *ht, int ns):NamSinh(ns) { HoTen = strdup(ht); } //... }; class SinhVien : public Nguoi { char *MaSo; public: SinhVien(char *ht, char *ms, int ns) : Nguoi(ht,ns) { MaSo = strdup(ms); } void Xuat() const; }; 57 of 111 Phương thức thiết lập và huỷ bỏ • Sau khi phương thức thiết lập của các lớp cơ sở được gọi, mã chương trình trong bản thân phương thức của lớp con sẽ được thực hiện. Nội dung của phương thức thiết lập ở lớp con chỉ nên thao tác trên dữ liệu của riêng lớp con, việc khởi động dữ liệu thuộc lớp cha do phương thức thiết lập ở lớp cha đảm nhiệm với các tham số cung cấp bởi lớp con. class SinhVien : public Nguoi { char *MaSo; public: SinhVien(char *ht, char *ms, int ns) : Nguoi(ht,ns) { MaSo = strdup(ms); } void Xuat() const; }; 58 of 111 Phương thức thiết lập và huỷ bỏ • Ta có thể khởi động các thành phần của lớp cha bên trong phương thức thiết lập của lớp con. Trong trường hợp này đối tượng thuộc lớp cha phải có khả năng tự khởi động: class Complex { protected: double re, im; public: Complex(double r = 0, double i = 0):re(r), im(i){} Complex operator +(Complex b); Complex operator -(Complex b); Complex operator *(Complex b); Complex operator /(Complex b); double Norm() const {return sqrt(re*re + im*im);} }; 59 of 111 Phương thức thiết lập và huỷ bỏ • Hai cách thiết lập đối tượng thuộc lớp con sau đây tương đương: // Cach 1 class Imag: public Complex { public: Imag(double i = 0):Complex(0, i){} //... }; // Cach 2 class Imag: public Complex { public: Imag(double i = 0){re = 0; im = i;){} //... }; 60 of 111 Phương thức thiết lập và huỷ bỏ class Nguoi { protected: char *HoTen; int NamSinh; public: Nguoi(char *ht = "Ng Van A", int ns = 1980) :NamSinh(ns) {HoTen = strdup(ht);} ~Nguoi() {delete [] HoTen;} //... }; 61 of 111 Phương thức thiết lập và huỷ bỏ // Cach 1 class SinhVien : public Nguoi { char *MaSo; public: SinhVien(char *ht, char *ms, int ns) : Nguoi(ht,ns) { MaSo = strdup(ms); } void Xuat() const; }; // Cach 2 class SinhVien : public Nguoi { char *MaSo; public: SinhVien(char *ht, char *ms, int ns) { HoTen = strdup(ht); MaSo = strdup(ms); NamSinh = ns; } }; 62 of 111 Phương thức thiết lập và huỷ bỏ • Phương thức thiết lập sao chép là cần thiết trong trường hợp đối tượng có nhu cầu cấp phát tài nguyên. class Nguoi { protected: char *HoTen; int NamSinh; public: Nguoi(char *ht, int ns):NamSinh(ns) { HoTen = strdup(ht); } Nguoi(const Nguoi &n):NamSinh(n.NamSinh) { HoTen = strdup(n.HoTen); } ~Nguoi() { delete [] HoTen;} //... }; 63 of 111 Phương thức thiết lập và huỷ bỏ • Ở lớp con liệu có cần phương thức thiết lập sao chép ? // Khong dung pttl sao chep : D hay S class SinhVien : public Nguoi { char *MaSo; public: SinhVien(char *ht, char *ms, int ns) : Nguoi(ht,ns) { MaSo = strdup(ms);} ~SinhVien() {delete [] MaSo;} //... }; void main() { SinhVien s1("Vo Vien Sinh", "200002541",1984); SinhVien s2(s1); } 64 of 111 Phương thức thiết lập và huỷ bỏ • Ở lớp con liệu có cần phương thức thiết lập sao chép ? // Lop con co pttl sao chep : co can thiet ? class SinhVien : public Nguoi { char *MaSo; public: SinhVien(char *ht, char *ms, int ns) : Nguoi(ht,ns) { MaSo = strdup(ms);} SinhVien(const SinhVien &s) : Nguoi(s) { MaSo = strdup(s.MaSo);} //... }; void main() { SinhVien s1("Vo Vien Sinh", "200002541",1984); SinhVien s2(s1); } 65 of 111 Phương thức thiết lập và huỷ bỏ • Khi một đối tượng bị huỷ đi, phương thức huỷ bỏ của nó sẽ được gọi, sau đó phương thức huỷ bỏ của các lớp cơ sở sẽ được gọi một cách tự động. Vì vậy lớp con không cần và cũng không được thực hiện các thao tác dọn dẹp cho các thành phần thuộc lớp cha. class SinhVien : public Nguoi { char *MaSo; public: SinhVien(char *ht, char *ms, int ns) : Nguoi(ht,ns) { MaSo = strdup(ms);} SinhVien(const SinhVien &s) : Nguoi(s) { MaSo = strdup(s.MaSo);} ~SinhVien() {delete [] MaSo;} //... }; 66 of 111 Câu Hỏi • Khi lớp có nhu cầu cấp phát tài nguyên thì cần có bộ tứ phương thức gì? Cho một ví dụ bằng ngôn ngữ C++. • Giả sử đã có lớp PhânSố có các phương thức khởi tạo, set/get, các phương thức Cộng và Trừ. Hãy xây dựng lớp MyPhanSo có thêm phương thức Nhân và Chia. • Khi xây dựng lớp con kế thừa từ lớp cha thì cần phải khai báo, cài đặt những phương thức gì ở lớp con. 67 of 111 Con trỏ và kế thừa Con trỏ trong kế thừa hoạt động theo nguyên tắc sau: • Con trỏ đến đối tượng thuộc lớp cơ sở thì có thể trỏ đến các đối tượng thuộc lớp con. • Điều ngược lại không đúng, con trỏ đến đối tượng thuộc lớp con thì không thể trỏ đến các đối tượng thuộc lớp cơ sở. • Ta có thể ép kiểu để con trỏ đến đối tượng thuộc lớp con có thể trỏ đến đối tượng thuộc lớp cơ sở. Tuy nhiên thao tác này có thể nguy hiểm. • Sử dụng ép kiểu đúng cách có thể giải quyết bài toán quản lý một danh sách các đối tượng khác kiểu. 68 of 111 Con trỏ và kế thừa void main() { clrscr(); Nguoi n("Nguyen Van Nhan", 1970); SinhVien s("Vo Vien Sinh", "200002541",1984); Nguoi *pn; SinhVien *ps; pn = &n; ps = &s; pn = &s; ps = &n; // Sai ps = pn; // Sai ps = (SinhVien *)&n; // Sai logic ps = (SinhVien *)pn; } Phương thức ảo và tính đa hình 70 of 111 Bài toán quản lý một danh sách các đối tượng khác kiểu - Giả sử ta cần quản lý một danh sách các đối tượng có kiểu có thể khác nhau, ta cần giải quyết hai vấn đề: Cách lưu trữ và thao tác xử lý. - Xét trường hợp cụ thể, các đối tượng có thể là người, sinh viên hoặc công nhân. - Về lưu trữ: Ta có thể dùng union, trong trường hợp này mỗi đối tượng phải có kích thước chứa được đối tượng có kích thước lớn nhất. Điều này gây lãng phí không gian lưu trữ. Một cách thay thế là lưu trữ đối tượng bằng đúng kích thước của nó và dùng một danh sách (mảng, dslk,...) các con trỏ để quản lý các đối tượng. - Về thao tác, phải thoả yêu cầu đa hình: Thao tác có hoạt động khác nhau ứng với các loại đối tượng khác nhau. Có hai cách giải quyết là vùng chọn kiểu và phương thức ảo. 71 of 111 Dùng vùng chọn kiểu • Về lưu trữ: Ta sẽ dùng một mảng các con trỏ đến lớp cơ sở để có thể trỏ đến các đối tượng thuộc lớp con. • Xét lớp Người và các lớp kế thừa sinh viên và công nhân. Thao tác ta quan tâm là xuat. Ta cần bảo đảm thao tác xuất áp dụng cho lớp sinh viên và lớp công nhân khác nhau. 72 of 111 Dùng vùng chọn kiểu class Nguoi { protected: char *HoTen; int NamSinh; public: Nguoi(char *ht, int ns):NamSinh(ns) {HoTen = strdup(ht);} ~Nguoi() {delete [] HoTen;} void An() const { cout << HoTen << " an 3 chen com";} void Ngu() const { cout << HoTen << " ngu ngay 8 tieng";} void Xuat() const { cout << "Nguoi, ho ten: " << HoTen << " sinh " << NamSinh; } }; 73 of 111 class SinhVien : public Nguoi { protected: char *MaSo; public: SinhVien(char *n, char *ms, int ns) : Nguoi(n,ns) { MaSo = strdup(ms);} ~SinhVien() {delete [] MaSo;} void Xuat() const { cout << "Sinh vien " << HoTen << ", ma so " << MaSo;} }; 74 of 111 class NuSinh : public SinhVien { public: NuSinh(char *ht, char *ms, int ns) : SinhVien(ht,ms,ns) {} void An() const { cout << HoTen << " ma so " << MaSo << " an 2 to pho";} }; 75 of 111 Dùng vùng chọn kiểu class CongNhan : public Nguoi { protected: double MucLuong; public: CongNhan(char *n, double ml, int ns) : Nguoi(n,ns), MucLuong(ml) { } void Xuat() const { cout << "Cong nhan, ten " << HoTen << " muc luong: " << MucLuong;} }; 76 of 111 void XuatDs(int n, Nguoi *an[]) { for (int i = 0; i < n; i++) { an[i]->Xuat(); cout << "\n"; } } 77 of 111 Dùng vùng chọn kiểu const int N = 4; void main() { Nguoi *a[N]; a[0] = new SinhVien("Vien Van Sinh", ”200001234", 1982); a[1] = new NuSinh("Le Thi Ha Dong", ”200001235", 1984); a[2] = new CongNhan("Tran Nhan Cong", 1000000, 1984); a[3] = new Nguoi("Nguyen Thanh Nhan", 1960); XuatDs(4,a); } 78 of 111 Dùng vùng chọn kiểu • Xuất liệu cho đoạn chương trình trên như sau: – Nguoi, ho ten: Vien Van Sinh sinh 1982 – Nguoi, ho ten: Le Thi Ha Dong sinh 1984 – Nguoi, ho ten: Tran Nhan Cong sinh 1984 – Nguoi, ho ten: Nguyen Thanh Nhan sinh 1960 • Tất cả mọi đối tượng đều được quan điểm như người vì thao tác được thực hiện thông qua con trỏ đến lớp Người. • Để bảo đảm xuất liệu tương ứng với đối tượng, phải có cách nhận diện đối tượng, ta thêm một vùng dữ liệu vào lớp cơ sở để nhận diện, vùng này có giá trị phụ thuộc vào loại của đối tượng và được gọi là vùng chọn kiểu. • Các đối tượng thuộc lớp người có cùng giá trị cho vùng chọn kiểu, các đối tượng thuộc lớp sinh viên có giá trị của vùng chọn kiểu khác của lớp người. 79 of 111 Dùng vùng chọn kiểu class Nguoi { public: enum LOAI {NGUOI, SV, CN}; protected: char *HoTen; int NamSinh; public: LOAI pl; Nguoi(char *ht, int ns):NamSinh(ns), pl(NGUOI) {HoTen = strdup(ht);} ~Nguoi() {delete [] HoTen;} void An() const { cout << HoTen << " an 3 chen com";} void Ngu() const { cout << HoTen << " ngu ngay 8 tieng";} void Xuat() const { cout << "Nguoi, ho ten: " << HoTen << " sinh " << NamSinh; } }; 80 of 111 Dùng vùng chọn kiểu class SinhVien : public Nguoi { protected: char *MaSo; public: SinhVien(char *n, char *ms, int ns) : Nguoi(n,ns) { MaSo = strdup(ms); pl = SV; } ~SinhVien() {delete [] MaSo;} void Xuat() const { cout << "Sinh vien " << HoTen << ", ma so " << MaSo;} }; 81 of 111 class NuSinh : public SinhVien { public: NuSinh(char *ht, char *ms, int ns) : SinhVien(ht,ms,ns) {} void An() const { cout << HoTen << " ma so " << MaSo << " an 2 to pho";} }; 82 of 111 Dùng vùng chọn kiểu class CongNhan : public Nguoi { protected: double MucLuong; public: CongNhan(char *n, double ml, int ns) : Nguoi(n,ns), MucLuong(ml) { pl = CN;} void Xuat() const { cout << "Cong nhan, ten " << HoTen << " muc luong: " << MucLuong;} }; 83 of 111 Dùng vùng chọn kiểu •Khi thao tác ta phải căn cứ vào giá trị của vùng chọn kiểu để “ép kiểu” phù hợp. void XuatDs(int n, Nguoi *an[]) { for (int i = 0; i < n; i++) { switch(an[i]->pl) { case Nguoi::SV: ((SinhVien *)an[i])->Xuat(); break; case Nguoi::CN: ((CongNhan *)an[i])->Xuat(); break; default: an[i]->Xuat(); break; } cout << "\n"; } } 84 of 111 Dùng vùng chọn kiểu const int N = 4; void main() { Nguoi *a[N]; a[0] = new SinhVien("Vien Van Sinh", "200001234", 1982); a[1] = new NuSinh("Le Thi Ha Dong", "200001235", 1984); a[2] = new CongNhan("Tran Nhan Cong", 1000000, 1984); a[3] = new Nguoi("Nguyen Thanh Nhan", 1960); XuatDs(4,a); } Sinh vien Vien Van Sinh, ma so 200001234 Sinh vien Le Thi Ha Dong, ma so 200001235 Cong nhan, ten Tran Nhan Cong muc luong: 1000000 Nguoi, ho ten: Nguyen Thanh Nhan sinh 1960 85 of 111 Dùng vùng chọn kiểu • Cách tiếp cận trên giải quyết được vấn để: Lưu trữ được các đối tượng khác kiểu nhau và thao tác khác nhau tương ứng với đối tượng. Tuy nhiên nó có các nhược điểm sau: – Dài dòng với nhiều switch, case. – Dễ sai sót, khó sửa vì trình biên dịch bị cơ chế ép kiểu che mắt. – Khó nâng cấp ví dụ thêm một loại đối tượng mới, đặc biệt khi chương trình lớn. • Các nhược điểm trên có thể được khắc phục nhờ phương thức ảo. 86 of 111 Phương thức ảo • Con trỏ thuộc lớp cơ sở có thể trỏ đến lớp con: – Nguoi* pn = new SinhVien(“Le Vien Sinh”, 200001234, 1982); • Ta mong muốn thông qua con trỏ thuộc lớp cơ sở có thể truy xuất hàm thành phần được định nghĩa lại ở lớp con: – pn->Xuat(); // Mong muon: goi Xuat cua lop sinh vien, – // thuc te: goi Xuat cua lop Nguoi • Phương thức ảo cho phép giải quyết vấn đề. Ta qui định một hàm thành phần là phương thức ảo bằng cách thêm từ khoá virtual vào trước khai báo hàm. • Trong ví dụ trên, ta thêm từ khoá virtual vào trước khai báo của hàm xuat. 87 of 111 Phương thức ảo class Nguoi { protected: char *HoTen; int NamSinh; public: Nguoi(char *ht, int ns):NamSinh(ns) {HoTen = strdup(ht);} ~Nguoi() {delete [] HoTen;} void An() const { cout << HoTen << " an 3 chen com";} void Ngu() const { cout << HoTen << " ngu ngay 8 tieng";} virtual void Xuat() const { cout << "Nguoi, ho ten: " << HoTen << " sinh " << NamSinh; } }; 88 of 111 Phương thức ảo class SinhVien : public Nguoi { protected: char *MaSo; public: SinhVien(char *n, char *ms, int ns) : Nguoi(n,ns) { MaSo = strdup(ms);} ~SinhVien() {delete [] MaSo;} void Xuat() const { cout << "Sinh vien " << HoTen << ", ma so " << MaSo; } }; 89 of 111 class NuSinh : public SinhVien { public: NuSinh(char *ht, char *ms, int ns) : SinhVien(ht,ms,ns) {} void An() const { cout << HoTen << " ma so " << MaSo << " an 2 to pho"; } }; 90 of 111 Phương thức ảo class CongNhan : public Nguoi { protected: double MucLuong; public: CongNhan(char *n, double ml, int ns) : Nguoi(n,ns), MucLuong(ml) { } void Xuat() const { cout << "Cong nhan, ten " << HoTen << " muc luong: " << MucLuong; } }; 91 of 111 void XuatDs(int n, Nguoi *an[]) { for (int i = 0; i < n; i++) { an[i]->Xuat(); cout << "\n"; } } 92 of 111 Phương thức ảo const int N = 4; void main() { Nguoi *a[N]; a[0] = new SinhVien("Vien Van Sinh", "200001234", 1982); a[1] = new NuSinh("Le Thi Ha Dong", "200001235", 1984); a[2] = new CongNhan("Tran Nhan Cong", 1000000, 1984); a[3] = new Nguoi("Nguyen Thanh Nhan", 1960); XuatDs(4,a); } • Phương thức ảo xuat được khai báo ở lớp Nguoi cho phép sử dụng con trỏ đến lớp cơ sở (Nguoi) nhưng trỏ đến một đối tượng thuộc lớp con (Sinh viên, công nhân) gọi đúng thao tác ở lớp con: 93 of 111 Phương thức ảo • Nguoi *pn; • pn = new SinhVien("Vien Van Sinh", "200001234", 1982); • pn->Xuat(); // Goi thao tac xuat cua lop Sinh vien • Con trỏ pn thuộc lớp Nguoi nhưng trỏ đến đối tượng sinh viên, vì vậy pn->Xuat() thực hiện thao tác xuất của lớp sinh viên. • Trở lại ví dụ trên, khi i a[i] lần lượt trỏ đến các đối tượng thuộc các loại khác nhau, thao tác tương ứng với lớp sẽ được gọi. • Dùng phương thức ảo khắc phục được các nhược điểm của cách tiếp cận dùng vùng chọn kiểu: • Thao tác đơn giản không phải dùng switch/case vì vậy khó sai, dễ sửa. 94 of 111 Thêm lớp con mới • Dùng phương thức ảo, ta dễ dàng nâng cấp sửa chữa. Việc thêm một loại đối tượng mới rất đơn giản, ta không cần phải sửa đổi thao tác xử lý (hàm XuatDs). Qui trình thêm chỉ là xây dựng lớp con kế thừa từ lớp cơ sở hoặc các lớp con đã có và định nghĩa lại phương thức (ảo) ở lớp mới tạo nếu cần class CaSi : public Nguoi { protected: double CatXe; public: CaSi(char *ht, double cx, int ns) : Nguoi(ht,ns), CatXe(cx) {} void Xuat() const { cout << "Ca si, " << HoTen << " co cat xe " << CatXe;} }; 95 of 111 Thêm lớp con mới void XuatDs(int n, Nguoi *an[]) { for (int i = 0; i < n; i++) { an[i]->Xuat(); cout << "\n"; } } • Hàm XuatDs không thay đổi, nhưng nó có thể hoạt động cho các loại đối tượng ca sĩ thuộc lớp mới ra đời. • Có thể xem như thao tác XuatDs được viết trước cho các lớp con cháu chưa ra đời. 96 of 111 Các lưu ý khi sử dụng phương thức ảo • Phương thức ảo chỉ hoạt động thông qua con trỏ. • Muốn một hàm trở thành phương thức ảo có hai cách: Khai báo với từ khoá virtual hoặc hàm tương ứng ở lớp cơ sở đã là phương thức ảo. • Phương thức ảo chỉ hoạt động nếu các hàm ở lớp cơ sở và lớp con có nghi thức giao tiếp giống hệt nhau. • Nếu ở lớp con không định nghĩa lại phương thức ảo thì sẽ gọi phương thức ở lớp cơ sở (gần nhất có định nghĩa). 97 of 111 Cơ chế thực hiện phương thức ảo • Khi gọi một thao tác, khả năng chọn đúng phiên bản tuỳ theo đối tượng để thực hiện thông qua con trỏ đến lớp cơ sở được gọi là tính đa hình (polymorphisms). • Cơ chế đa hình được thực hiện nhờ ở mỗi đối tượng có thêm một bảng phương thức ảo. Bảng này chứa địa chỉ của các phương thức ảo và nó được trình biên dịch khởi tạo một cách ngầm định khi thiết lập đối tượng. • Khi thao tác được thực hiện thông qua con trỏ, hàm có địa chỉ trong bảng phương thức ảo sẽ được gọi. • Trong ví dụ trên, mỗi đối tượng thuộc lớp cơ sở Người có bảng phương thức ảo có một phần tử là địa chỉ hàm Nguoi::Xuat. Mỗi đối tượng thuộc lớp SinhVien có bảng tương tự nhưng nội dung là địa chỉ của hàm SinhVien::Xuat. 98 of 111 Cơ chế thực hiện phương thức ảo • Trong ví dụ 2, mỗi đối tượng thuộc các lớp Mamal, Dog, Cat, Horse, Pig đều có bảng phương thức ảo với hai phần tử, địa chỉ của hàm Speak và của hàm Move. 99 of 111 100 of 111 Phương thức huỷ bỏ ảo • Trong ví dụ quản lý danh sách các đối tượng thuộc các lớp Nguoi, SinhVien, CongNhan, Thao tác dọn dẹp đối tượng là cần thiết. const int N = 4; void main() { Nguoi *a[N]; a[0] = new SinhVien("Vien Van Sinh", "20001234", 1982); a[1] = new NuSinh("Le Thi Ha Dong", "20001235", 1984); a[2] = new CongNhan("Tran Nan Cong", 1000000, 1984); a[3] = new Nguoi("Nguyen Thanh Nhan", 1960); XuatDs(4,a); for (int i = 0; i < 4; i++) delete a[i]; } 101 of 111 Phương thức huỷ bỏ ảo • Thông qua con trỏ thuộc lớp cơ sở Nguoi, chỉ có phương thức huỷ bỏ của lớp Nguoi được gọi. • Để bảo đảm việc dọn dẹp là đầy đủ, ta dùng phương thức huỷ bỏ ảo. 102 of 111 class Nguoi { protected: char *HoTen; int NamSinh; public: Nguoi(char *ht, int ns):NamSinh(ns) {HoTen = strdup(ht);} virtual ~Nguoi() {delete [] HoTen;} virtual void Xuat(ostream &os) const { os << "Nguoi, ho ten: " << HoTen << " sinh " << NamSinh; } void Xuat() const { Xuat(cout); } }; 103 of 111 Phương thức thiết lập ảo • C++ không cung cấp cơ chế thiết lập đối tượng có khả năng đa hình theo cơ chế hàm thành phần ảo. • Tuy nhiên ta có thể “thu xếp” để có thể tạo đối tượng theo nghĩa “ảo”. • Phương thức thiết lập ảo cũng có thể được hiện thực bằng cách dùng hàm thành phần tĩnh để tạo đối tượng. 104 of 111 Phương thức thiết lập ảo enum FILETYPE {UNKNOWN, BMP, GIF, JPG}; class CGBmp; typedef CGBmp * CGBmpPtr; class CGBmp { protected: // ... public: virtual ~CGBmp(){Release();} virtual void Release() = 0; static CGBmpPtr NewBmp(const String &pathName); virtual void Draw() const; }; 105 of 111 Phương thức thiết lập ảo class CBmp : public CGBmp { //... public: CBmp(const String &pathName); void Release(); void Draw() const; //... }; 106 of 111 class CGif: public CGBmp { //.. public: CGif(const String &pathName); void Release(); void Draw() const; //... }; 107 of 111 Phương thức thiết lập ảo class CJpg: public CGBmp { //.. public: CJpg(const String &pathName); void Release(); void Draw() const; //... }; 108 of 111 FILETYPE GetFileType(const String &pathName) { String FileExt = GetFileExt(pathName); if (FileExt == ".DIB" || FileExt == ".BMP") return BMP; else if (FileExt == ".JPG") return JPG; else if (FileExt == ".GIF") return GIF; else return UNKNOWN; } 109 of 113 Phương thức thiết lập ảo CGBmpPtr CGBmp::NewBmp(const String &pathName) { switch (GetFileType(pathName)) { case BMP: return new CBmp(pathName); case GIF: return new CGif(pathName); case JPG: return new CJpg(pathName); default: return NULL; } } void main() { CGBmpPtr p = CGBmp::NewBmp("Lena.jpg"); p->Draw(); } 110 of 113 Phương thức thuần ảo và lớp cơ sở trừu tượng • Lớp cơ sở trừu tượng là lớp cơ sở không có đối tượng nào thuộc chính nó. Một đối tượng thuộc lớp cơ sở trừu tượng

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

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