c++chuong6

Chương 6

Tương ứng bội và phương thức ảo

Tương ứng bội và phương thức ảo là  công cụ mạnh của C++ cho phép tổ chức quản lý các đối tượng khác nhau theo cùng một lược đồ. Một khái niệm khác liên quan là: lớp cơ sở trừu tượng.  Chương này sẽ trình bầy cách sử dụng các công cụ trên để xây dựng chương trình quản lý nhiều đối tượng khác nhau theo một lược đồ thống nhất.

§ 1. Phương thức tĩnh

1.1. Lời gọi tới phương thức tĩnh

Như đã biết một lớp dẫn xuất được thừa kế các phương thức của các lớp cơ sở tiền bối của nó.  Ví dụ lớp A là cơ sở của B, lớp B lại là cơ sở của C, thì C có 2 lớp cơ sở tiền bối là B và A. Lớp C được thừa kế các phương thức của A và B. Các phương thức mà chúng ta vẫn nói là các phương thức tĩnh. Để tìm hiểu thêm về cách gọi tới các phương thức tĩnh, ta xét ví dụ về các lớp A, B và  C  như sau:

class A

{

public:

void xuat()

{

cout << "

Lop A " ;

}

};

class B:public A

{

public:

void xuat()

317                         318                                 

{

cout << "

Lop B " ;

}

};

class C:public B

{

public:

void xuat()

{

cout << "

Lop C " ;

}

};

Lớp C có 2 lớp cơ sở tiền bối là A , B và C kế thừa các phương thức của A và B. Do đó một đối tượng của C sẽ có  tới 3 phương thức xuat. Hãy theo rõi các câu lệnh sau:

C  h ; //  h là đối tượng kiểu C

h.xuat() ; // Gọi tới phương thức h.D::xuat()

h.B::xuat() ; // Gọi tới phương thức h.B::xuat()

h.A::xuat() ; // Gọi tới phương thức h.A::xuat()

Các lời gọi phương thức trong ví dụ trên đều xuất phát từ đối tượng h và mọi lời gọi đều xác định rõ phương thức cần gọi.

Bây giờ chúng ta hãy xét các lời gọi không phải từ một biến đối tượng mà từ một con trỏ. Xét các câu lệnh:

A  *p, *q, *r;   // p, q, r là con trỏ kiểu A

A   a;    //  a là đối tượng kiểu A

B   b;    //  b là đối tượng kiểu B

C   c;    //  c là đối tượng kiểu c

Chúng ta hãy ghi nhớ mệnh đề sau về con trỏ của các lớp dẫn xuất và cơ sở:

Phép gán con trỏ:  Con trỏ của lớp cơ sở có thể dùng để chứa địa chỉ các đối tượng của lớp dẫn xuất.

Như vậy cả 3 phép gán sau đều hợp lệ:

p =  &a ;

q =  &b ;

r =   &c ;

Chúng ta tiếp tục xét các lời gọi phương thức từ các con trỏ p, q, r:

p->xuat();

q->xuat();

r->xuat();

và hãy lý giải xem phương thức nào (trong các phương thức A::xuat, B::xuat và C::xuat) được gọi. Câu trả lời như sau:

Cả 3 câu lệnh trên đều gọi tới phương thức A::xuat() , vì các con trỏ p, q và r đều có kiểu A.

Như vậy có thể tóm lược cách thức gọi các phương thức tĩnh như sau:

Quy tắc gọi phương thức tĩnh: Lời gọi tới phương thức tĩnh bao giờ cũng xác định rõ phương thức nào (trong số các phương thức trùng tên của các lớp có quan hệ thừa kế) được gọi:

1. Nếu lời gọi xuất phát từ một đối tượng của lớp nào,  thì phương thức của lớp đó sẽ được gọi.

2. Nếu lời gọi xuất phát từ một  con trỏ kiểu lớp nào,  thì phương thức của lớp đó sẽ được gọi bất kể con trỏ chứa địa chỉ của đối tượng nào.

1.2. Ví dụ

Xét 4 lớp A, B, C và D. Lớp B và C có chung lớp cơ sở A. Lớp D dẫn xuất từ C. Cả 4 lớp đều có phương thức xuat(). Xét hàm:

void hien(A *p)

{

p->xuat();

319                         320                                 

}

Không cần biết tới địa chỉ của đối tượng nào sẽ truyền cho đối con trỏ p, lời gọi trong hàm luôn luôn gọi tới phương thức A::xuat() vì con trỏ p kiểu A. Như vậy bốn câu lệnh:

hien(&a);

hien(&b);

hien(&c);

hien(&d);

trong hàm main (của chương trình dưới đây) đều gọi tới A::xuat().

//CT6-01

// Phuong thuc tinh

#include <conio.h>

#include <stdio.h>

#include <iostream.h>

#include <ctype.h>

class A

{

private:

int n;

public:

A()

{

n=0;

}

A(int n1)

{

n=n1;

}

void xuat()

{

cout << "

Lop A: "<< n;

}

int getN()

{

return n;

}

};

class B:public A

{

public:

B():A()

{

}

B(int n1):A(n1)

{

}

void xuat()

{

cout << "

Lop B: "<<getN();

}

};

class C:public A

{

public:

C():A()

{

}

C(int n1):A(n1)

{

321                         322                                 

}

void xuat()

{

cout << "

Lop C: "<<getN();

}

};

class D:public C

{

public:

D():C()

{

}

D(int n1):C(n1)

{

}

void xuat()

{

cout << "

Lop D: "<<getN();

}

};

void hien(A *p)

{

p->xuat();

}

void main()

{

A a(1);

B b(2);

C c(3);

D d(4);

clrscr();

hien(&a);

hien(&b);

hien(&c);

hien(&d);

getch();

}

§ 2. Sự hạn chế của phương thức tĩnh

Ví dụ sau cho thấy sự hạn chế của phương thức tĩnh trong việc sử dụng tính thừa kế để phát triển chương trình.

Giả sử cần xây dựng chương trình quản lý thí sinh. Mỗi thí sinh đưa vào ba thuộc tính: Họ tên, số báo danh và tổng điểm. Chương trình gồm ba chức năng: Nhập dữ liệu thí sinh, in dữ liệu thí sinh ra máy in và xem - in (in họ tên ra màn hình, sau đó lựa chọn hoặc in hoặc không). Chương trình dưới đây sử dụng lớp TS (Thí sinh) đáp ứng được yêu cầu đặt ra.

//CT6-02

// Han che phuong thuc tinh

// Lop TS

#include <conio.h>

#include <stdio.h>

#include <iostream.h>

#include <ctype.h>

class TS

{

private:

char ht[25];

int sobd;

float td;

public:

void nhap()

323                         324                                 

{

cout << "

Ho ten: " ;

fflush(stdin); gets(ht);

cout << "So bao danh: " ;

cin >> sobd;

cout << "Tong diem: " ;

cin >> td;

}

void in()

{

fprintf(stdprn,"

Ho ten: %s", ht);

fprintf(stdprn,"

So bao danh: %d", sobd);

fprintf(stdprn,"

Tong diem: %0.1f", td);

}

void xem_in()

{

int ch;

cout << "

Ho ten: " << ht ;

cout << "

Co in khong? - C/K" ;

ch = toupper(getch());

if (ch=='C')

this->in();

}

} ;

void main()

{

TS t[100];

int i, n;

cout << "

So thi sinh: ";

cin >> n;

for (i=1; i<=n; ++i)

t[i].nhap();

for (i=1; i<=n; ++i)

t[i].xem_in();

getch();

}

Giả sử Nhà trường muốn quản lý thêm địa chỉ  của thí sinh. Vì sự thay đổi ở đây là không nhiều, nên chúng ta không đả động đến lớp TS mà xây dựng lớp mới TS2 dẫn xuất từ lớp TS. Trong lớp TS2 đưa thêm thuộc tính dc (địa chỉ) và các phương thức nhap, in. Cụ thể lớp TS2 được định nghĩa như sau:

class TS2:public TS

{

private:

char dc[30] ; // Dia chi

public:

void nhap()

{

TS::nhap();

cout << "Dia chi: " ;

fflush(stdin); gets(dc);

}

void in()

{

TS::in();

fprintf(stdprn,"

Dia chi: %s", dc);

}

};

325                         326                                 

Trong lớp TS2 không xây dựng lại phương thức xem_in, mà sẽ dùng phương thức xem_in của lớp TS. Chương trình mới như sau:

//CT6-03

// Han che phuong thuc tinh

// Lop TS TS2

#include <conio.h>

#include <stdio.h>

#include <iostream.h>

#include <ctype.h>

class TS

{

private:

char ht[25];

int sobd;

float td;

public:

void nhap()

{

cout << "

Ho ten: " ;

fflush(stdin); gets(ht);

cout << "So bao danh: " ;

cin >> sobd;

cout << "Tong diem: " ;

cin >> td;

}

void in()

{

fprintf(stdprn,"

Ho ten: %s", ht);

fprintf(stdprn,"

So bao danh: %d", sobd);

fprintf(stdprn,"

Tong diem: %0.1f", td);

}

void xem_in()

{

int ch;

cout << "

Ho ten: " << ht ;

cout << "

Co in khong? - C/K" ;

ch = toupper(getch());

if (ch=='C')

this->in(); //Goi den TS::in() (Vi this la con tro

//kieu TS)

}

} ;

class TS2:public TS

{

private:

char dc[30] ; // Dia chi

public:

void nhap()

{

TS::nhap();

cout << "Dia chi: " ;

fflush(stdin); gets(dc);

}

void in()

{

TS::in();

fprintf(stdprn,"

Dia chi: %s", dc);

}

327                         328                                 

};

void main()

{

TS2 t[100];

int i, n;

cout << "

So thi sinh: ";

cin >> n;

for (i=1; i<=n; ++i)

t[i].nhap();

for (i=1; i<=n; ++i)

t[i].xem_in();

getch();

}

Khi thực hiện chương trình này, chúng ta nhận thấy: Dữ liệu in ra vẫn không có địa chỉ. Điều này có thể giải thích như sau:

Xét câu lệnh (thứ 2 từ dưới lên trong hàm main):

t[i].xem_in() ;

Câu lệnh này gọi tới phương thức xem_in của lớp TS2 (vì t[i] là đối tượng của lớp TS2). Nhưng lớp TS2 không định nghĩa phương thức xem_in, nên phương thức TS::xem_in() sẽ được gọi tới. Hãy theo rõi phương thức này:

void xem_in()

{

int ch;

cout << "

Ho ten: " << ht ;

cout << "

Co in khong? - C/K" ;

ch = toupper(getch());

if(ch=='C')

this->in(); //Goi den TS::in() (Vi this la con tro kieu TS)

}

Các lệnh đầu của phương thức sẽ in họ tên thí sinh. Nếu chọn có (bấm phím C), thì  câu lệnh:

this->in() ;

sẽ được thực hiện. Mặc dù địa chỉ của t[i] (là đối tượng của lớp TS2) được truyền cho con trỏ this, thế nhưng  câu lệnh này luôn luôn gọi tới phương thức TS::in(), vì con trỏ this ở đây có kiểu TS và vì in() là phương thức tĩnh. Kết quả là không in được địa chỉ của thí sinh.

Như vậy việc sử dụng các phương thức tĩnh in() (trong các lớp TS và TS2) đã không đáp ứng được yêu cầu phát triển chương trình. Có một giải pháp rất đơn giản là: Định nghĩa các phương thức in() trong các lớp TS và TS2 như các phương thức ảo (virtual).

§ 3. Phương thức ảo và tương ứng bội

3.1. Cách định nghĩa phương thức ảo

Giả sử A là lớp cơ sở, các lớp B, C, D dẫn xuất (trực tiếp hoặc dán tiếp) từ A. Giả sử trong 4 lớp trên đều có các phương thức trùng dòng tiêu đề (trùng kiểu, trùng tên, trùng các đối). Để định nghĩa các phương thức này là các phương thức ảo, ta chỉ cần:

+ Hoặc thêm từ khoá virtual vào dòng tiêu đề của phương thức bên trong định nghĩa lớp cơ sở A.

+ Hoặc thêm từ khoá virtual vào dòng tiêu đề bên trong định nghĩa của  tất cả các lớp A, B, C và D.

Ví dụ:

Cách 1:

class A

{

...

virtual void  hien_thi()

{

329                         330                                 

cout << “

Đây là  lớp A” ;

};

} ;

class B : public A

{

...

void  hien_thi()

{

cout << “

Đây là  lớp B” ;

};

} ;

class C : public B

{

...

void  hien_thi()

{

cout << “

Đây là  lớp C” ;

};

} ;

class D : public A

{

...

void  hien_thi()

{

cout << “

Đây là  lớp D” ;

};

} ;

Cách 2:

class A

{

...

virtual void  hien_thi()

{

cout << “

Đây là  lớp A” ;

};

} ;

class B : public A

{

...

virtual  void  hien_thi()

{

cout << “

Đây là  lớp B” ;

};

} ;

class C : public B

{

...

virtual void  hien_thi()

{

cout << “

Đây là  lớp C” ;

};

} ;

class D : public A

{

...

virtual void  hien_thi()

{

cout << “

Đây là  lớp D” ;

};

331                         332                                 

} ;

Chú ý:  Từ khoá virtual không được đặt bên ngoài định nghĩa lớp. Ví dụ nếu viết như sau là sai (CTBD sẽ báo lỗi).

class A

{

...

virtual void  hien_thi() ;    

} ;

virtual void  hien_thi()  // Sai

{

cout << “

Đây là  lớp A” ;

};

Cần sửa lại như sau:

class A

{

...

virtual void  hien_thi() ;    

} ;

void  hien_thi()  // Đúng

{

cout << “

Đây là  lớp A” ;

};

3.2. Quy tắc gọi phương thức ảo

Để có sự so sánh với phương thức tĩnh, ta nhắc lại quy tắc gọi phương thức tĩnh nêu trong §1.

3.2.1. Quy tắc gọi phương thức tĩnh

Lời gọi tới phương thức tĩnh bao giờ cũng xác định rõ phương thức nào (trong số các phương thức trùng tên của các lớp có quan hệ thừa kế) được gọi:

1. Nếu lời gọi xuất phát từ một đối tượng của lớp nào,  thì phương thức của lớp đó sẽ được gọi.

2. Nếu lời gọi xuất phát từ một  con trỏ kiểu lớp nào,  thì phương thức của lớp đó sẽ được gọi bất kể con trỏ chứa địa chỉ của đối tượng nào.

3.2.2. Quy tắc gọi phương thức ảo

Phương thức ảo chỉ khác phương thức tĩnh khi được gọi từ một con trỏ (trường hợp 2 nêu trong mục 3.2.1). Lời gọi tới phương thức ảo từ một con trỏ chưa cho biết rõ phương thức nào (trong số các phương thức ảo trùng tên của các lớp có quan hệ thừa kế) sẽ được gọi. Điều này phụ thuộc vào đối tượng cụ thể mà con trỏ đang trỏ tới: Con trỏ đang trỏ tới đối tượng của lớp nào thì phương thức của lớp đó sẽ được gọi.

Ví dụ A, B, C và D là các lớp đã định nghĩa trong 3.1. Ta khai báo một con trỏ kiểu A và 4 đối tượng:

A  *p ;  //  p là con trỏ kiểu A

A   a ;   //  a là biến đối tượng kiểu A

B   b ;   //  b là biến đối tượng kiểu B

C   c ;   //  c là biến đối tượng kiểu C

D   d ;   //  d là biến đối tượng kiểu D

Xét lời gọi tới các phương thức ảo hien_thi sau:

p = &a;             // p trỏ tới đối tượng a của lớp A

p->hien_thi() ;  // Gọi tới A::hien_thi() 

p = &b;             // p trỏ tới đối tượng b của lớp B

p->hien_thi() ;  // Gọi tới B::hien_thi() 

p = &c;             // p trỏ tới đối tượng c của lớp C

p->hien_thi() ;  // Gọi tới C::hien_thi() 

p = &d;             // p trỏ tới đối tượng d của lớp D

p->hien_thi() ;  // Gọi tới D::hien_thi() 

333                         334                                 

3.3. Tương ứng bội

Chúng ta nhận thấy cùng một câu lệnh

p->hien_thi();

tương ứng với nhiều phương thức khác nhau. Đây chính là tương ứng bội. Khả năng này rõ ràng cho phép xử lý nhiều đối tượng khác nhau, nhiều công việc, thậm chí nhiều thuật toán khác nhau theo cùng một cách thức, cùng một lược đồ. Điều này sẽ được minh hoạ trong các mục tiếp theo.

3.4. Liên kết động

Có thể so sánh sự khác nhau giữ phương thức tĩnh và phương thức ảo trên khía cạnh liên kết một lời gọi với một phương thức.  Trở lại ví dụ trong 3.2:

A  *p ;  //  p là con trỏ kiểu A

A   a ;   //  a là biến đối tượng kiểu A

B   b ;   //  b là biến đối tượng kiểu B

C   c ;   //  c là biến đối tượng kiểu C

D   d ;   //  d là biến đối tượng kiểu D

Nếu hien_thi() là các phương thức tĩnh, thì dù p chứa địa chỉ của các đối tượng a, b, c hay d, thì lời gọi:

p->hien_thi() ; 

luôn luôn gọi tới phương thức A::hien_thi() 

Như vậy một lời gọi (xuất phát từ con trỏ) tới phương thức tĩnh luôn luôn liên kết với một phương thức cố định và sự liên kết này xác định trong quá trình biên dịch chương trình.

Cũng với lời gọi:

p->hien_thi() ;  

như trên, nhưng nếu hien_thi() là các phương thức ảo, thì lời gọi này không liên kết cứng với một phương thức cụ thể nào. Phương thức mà nó liên kết (gọi tới) còn chưa xác định trong giai đoạn dịch chương trình. Lời gọi này sẽ:

+ liên kết với A::hien_thi() , nếu p chứa địa chỉ đối tượng lớp A

+ liên kết với B::hien_thi() , nếu p chứa địa chỉ đối tượng lớp B

+ liên kết với C::hien_thi() , nếu p chứa địa chỉ đối tượng lớp C

+ liên kết với D::hien_thi() , nếu p chứa địa chỉ đối tượng lớp D

Như vậy một lời gọi (xuất phát từ con trỏ) tới phương thức ảo không liên kết với một phương thức cố định, mà tuỳ thuộc vào nội dung con trỏ. Đó là sự liên kết động và phương thức được liên kết (được gọi) thay đổi mỗi khi có sự thay đổi nội dung con trỏ trong quá trình chạy chương trình.

3.5. Quy tắc gán địa chỉ đối tượng cho con trỏ lớp cơ sở

+ Như đã nói trong §1, C++ cho phép gán địa chỉ đối tượng của một lớp dẫn xuất cho con trỏ của lớp cơ sở. Như vậy các phép gán sau (xem 3.2) là đúng:

A  *p ;  //  p là con trỏ kiểu A

A   a ;   //  a là biến đối tượng kiểu A

B   b ;   //  b là biến đối tượng kiểu B

C   c ;   //  c là biến đối tượng kiểu C

D   d ;   //  d là biến đối tượng kiểu D

p = &a;    // p và a cùng lớp A

p = &b;   // p là con trỏ lớp cơ sở, b là đối tượng lớp dẫn xuất

p = &c;   // p là con trỏ lớp cơ sở, c là đối tượng lớp dẫn xuất

p = &d;   // p là con trỏ lớp cơ sở, d là đối tượng lớp dẫn xuất

+ Tuy nhiên cần chú ý là: Không cho phép gán địa chỉ đối tượng của lớp cở sở cho con trỏ của lớp dẫn xuất. Như vậy ví dụ sau là sai:

B  *q ;

A  a ;

q = &a;

Sai vì:  Gán địa chỉ đối tượng của lớp cơ sở A cho con trỏ của lớp dẫn xuất B

3.6. Ví dụ

335                         336                                 

Ta sửa chương trình trong §1 bằng cách định nghĩa các phương thức xuat() là ảo. Khi đó bốn câu lệnh:

hien(&a);

hien(&b);

hien(&c);

hien(&d);

trong hàm main (của chương trình dưới đây) sẽ lần lượt gọi tới 4 phương thức khác nhau:

A::xuat()

B::xuat()

C::xuat()

D::xuat()

//CT6-01B

// Phuong thuc ảo và tương ứng bội

#include <conio.h>

#include <stdio.h>

#include <iostream.h>

#include <ctype.h>

class A

{

private:

int n;

public:

A()

{

n=0;

}

A(int n1)

{

n=n1;

}

virtual void xuat()

{

cout << "

Lop A: "<< n;

}

int getN()

{

return n;

}

};

class B:public A

{

public:

B():A()

{

}

B(int n1):A(n1)

{

}

void xuat()

{

cout << "

Lop B: "<<getN();

}

};

class C:public A

{

public:

C():A()

{

337                         338                                 

}

C(int n1):A(n1)

{

}

void xuat()

{

cout << "

Lop C: "<<getN();

}

};

class D:public C

{

public:

D():C()

{

}

D(int n1):C(n1)

{

}

void xuat()

{

cout << "

Lop D: "<<getN();

}

};

void hien(A *p)

{

p->xuat();

}

void main()

{

A a(1);

B b(2);

C c(3);

D d(4);

clrscr();

hien(&a);

hien(&b);

hien(&c);

hien(&d);

getch();

}

3.5. Sự thừa kế của các phương thức ảo

Cũng giống như các phương thức thông thường khác, phương thức ảo cũng có tính thừa kế. Chẳng hạn trong chương trình trên (mục 3.4) ta bỏ đi phương thức xuat() của lớp D, thì câu lệnh:

hien(&d) ;

(câu lệnh gần cuối trong hàm main) sẽ gọi tới C::xuat() , phương thức này được kế thừa trong lớp D (vì D dẫn xuất từ C).

§ 4. Sự linh hoạt của phương thức ảo trong phát triển nâng cấp chương trình

Ví dụ về các lớp TS và TS2 trong §2 đã chỉ ra sự hạn chế của phương thức tĩnh trong việc sử dụng tính thừa kế để nâng cấp, phát triển chương trình. Trong §2 cũng đã chỉ ra lớp TS2 chưa đáp ứng được yêu cầu nêu ra là in địa chỉ của thí sinh. Giải pháp cho vấn đề này rất đơn giản: Thay các phương thức tĩnh in() bằng cách dùng chúng như các phương thức ảo. Chương trình khi đó sẽ như sau:

//CT6-03B

// Sự linh hoạt của phương thức ảo

// Lop TS TS2

#include <conio.h>

339                         340                                 

#include <stdio.h>

#include <iostream.h>

#include <ctype.h>

class TS

{

private:

char ht[25];

int sobd;

float td;

public:

void nhap()

{

cout << "

Ho ten: " ;

fflush(stdin); gets(ht);

cout << "So bao danh: " ;

cin >> sobd;

cout << "Tong diem: " ;

cin >> td;

}

virtual  void in()

{

fprintf(stdprn,"

Ho ten: %s", ht);

fprintf(stdprn,"

So bao danh: %d", sobd);

fprintf(stdprn,"

Tong diem: %0.1f", td);

}

void xem_in()

{

int ch;

cout << "

Ho ten: " << ht ;

cout << "

Co in khong? - C/K" ;

ch = toupper(getch());

if (ch=='C')

this->in(); // Vì in() là phương thức ảo nên

//có thể gọi đến  TS::in() hoặc TS2::in()

}

} ;

class TS2:public TS

{

private:

char dc[30] ; // Dia chi

public:

void nhap()

{

TS::nhap();

cout << "Dia chi: " ;

fflush(stdin); gets(dc);

}

void in()

{

TS::in();

fprintf(stdprn,"

Dia chi: %s", dc);

}

};

void main()

{

TS2 t[100];

int i, n;

cout << "

So thi sinh: ";

341                         342                                 

cin >> n;

for (i=1; i<=n; ++i)

t[i].nhap();

for (i=1; i<=n; ++i)

       t[i].xem_in();

getch();

}

Khi thực hiện chương trình này, chúng ta nhận thấy: Dữ liệu thí sinh in ra đã có địa chỉ. Điều này có thể giải thích như sau:

Xét câu lệnh (thứ 2 từ dưới lên trong hàm main):

t[i].xem_in() ;

Câu lệnh này gọi tới phương thức xem_in của lớp TS2 (vì t[i] là đối tượng của lớp TS2). Nhưng lớp TS2 không định nghĩa phương thức xem_in, nên phương thức TS::xem_in() sẽ được gọi tới. Hãy theo rõi phương thức này:

void xem_in()

{

int ch;

cout << "

Ho ten: " << ht ;

cout << "

Co in khong? - C/K" ;

ch = toupper(getch());

this->in(); // Vì in() là phương thức ảo nên

                //có thể gọi đến  TS::in() hoặc TS2::in()

}

Các lệnh đầu của phương thức sẽ in họ tên thí sinh. Nếu chọn Có (bấm phím C), thì  câu lệnh:

this->in() ;

sẽ được thực hiện. Địa chỉ của t[i] (là đối tượng của lớp TS2) được truyền cho con trỏ this (của lớp cơ sở TS). Vì in() là phương thức ảo và vì this đang trỏ tới đối tượng t[i] của lớp TS2, nên câu lệnh này  gọi tới phương thức TS2::in(). Trong phương thức TS2::in() có in địa chỉ của thí sinh.

Như vậy việc sử dụng các phương thức tĩnh in() (trong các lớp TS và TS2) đã không đáp ứng được yêu cầu phát triển chương trình. Có một giải pháp rất đơn giản là: Định nghĩa các phương thức in() trong các lớp TS và TS2 như các phương thức ảo (virtual).

§ 5. Lớp cơ sở trừu tượng

5.1. Lớp cơ sở trừu tượng

       Một lớp cơ sở trừu tượng là một lớp chỉ được dùng làm cơ sở cho các lớp khác. Không hề có đối tượng nào của một lớp trừu tượng được tạo ra cả, bởi vì nó chỉ được dùng để định nghĩa một số khái niệm tổng quát, chung cho các lớp khác. Một ví dụ về lớp trừu tượng là lớp CON_VAT (con vật), nó sẽ dùng làm cơ sở để xây dựng các lớp con vật cụ thể như lớp CON_CHO (con chó), CON_MEO (con mèo),... (xem ví dụ bên dưới)

Trong C++ , thuật ngữ “Lớp trừu tượng” đặc biệt áp dụng cho các lớp có chứa các phương thức ảo thuần tuý. Phương thức ảo thuần tuý là một phương thức ảo mà nội dung của nó không có gì. Cách thức định nghĩa một phương thức ảo thuần tuý như sau:

virtual  void  tên_phương_thức() = 0 ;

Ví dụ:

class A

{

public:

virtual void  nhap() = 0 ;

virtual void  xuat() = 0 ;

void  chuong();

} ;

343                         344                                 

Trong ví dụ trên, thì A là lớp cơ sở trừu tượng. Các phương thức nhap và xuat được khai báo là các lớp ảo thuần tuý (bằng cách gán số 0 cho chúng thay cho việc cài đặt các phương thức này). Phương thức chuong() là một phương thức bình thường và sẽ phải có một định nghĩa ở đâu đó cho phương thức này.

Không có đối tượng nào của một lớp trừu tượng lại có thể được phát sinh. Tuy nhiên các con trỏ và các biến tham chiếu đến các đối tượng của lớp trừu tượng thì vẫn hợp lệ. Bất kỳ lớp nào dẫn xuất từ một lớp cớ sở trừu tượng phải định nghĩa lại tất cả các phương thức thuần ảo mà nó thừa hưởng, hoặc bằng các phương thức ảo thuần tuý, hoặc bằng những định nghĩa thực sự. Ví dụ:

class B : public A

{

public:

virtual void  nhap() = 0 ;

virtual void  xuat()

{

// Các câu lệnh

}

} ;

Theo ý nghĩa về hướng đối tượng, ta vẫn có thể có một lớp trừu tượng mà không nhất thiết phải chứa đựng những phương thức thuần tuý ảo.

 Một cách tổng quát mà nói thì bất kỳ lớp nào mà nó chỉ được dùng làm cơ sở cho những lớp khác đều có thể được gọi là “lớp trừu tượng”. Một cách dễ dàng để nhận biết một lớp trừu tượng là xem có dùng lớp đó để khai báo các đối tượng hay không? . Nếu không thì đó là lớp cơ sở trừu tượng.

5.2. Ví dụ

Giả sử có 20 ô, mỗi ô có thể nuôi một con chó hoặc một con mèo. Yêu cầu xây dựng chương trình gồm các chức năng:

+ Nhập một con vật mới mua (hoặc chó, hoặc mèo) vào ô rỗng đầu tiên.

+ Xuất (đem bán) một con vật (hoặc chó, hoặc mèo).

+ Thống kê các con vật đang nuôi trong 20 ô.

Chương trình được tổ chức như sau:

+ Trước tiên định nghĩa lớp CON_VAT là lớp cơ sở ảo. Lớp này có một thuộc tính là tên con vật và một phương thức ảo dùng để xưng tên.

+ Hai lớp là CON_MEO và CON_CHO được dẫn xuất từ lớp CON_VAT

+ Cuối cùng là lớp DS_CON_VAT (Danh sách con vật) dùng để quản lý chung cả mèo và chó. Lớp này có 3 thuộc tính là: số con vật cực đại (chính bằng số ô), số con vật đang nuôi và một mảng con trỏ kiểu CON_VAT. Mỗi phần tử mảng sẽ chứa địa chỉ của một đối tượng kiểu CON_MEO hoặc CON_CHO.

Lớp sẽ có 3 phương thức để thực hiện 3 chức năng nêu trên của chương trình. Nội dung chương trình như sau:

//CT6-04

// Lop co so truu tuong

// Lop CON_VAT

#include <conio.h>

#include <stdio.h>

#include <iostream.h>

#include <ctype.h>

#include <string.h>

class CON_VAT

{

protected:

char *ten;

public:

CON_VAT()

{

ten = NULL;

345                         346                                 

}

CON_VAT(char *ten1)

{

ten = strdup(ten1);

}

virtual void xung_ten()

{

}

} ;

class CON_MEO:public CON_VAT

{

public:

CON_MEO() : CON_VAT()

{

}

CON_MEO(char *ten1) : CON_VAT(ten1)

{

}

virtual void xung_ten()

{

cout << "

Toi la chu meo: " << ten ;

}

};

class CON_CHO:public CON_VAT

{

public:

CON_CHO() : CON_VAT()

{

}

CON_CHO(char *ten1) : CON_VAT(ten1)

{

}

virtual void xung_ten()

{

cout << "

Toi la chu cho: " << ten ;

}

};

class DS_CON_VAT // Danh sach con vat

{

private:

int max_so_con_vat;

int so_con_vat;

CON_VAT **h ;

public:

DS_CON_VAT(int max);

~DS_CON_VAT();

int nhap(CON_VAT *c);

CON_VAT* xuat(int n);

void thong_ke();

} ;

DS_CON_VAT::DS_CON_VAT(int max)

{

max_so_con_vat = max;

so_con_vat = 0;

h = new CON_VAT*[max];

for (int i=0; i<max; ++i)

h[i] = NULL;

347                         348                                 

}

DS_CON_VAT::~DS_CON_VAT()

{

max_so_con_vat = 0;

so_con_vat = 0;

delete h;

}

int DS_CON_VAT::nhap(CON_VAT *c)

{

if (so_con_vat==max_so_con_vat)

return 0;

int i=0;

while (h[i]!=NULL) ++i;

h[i]=c;

so_con_vat++ ;

return (i+1);

}

CON_VAT* DS_CON_VAT::xuat(int n)

{

if (n<1 || n > max_so_con_vat)

return NULL ;

--n ;

if (h[n])

{

CON_VAT *c = h[n];

h[n]=NULL;

so_con_vat-- ;

return c;

}

else

return NULL;

}

void DS_CON_VAT::thong_ke()

{

if (so_con_vat)

{

cout << "

" ;

for (int i=0; i<max_so_con_vat; ++i)

if (h[i])

h[i]->xung_ten();

}

}

CON_CHO  c1("MUC");

CON_CHO  c2("VEN");

CON_CHO  c3("LAI");

CON_CHO  c4("NHAT");

CON_CHO  c5("BONG");

CON_MEO  m1("MUOP");

CON_MEO  m2("DEN");

CON_MEO  m3("TRANG");

CON_MEO  m4("TAM THE");

CON_MEO  m5("VANG");

void main()

{

DS_CON_VAT d(20);

clrscr();

d.nhap(&c1);

int im2 = d.nhap(&m2);

349                         350                                 

d.nhap(&c3);

d.nhap(&m1);

int ic4 = d.nhap(&c4);

d.nhap(&c5);

d.nhap(&m5);

d.nhap(&c2);

d.nhap(&m3);

d.thong_ke();

d.xuat(im2);

d.xuat(ic4);

d.thong_ke();

getch();

}

Chú ý: Theo quan điểm chung về cách thức sử dụng, thì lớp CON_VAT là lớp cơ sở trừu tượng. Tuy nhiên theo quan điểm của C++ thì lớp này chưa phải là lớp cơ sở trừu tượng, vì trong lớp không có các phương thức thuần tuý ảo. Phương thức xung_ten:

virtual void xung_ten()

{

}

là phương thức ảo, được định nghĩa đầy đủ , mặc dù thân của nó là rỗng.

Do vậy khai báo:

CON_VAT   cv(“Con vat chung”);

vẫn được C++ chấp nhận.

Bây giờ nếu định nghĩa lại phương thức xung_ten như sau:

virtual void xung_ten() = 0 ;

thì nó trở thành phương thức thuần ảo và C++ sẽ quan niệm lớp CON_VAT là lớp trừu tượng. Khi đó câu lệnh khai báo:

CON_VAT   cv(“Con vat chung”);

sẽ bị C++ bắt lỗi với thông báo:

Cannot  create instance of abstruct class ‘CON_VAT’

§ 6. Sử dụng tương ứng bội và phương thức ảo

6.1. Chiến lược sử dụng tương ứng bội

Tương ứng bội cho phép xét các vấn đề khác nhau, các đối tượng khác nhau, các phương pháp khác nhau, các cách giải quyết khác nhau theo cùng một lược đồ chung.

Các bước áp dụng tương ứng bội có thể tổng kết lại như sau:

1. Xây dựng lớp cơ sở trừu tượng bao gồm những thuộc tính chung nhất của các thực thể cần quản lý. Đưa vào các phương thức ảo hay thuần ảo dùng để xây dựng các nhóm phương thức ảo cho các lớp dẫn xuất sau này. Mỗi nhóm phương thức ảo sẽ thực hiện một chức năng nào đó trên các lớp.

2. Xây dựng các lớp dẫn xuất bắt đầu từ lớp cơ sở ảo. Số mức dẫn xuất là không hạn chế. Các lớp dẫn xuất sẽ mô tả các đối tượng cụ thể cần quản lý.

3. Xây dựng các phương thức ảo trong các dẫn xuất. Các phương thức này tạo thành các nhóm phương thức ảo trong sơ đồ các lớp có quan hệ thừa kế.

4. Xây dựng lớp quản lý các đối tượng. Dữ liệu của lớp này là một dẫy con trỏ của lớp cơ sở trừu tượng ban đầu. Các con trỏ này có thể chứa địa chỉ đối tượng của các lớp dẫn xuất. Do vậy  có thể dùng các con trỏ này để thực hiện các thao tác trên các đối tượng của bất kỳ lớp dẫn xuất nào.

6.2. Ví dụ

Chương trình quản lý các con vật trong §5 là một ví dụ về cách sử dụng tương ứng bội. Dưới đây là một ví dụ khác. Giả sử có 4 hình vẽ: Đoạn  thẳng, hình tròn, hình chữ nhật và hình vuông. Bốn hình cho hiện thẳng hàng trên màn hình tạo thành một bức tranh. Nếu thay đổi thứ tự các hình sẽ nhận được các bức tranh khác nhau. Chương trình dưới đây sẽ cho hiện tất cả các bức tranh khác nhau. Chương trình được tổ chức theo các bước nêu trong 6.1:

351                         352                                 

+ Lớp cơ sở trừu tượng là lớp HINH (hình) gồm một thuộc tính mau (mầu) và một phương thức ảo thuần tuý:

virtual void draw(int x, int y) = 0 ;

+ Các lớp dẫn xuất trực tiếp từ lớp hình là : DTHANG , HTRON và CHUNHAT.

+ Lớp VUONG dẫn xuất từ lớp CHUNHAT.

+ Lớp quản lý chung là lớp picture có thuộc tính là một mảng con trỏ kiểu HINH gồm 4 phần tử dùng để chứa địa chỉ 4 đối tượng: DTHANG, HTRON, CHUNHAT và VUONG. Sử dụng phương thức draw gọi từ 4 phần tử mảng nói trên sẽ nhận được một bức tranh. Bằng cách hoán vị các phần tử này, sẽ nhận được tất cả các bức tranh khác nhau.

//CT6-05

// Lop co so truu tuong

// Lop hinh hoc

#include <conio.h>

#include <graphics.h>

class HINH

{

private:

int mau;

public:

HINH(int m)

{

mau = m;

}

getmau()

{

return mau;

}

virtual void draw(int x, int y) = 0;

};

class DTHANG : public HINH

{

private:

int  dodai;

public:

DTHANG(int d, int m):HINH(m)

{

dodai = d ;

}

virtual void draw(int x, int y)

{

setcolor(getmau()) ;

line(x,y,x+dodai,y);

}

};

class CHUNHAT: public HINH

{

private:

int rong, cao;

public:

CHUNHAT(int r, int c, int m):HINH(m)

{

rong = r; cao = c;

}

virtual void draw(int x, int y )

{

setcolor(getmau()) ;

rectangle(x,y,x+rong,y+cao);

setfillstyle(1,getmau());

353                         354                                 

floodfill(x+rong/2,y+cao/2, getmau() );

}

};

class VUONG : public CHUNHAT

{

public:

VUONG(int a, int m): CHUNHAT(a,a,m)

{

}

};

class HTRON: public HINH

{

private:

int bk; //Ban kinh

public:

HTRON(int bk1, int m):HINH(m)

{

bk = bk1;

}

virtual void draw(int x, int y)

{

setcolor(getmau()) ;

circle(x+bk,y+bk,bk);

setfillstyle(1,getmau());

floodfill(x + bk, y + bk,getmau());

}

};

class picture

{

private:

HINH *h[4];

public:

picture(HINH *h0,HINH *h1,HINH *h2,HINH *h3)

{

h[0]=h0;

h[1]=h1;

h[2]=h2;

h[3]=h3;

}

void paint(int *k);

void listpaint();

} ;

void picture::paint(int *k)

{

for (int i=0; i<4; ++i)

h[k[i]]->draw(10+i*150, 200);

}

void picture::listpaint()

{

int k[4],i1,i2,i3,i4;

for (i1=0;i1<4;++i1)

for (i2=0;i2<4;++i2)

if (i2!=i1)

for (i3=0;i3<4;++i3)

if (i3!=i2 && i3!=i1)

for (i4=0;i4<4;++i4)

if (i4!=i3 && i4!=i2 && i4!=i1)

{

k[0]=i1;k[1]=i2;

355                         356                                 

k[2]=i3;k[3]=i4;

paint(k);

getch();

cleardevice();

}

}

DTHANG dt(120,14);

HTRON ht(60,RED);

CHUNHAT cn(120,100,MAGENTA);

VUONG v(120,CYAN);

} ;

void main()

{

int mh=0,mode=0;

initgraph(&mh,&mode,"");

picture pic(&dt,&ht,&cn,&v);

pic.listpaint();

getch();

closegraph();

}

§ 7. Xử lý các thuật toán khác nhau

Có thể sử dụng tương ứng bội để tổ chức thực hiện các thuật toán khác nhau trên cùng một bài toán như sau:

+ Lớp cơ sở trừu tượng sẽ chứa dữ liệu bài toán và một phương thức ảo.

+ Mỗi lớp dẫn xuất ứng với một thuật toán cụ thể. Phương thức ảo của lớp dẫn xuất sẽ thực hiện một thuật toán cụ thể.

+ Sử dụng một mảng con trỏ của lớp cơ sở và gán cho mỗi phần tử mảng  địa chỉ của  một đối tượng của lớp dẫn xuất. Sau đó dùng các phần tử mảng con trỏ để gọi tới các phương thức ảo. Bằng cách đó sẽ thực hiện cùng một bài toán theo các thuật toán khác nhau và dễ dàng so sánh hiêụ quả của các thuật toán.

Ví dụ sau minh hoạ việc thực hiện bài toán sắp xếp dẫy số nguyên theo thứ tự tăng bằng cách dùng đồng thời 3 thuật toán: Thuật toán lựa chọn (Select_Sort), thuật toán sắp xếp nhanh (Quick_Sort) và thuật toán vun đống (Heap_Sort). Chương trình gồm 4 lớp:

+ Lớp cơ sở trừu tượng:

class sort

{

protected:

int *a;

void hoan_vi(long i, long j) ; 

public:

virtual void sapxep(int *a1, long n) ;

} ;

Lớp này gồm:

- Một thành phần dữ liệu là con trỏ a trỏ tới một vùng nhớ chứa dẫy số nguyên cần sắp xếp.

- Phương thức hoan_vi(i,j) dùng để hoán vị các phần tử a[i] và a[j]. Phương thức này được dùng trong 3 lớp dẫn xuất bên dưới.

- Phương thức ảo sapxep(a1,n) dùng để sắp xếp dẫy n số nguyên chứa trong mảng a1.

+ Ba lớp dẫn xuất là: SELECT_SORT, QUICK_SORT và HEAP_SORT. Mỗi lớp đều có phương thức ảo:

virtual void sapxep(int *a1, long n) ;

để thực hiện hiện việc sắp xếp theo theo một thuật toán cụ thể.

+ Trong hàm main() sẽ tạo ra một dẫy 30000 số nguyên một cách ngẫu nhiên, sau đó lần lượt sử dụng 3 thuật toán sắp xếp để so sánh. Kết quả như sau:

357                         358                                 

Thời gian sắp xếp theo thuật toán Select sort là: 19.20 giây

Thời gian sắp xếp theo thuật toán Quick sort là: 0.11 giây

Thời gian sắp xếp theo thuật toán  Heap  sort là: 0.44 giây

Nội dung chương trình như sau:

//CT6-06

// Lop co so truu tuong

// Lop sort

#include <conio.h>

#include <stdio.h>

#include <time.h>

#include <stdlib.h>

#include <iostream.h>

#include <dos.h>

class sort

{

protected:

int *a;

void hoan_vi(long i, long j)

{

int tg = a[i];

a[i] = a[j];

a[j] = tg;

}

public:

virtual void sapxep(int *a1, long n)

{

a = a1;

}

} ;

class select_sort : public sort

{

public:

virtual void sapxep(int *a1, long n) ;

} ;

void select_sort::sapxep(int *a1, long n)

{

long i,j,r;

sort::sapxep(a1,n);

for (i=1; i<n; ++i)

{

r=i;

for (j=i+1; j<=n; ++j)

if(a[j] < a[r]) r = j;

if(r!=i) hoan_vi(i,r);

}

}

class quick_sort : public sort

{

private:

void q_sort(long l, long r);

public:

virtual void sapxep(int *a1, long n) ;

} ;

void quick_sort::q_sort(long l, long r)

{

int x;

long i,j;

if (l < r)

359                         360                                 

{

x = a[l]; i = l; j = r+1;

do

{

++i;  --j;

while (i<r && a[i] < x) ++i ;

while (a[j] > x) --j ;

if (i<j) hoan_vi(i,j);

} while (i<j);

hoan_vi(l,j);

q_sort(l,j-1);

q_sort(j+1,r);

}

}

void quick_sort::sapxep(int *a1, long n)

{

sort::sapxep(a1,n);

q_sort(1,n);

}

class heap_sort : public sort

{

private:

void shift(long i, long n);

public:

virtual void sapxep(int *a1, long n) ;

} ;

void heap_sort::shift(long i, long n)

{

long l,r,k;

l = 2*i; r = l+1;

if (l>n) return;

if (l==n)

{

if (a[i]<a[l]) hoan_vi(i,l);

return;

}

if (a[l] > a[r])

k = l;

else

k = r;

if (a[i]>=a[k])

return;

else

{

hoan_vi(i,k);

shift(k,n);

}

}

void heap_sort::sapxep(int *a1, long n)

{

long i;

sort::sapxep(a1,n);

/* Tao dong */

for (i=n/2 ; i>=1; --i) shift(i,n);

/* Lap */

for (i=n ; i>=2; --i)

{

hoan_vi(1,i);

361                         362                                 

shift(1,i-1);

}

}

void main()

{

long i,n;

struct time t1,t2;

int *a, k, tg, sec, hund;

n=30000;

a=(int*) malloc((n+1)*sizeof(int));

if (a==NULL)

{

puts("

Loi BN");

getch();

exit(0);

}

sort *s[3];

select_sort ss;

quick_sort  qs;

heap_sort   hs;

s[0]=&ss; s[1]=&qs; s[2]=&hs;

clrscr();

for (k=0; k<3; ++k)

{

srand(5000);

for (i=1;i<=n;++i)

a[i]=rand();

gettime(&t1);

s[k]->sapxep(a,n);

gettime(&t2);

tg =  (t2.ti_sec - t1.ti_sec)*100 + t2.ti_hund - t1.ti_hund ;

sec = tg / 100;

hund = tg % 100;

printf("

Sap xep %d  %d  %d %d  %d",k+1,

t2.ti_sec,t2.ti_hund,t1.ti_sec,t1.ti_hund);

printf("

Sap xep %d  Thoi gian %d sec %d hund",

k+1,sec,hund);

}

getch();

}

363                                                                

Bạn đang đọc truyện trên: AzTruyen.Top

Tags: #chet7h