c++chuong3
Chương 3
Khái niệm về lớp
Như đã nói ở trên, lớp là khái niệm trung tâm của lập trình hướng đối tượng, nó là sự mở rộng của các khái niệm cấu trúc (struct) của C và bản ghi (record) của PASCAL. Ngoài các thành phần dữ liệu (như cấu trúc), lớp còn chứa các thành phần hàm , còn gọi là phương thức (method) hay hàm thành viên (member function). Cũng giống như cấu trúc, lớp có thể xem như một kiểu dữ liệu. Vì vậy lớp còn gọi là kiểu đối tượng và lớp được dùng để khai báo các biến, mảng đối tượng (như thể dùng kiểu int để khai báo các biến mảng nguyên). Như vậy từ một lớp có thể tạo ra (bằng cách khai báo) nhiều đối tượng (biến, mảng) khác nhau. Mỗi đối tượng có vùng nhớ riêng của mình. Vì vậy cũng có thể quan niệm lớp là tập hợp các đối tượng cùng kiểu.
Chương này sẽ trình bầy cách định nghĩa lớp, cách xây dựng phương thức, giải thích về phạm vi truy nhập, sư dụng các thành phần của lớp, cách khai báo biến, mảng cấu trúc, lời gọi tới các phương thức.
§ 1. Định nghĩa lớp
1. Lớp được định nghĩa theo mẫu:
class tên_lớp
{
// Khai báo các thành phần dữ liệu (thuộc tính)
// Khai báo các phương thức
} ;
// Định nghĩa (xây dựng) các phương thức
Chú ý:
93 94
Thuộc tính của lớp có thể là các biến, mảng, con trỏ có kiểu chuẩn (int, float, char, char*, long,...) hoặc kiểu ngoài chuẩn đã định nghĩa trước (cấu trúc, hợp, lớp, ...) . Thuộc tính của lớp không thể có kiểu của chính lớp đó, nhưng có thể là kiểu con trỏ lớp này, ví dụ:
class A
{
A x ; // Không cho phép, vì x có kiểu lớp A
A *p ; // Cho phép , vì p là con trỏ kiểu lớp A
...
} ;
2. Khi báo các thành phần của lớp (thuộc tính và phương thức) có thể dùng các từ khoá private và public để quy định phạm vi sử dụng của các thành phần. Nếu không quy định cụ thể (không dùng các từ khoá private và public) thì C++ hiểu đó là private.
Các thành phần private (riêng) chỉ được sử dụng bên trong lớp (trong thân của các phương thức của lớp). Các hàm không phải là phương thức của lớp không được phép sử dụng các thành phần này.
Các thành phần public (công cộng) được phép sử dụng ở cả bên trong và bên ngoài lớp.
3. Các thành phần dữ liệu thường (nhưng không bắt buộc) khai báo là private để bảo đảm tính giấu kín, bảo vệ an toàn dữ liệu của lớp, không cho phép các hàm bên ngoài xâm nhập vào dữ liệu của lớp.
4. Các phương thức thường khai báo là public để chúng có thể được gọi tới (sử dụng) từ các hàm khác trong chương trình.
5. Các phương thức có thể được xây dựng bên ngoài hoặc bên trong định nghĩa lớp. Thông thường, các phương thức ngắn được viết bên trong định nghĩa lớp, còn các phương thức dài thì viết bên ngoài định nghĩa lớp.
6. Trong thân phương thức của một lớp (giả sử lớp A) có thể sử dụng:
+ Các thuộc tính của lớp A
+ Các phương thức của lớp A
+ Các hàm tự lập trong chương trình. Vì phạm vi sử dụng của hàm là toàn chương trình.
7. Giá trị trả về của phương thức có thể có kiểu bất kỳ (chuẩn và ngoài chuẩn)
Ví dụ sau sẽ minh hoạ các điều nói trên. Chúng ta sẽ định nghĩa lớp để mô tả và xử lý các điểm trên màn hình đồ hoạ. Lớp được đăt tên là DIEM.
+ Các thuộc tính của lớp gồm:
int x ; // hoành độ (cột)
int y ; // tung độ (hàng)
int m ; // mầu
+ Các phương thức:
Nhập dữ liệu một điểm
Hiển thị một điểm
ẩn một điểm
Lớp điểm được xây dựng như sau:
class DIEM
{
private:
int x, y, m ;
public:
void nhapsl() ;
void hien() ;
void an()
{
putpixel(x, y, getbkcolor());
}
} ;
void DIEM::nhap()
{
95 96
cout << “
Nhập hoành độ (cột) và tung độ (hàng) của điểm: “
cin >> x >> y ;
cout << “
Nhập mã mầu của điểm: “
cin >> m ;
}
void DIEM::hien()
{
int mau_ht ;
mau_ht = getcolor();
putpixel(x, y, m);
setcolor(mau_ht);
}
Qua ví dụ trên có thể rút ra một số điều cần nhớ sau:
+ Trong cả 3 phương thức (dù viết trong hay viết ngoài định nghĩa lớp) đều được phép truy nhập đến các thuộc tính x, y và m của lớp.
+ Các phương thức viết bên trong định nghĩa lớp (như phương thức an() ) được viết như một hàm thông thường.
+ Khi xây dựng các phương thức bên ngoài lớp, cần dùng thêm tên lớp và toán tử phạm vi :: đặt ngay trước tên phương phức để quy định rõ đây là phương thức của lớp nào.
§ 2. Biến, mảng đối tượng
Như đã nói ở trên, một lớp (sau khi định nghĩa) có thể xem như một kiểu đối tượng và có thể dùng để khai báo các biến, mảng đối tượng. Cách khai báo biến, mảng đối tượng cũng giống như khai báo biến, mảng các kiểu khác (như int, float, cấu trúc, hợp, ...), theo mẫu sau:
Tên_lớp danh sách đối ;
Tên_lớp danh sách mảng ;
Ví dụ sử dụng lớp DIEM ở §1, có thể khai báo các biến, mảng DIEM như sau:
DIEM d1, d2, d3 ; // Khai báo 3 biến đối tượng d1, d2, d3
DIEM d[20] ; // Khai báo mảng đối tượng d gồm 20 phần tử
Mỗi đối tượng sau khi khai báo sẽ được cấp phát một vùng nhớ riêng để chứa các thuộc tính của chúng. Chú ý rằng sẽ không có vùng nhớ riêng để chứa các phương thức cho mỗi đối tượng. Các phương thức sẽ được sử dụng chung cho tất cả các đối tượng cùng lớp. Như vậy về bộ nhớ được cấp phát thì đối tượng giống cấu trúc. Trong trương hợp này:
sizeof(d1) = sizeof(d2) = sizeof(d3) = 3*sizeof(int) = 6
sizeof(d) = 20*6 = 120
Thuộc tính của đối tượng:
Trong ví dụ trên, mỗi đối tượng d1, d2, d3 và mỗi phần tử d[i] đều có 3 thuộc tính là x, y, m. Chú ý là mỗi thuộc đều thuộc về một đối tượng, vì vậy không thể viết tên thuộc một cách riêng rẽ mà bao giờ cũng phải có tên đối tượng đi kèm, giống như cách viết trong cấu trúc của C hay bản ghi của PASCAL. Nói cách khác, cách viết thuộc tính của đối tượng như sau:
tên_đối_tượng.Tên_thuộc_tính
Với các đối tượng d1, d2, d3 và mảng d, có thể viết như sau:
d1.x // Thuộc tính x của đối tượng d1
d2.x // Thuộc tính x của đối tượng d2
d3.y // Thuộc tính y của đối tượng d3
d[2].m // Thuộc tính m của phần tử d[2]
d1.x = 100 ; // Gán 100 cho d1.x
d2.y = d1.x; // Gán d1.x cho d2.y
Sử dụng các phương thức
Cũng giống như hàm, một phương thức được sử dụng thông qua lời gọi. Tuy nhiên trong lời gọi phương thức bao giờ cũng phải có tên đối tượng để chỉ rõ phương thức thực hiện trên các thuộc tính của đối tượng nào. Ví dụ lời gọi:
97 98
d1.nhapsl();
sẽ thực hiện nhập số liệu vào các thành phần d1.x, d1.y và d1.m
Câu lệnh
d[3].nhapsl() ;
sẽ thực hiện nhập số liệu vào các thành phần d[3].x, d[3].y và d[3].m
Chúng ta sẽ minh hoạ các điều nói trên bằng một chương trình đơn giản sử dụng lớp DIEM để nhập 3 điểm, hiện rồi ẩn các điểm vừa nhập. Trong chương trình đưa vào hàm kd_do_hoa() dùng để khởi động hệ đồ hoạ.
#include <conio.h>
#include <iostream.h>
#include <graphics.h>
class DIEM
{
private:
int x, y, m ;
public:
void nhapsl();
void an()
{
putpixel(x,y,getbkcolor());
}
void hien();
};
void DIEM::nhapsl()
{
cout << "
Nhap hoanh do (cot) va tung do (hang) cua diem: " ;
cin >> x >> y ;
cout << "
Nhap ma mau cua diem: " ;
cin >> m ;
}
void DIEM::hien()
{
int mau_ht;
mau_ht = getcolor() ;
putpixel(x,y,m);
setcolor(mau_ht);
}
void kd_do_hoa()
{
int mh, mode ;
mh=mode=0;
initgraph(&mh, &mode, "");
}
void main()
{
DIEM d1, d2, d3 ;
d1.nhapsl();
d2.nhapsl();
d3.nhapsl();
kd_do_hoa();
setbkcolor(BLACK);
d1.hien();
d2.hien();
d3.hien();
getch();
d1.an();
99 100
d2.an();
d3.an();
getch();
closegraph();
}
§ 3. Con trỏ đối tượng
Con trỏ đối tượng dùng để chứa địa chỉ của biến, mảng đối tượng. Nó được khai báo như sau:
Tên_lớp *con trỏ ;
Ví dụ dùng lớp DIEM có thể khai báo:
DIEM *p1 , *p2, *p3 ; // khai báo 3 con trỏ p1, p2, p3
DIEM d1, d2 ; // Khai báo 2 đối tượng d1, d2
DIEM d[20] ; // Khai báo mảng đối tượng
và có thể thực hiện các câu lệnh:
p1 = &d2 ; // p1 chứa địa chỉ của d2 , hay p1 trỏ tới d2
p2 = d ; // p2 trỏ tới đầu mảng d
p3 = new DIEM // Tạo một đối tượng và chứa địa chỉ của nó
// vào p3
Để sử dụng thuộc tính của đối tượng thông qua con trỏ, ta viết như sau:
Tên_con_trỏ->Tên_thuộc_tính
Chú ý: Nếu con trỏ chứa địa chỉ đầu của mảng, có thể dùng con trỏ như tên mảng.
Như vậy sau khi thực hiện các câu lệnh trên thì:
p1->x và d2.x là như nhau
p2[i].y và d[i].y là như nhau
Tóm lại ta có quy tắc sau
Quy tắc sử dụng thuộc tính: Để sử dụng một thuộc tính của đối tượng ta phải dùng phép . hoặc phép -> . Trong chương trình, không cho phép viết tên thuộc tính một cách đơn độc mà phải đi kèm tên đối tượng hoặc tên con trỏ theo các mẫu sau:
Tên_đối_tượng.Tên_thuộc_tính
Tên_con_trỏ->Tên_thuộc_tính
Tên_mảng_đối_tượng[chỉ_số].Tên_thuộc_tính
Tên_con_trỏ[chỉ_số].Tên_thuộc_tính
Chương trình dưới đây cũng sử dụng lớp DIEM (trong §1) để nhập một dẫy điểm, hiển thị và ẩn các điểm vừa nhập. Chương trình dùng một con trỏ kiểu DIEM và dùng toán tử new để tạo ra một dẫy đối tượng.
#include <conio.h>
#include <iostream.h>
#include <graphics.h>
class DIEM
{
private:
int x, y, m ;
public:
void nhapsl();
void an()
{
putpixel(x,y,getbkcolor());
}
void hien();
};
void DIEM::nhapsl()
{
cout <<"
Nhap hoanh do (cot) va tung do (hang) cua diem:" ;
101 102
cin >> x >> y ;
cout << "
Nhap ma mau cua diem: " ;
cin >> m ;
}
void DIEM::hien()
{
int mau_ht;
mau_ht = getcolor() ;
putpixel(x,y,m);
setcolor(mau_ht);
}
void kd_do_hoa()
{
int mh, mode ;
mh=mode=0;
initgraph(&mh, &mode, "");
}
void main()
{
DIEM *p;
int i, n;
cout << "So diem: " ;
cin >> n;
p = new DIEM[n+1];
for (i=1; i<=n; ++i)
p[i].nhapsl();
kd_do_hoa();
for (i=1; i<=n; ++i)
p[i].hien();
getch();
for (i=1; i<=n; ++i)
p[i].an();
getch();
closegraph();
}
§ 4. Đối của phương thức, con trỏ this
4.1. Con trỏ this là đối thứ nhất của phương thức
Chúng ta hãy xem lại phương thức nhapsl của lớp DIEM
void DIEM::nhapsl()
{
cout <<"
Nhap hoanh do (cot) va tung do (hang) cua diem:" ;
cin >> x >> y ;
cout << "
Nhap ma mau cua diem: " ;
cin >> m ;
}
Rõ ràng trong phương thức này chúng ta sử dụng tên các thuộc tính x, y và m một cách đơn độc. Điều này có vẻ như mâu thuẫn với quy tắc sử dụng thuộc tính nêu trong mục trước. Song sự thể như sau:
C++ sử dụng con trỏ đặc biệt this trong các phương thức. Các thuộc tính viết trong phương thức được hiểu là thuộc một đối tượng do con trỏ this trỏ tới. Như vậy phương thức nhapsl() có thể viết một cách tường minh như sau:
void DIEM::nhapsl()
{
cout << "
Nhap hoanh do (cot) va tung do (hang) cua diem:" ;
cin >> this->x >> this->y ;
103 104
cout << "
Nhap ma mau cua diem: " ;
cin >> this->m ;
}
Từ góc độ hàm số có thể kết luận rằng: Phương thức bao giờ cũng có ít nhất một đối là con trỏ this và nó luôn luôn là đối đầu tiên của phương thức.
4.2. Tham số ứng với đối con trỏ this
Xét một lời gọi tới phương thức nhapsl() :
DIEM d1;
d1.nhapsl() ;
Trong trường hợp này tham số truyền cho con trỏ this chính là địa chỉ của d1:
this = &d1
Do đó:
this->x chính là d1.x
this->y chính là d1.y
this->m chính là d1.m
Như vậy câu lệnh
d1.nhapsl() ;
sẽ nhập dữ liệu cho các thuộc tính của đối tượng d1. Từ đó có thể rút ra kết luận sau:
Tham số truyền cho đối con trỏ this chính là địa chỉ của đối tượng đi kèm với phương thức trong lời gọi phương thức.
4.3. Các đối khác của phương thức
Ngoài đối đặc biệt this (đối này không xuất hiện một cách tường minh), phương thức còn có các đối khác được khai báo như trong các hàm. Đối của phương thức có thể có kiểu bất kỳ (chuẩn và ngoài chuẩn).
Ví dụ để xây dựng phương thức vẽ đường thẳng qua 2 điểm ta cần đưa vào 3 đối: Hai đối là 2 biến kiểu DIEM, đối thứ ba kiểu nguyên xác định mã mầu. Vì đã có đối ngầm định this là đối thứ nhất, nên chỉ cần khai báo thêm 2 đối. Phương thức có thể viết như sau:
void DIEM::doan_thang(DIEM d2, int mau)
{
int mau_ht;
mau_ht = getcolor();
setcolor(mau);
line(this->x,this->y,d2.x,d2.y);
setcolor(mau_ht);
}
Chương trình sau minh hoạ các phương thức có nhiều đối. Ta vẫn dùng lớp DIEM nhưng có một số thay đổi:
+ Bỏ thuộc tính m (mầu)
+ Bỏ các phương thức hien và an
+Đưa vào 4 phương thức mới:
ve_ doan_thang (Vẽ đoạn thẳng qua 2 điểm)
ve_tam_giac (Vẽ tam giác qua 3 điểm)
do_dai (Tính độ dài của đoạn thẳng qua 2 điểm)
chu_vi (Tính chu vi tam giác qua 3 điểm)
Chương trình còn minh hoạ:
+ Việc phương thức này sử dụng phương thức khác (phương thức ve_tam_giac sử dụng phương thức ve_doan_thang, phương thức chu_vi sử dụng phương thức do_dai)
+ Sử dụng con trỏ this trong thân các phương thức ve_tam_giac và chu_vi
Nội dung chương trình là nhập 3 điểm, vẽ tam giác có đỉnh là 3 điểm vừa nhập sau đó tính chu vi tam giác.
#include <conio.h>
105 106
#include <iostream.h>
#include <graphics.h>
#include <math.h>
#include <stdio.h>
class DIEM
{
private:
int x, y ;
public:
void nhapsl();
void ve_doan_thang(DIEM d2, int mau) ;
void ve_tam_giac(DIEM d2, DIEM d3,int mau) ;
double do_dai(DIEM d2)
{
DIEM d1 = *this ;
return sqrt( pow(d1.x - d2.x,2) + pow(d1.y - d2.y,2) ) ;
}
double chu_vi(DIEM d2, DIEM d3);
};
void DIEM::nhapsl()
{
cout <<"
Nhap hoanh do (cot) va tung do (hang) cua diem:" ;
cin >> x >> y ;
}
void kd_do_hoa()
{
int mh, mode ;
mh=mode=0;
initgraph(&mh, &mode, "");
}
void DIEM::ve_doan_thang(DIEM d2, int mau)
{
setcolor(mau);
line(this->x,this->y,d2.x,d2.y);
}
void DIEM::ve_tam_giac(DIEM d2, DIEM d3,int mau)
{
(*this).ve_doan_thang(d2,mau);
d2.ve_doan_thang(d3,mau);
d3.ve_doan_thang(*this,mau);
}
double DIEM::chu_vi(DIEM d2, DIEM d3)
{
double s;
s= (*this).do_dai(d2) + d2.do_dai(d3) + d3.do_dai(*this) ;
return s;
}
void main()
{
DIEM d1, d2, d3;
char tb_cv[20] ;
d1.nhapsl();
d2.nhapsl();
d3.nhapsl();
kd_do_hoa();
d1.ve_tam_giac(d2,d3,15);
double s = d1.chu_vi(d2,d3);
sprintf(tb_cv,"Chu vi = %0.2f", s);
outtextxy(10,10,tb_cv);
getch();
closegraph();
107 108
}
Một số nhận xét về đối của phương thức và lời gọi phương thức
+ Quan sát nguyên mẫu phương thức:
void ve_doan_thang(DIEM d2, int mau) ;
sẽ thấy phương thức có 3 đối:
Đối thứ nhât là một đối tượng DIEM do this trỏ tới
Đối thứ hai là đối tượng DIEM d2
Đối thứ ba là biến nguyên mau
Nội dung phương thức là vẽ một đoạn thẳng đi qua các điểm *this và d2 theo mã mầu mau. Xem thân của phương sẽ thấy được nội dung này:
void DIEM::ve_doan_thang(DIEM d2, int mau)
{
setcolor(mau);
line(this->x,this->y,d2.x,d2.y);
}
Tuy nhiên trong trương hợp này, vai trò của this không cao lắm, vì nó được đưa vào chỉ cốt làm rõ đối thứ nhất. Trong thân phương thức có thể bỏ từ khoá this vẫn được.
+ Vai trò của this trở nên quan trọng trong phương thức ve_tam_giac:
void ve_tam_giac(DIEM d2, DIEM d3,int mau) ;
Phương thức này có 4 đối là:
this trỏ tới một đối tượng kiểu DIEM
d2 một đối tượng kiểu DIEM
d3 một đối tượng kiểu DIEM
mau một biến nguyên
Nội dung phương thức là vẽ 3 cạnh:
cạnh 1 đi qua *this và d2
cạnh 2 đi qua d2 và d3
cạnh 3 đi qua d3 và *this
Các cạnh trên được vẽ nhờ sử dụng phương thức ve_doan_thang:
Vẽ cạnh 1 dùng lệnh: (*this).ve_doan_thang(d2,mau) ;
Vẽ cạnh 2 dùng lệnh: d2.ve_doan_thang(d3,mau);
Vẽ cạnh 3 dùng lệnh: d3.ve_doan_thang(*this,mau);
Trong trường này rõ ràng vai trò của this rất quan trọng. Nếu không dùng nó thì công việc trơ nên khó khăn, dài dòng và khó hiểu hơn. Chúng ta hãy so sánh 2 phương án:
Phương án dùng this trong phương thức ve_tam_giac:
void DIEM::ve_tam_giac(DIEM d2, DIEM d3,int mau)
{
(*this).ve_doan_thang(d2,mau);
d2.ve_doan_thang(d3,mau);
d3.ve_doan_thang(*this,mau);
}
Phương án không dùng this trong phương thức ve_tam_giac:
void DIEM::ve_tam_giac(DIEM d2, DIEM d3,int mau)
{
DIEM d1;
d1.x = x;
d1.y = y;
d1.ve_doan_thang(d2,mau);
d2.ve_doan_thang(d3,mau);
d3.ve_doan_thang(d1,mau);
109 110
}
§ 5. Nói thêm về kiểu phương thức và kiểu đối của phương thức
5.1. Kiểu phương thức
Phương thức có thể không có giá trị trả về (kiểu void) hoặc có thể trả về một giá trị có kiểu bất kỳ, kể cả giá trị kiểu đối tượng, con trỏ đối tượng, tham chiếu đối tượng.
5.2. Đối của phương thức
Đối của phương thức (cũng giống như đối của hàm) có thể có kiểu bất kỳ:
+ Kiểu dữ liệu chuẩn như int, float, char,... . Con trỏ hoặc tham chiếu đến kiểu dữ liệu chuẩn như int*, float*, char*, int&, float&, char&,...
+ Các kiểu ngoài chuẩn đã định nghĩa trước như đối tượng, cấu trúc, hợp, enum,... . Con trỏ hoặc tham chiếu đến các kiểu ngoài chuẩn này.
+ Kiểu đối tượng của chính phương thức, con trỏ hoặc tham chiếu đến kiểu đối tượng này.
5.3. Các ví dụ
Ví dụ 1 minh hoạ:
+ Thuộc tính (thành phần dữ liệu) của lớp có thể là đối tượng của lớp khác đã định nghĩa bên trên.
+ Phương thức có giá trị trả về kiểu đối tượng và con trỏ đối tượng.
Nội dung chương trình là nhập một dẫy hình chữ nhật, sau đó tìm hình chữ nhật có max diện tích và hình chữ nhật có max chu vi.
Chương trình được tổ chức thành 2 lớp:
+ Lớp HINH_CN gồm:
- Các thuộc tính: d và r (chiều dài và chiều rộng)
- Các phương thức
void nhapsl() ; // Nhập chiều dài, rộng
int dien_tich(); // Tính diện tích
int chu_vi() ; // Tính chu vi
+ Lớp DAY_HINH_CN gồm
- Các thuộc tính:
int n ; //số hình chữ nhật của dẫy
HINH_CN *h; //Con trỏ tới dẫy đối tượng của lớp HINH_CN
- Các phương thức
void nhapsl(); // Nhập một dẫy hình chữ nhật
HINH_CN hinh_dt_max() ; //Trả về hình chữ nhật có
// diện tích max
HINH_CN *hinh_cv_max() ; // Trả về con trỏ tới HCN có
// chu vi max
#include <conio.h>
#include <iostream.h>
class HINH_CN
{
private:
int d, r; // chieu dai va chieu rong
public:
void nhapsl()
{
cout << "
Nhap chieu dai va chieu rong: " ;
cin >> d >> r ;
}
void in()
{
cout << "
chieu dai = " << d ;
cout << " chieu rong= " << r;
111 112
}
int dien_tich()
{
return d*r;
}
int chu_vi()
{
return 2*(d+r);
}
} ;
class DAY_HINH_CN
{
private:
int n; // So hinh ch nhat
HINH_CN *h;
public:
void nhapsl();
HINH_CN hinh_dt_max() ;
HINH_CN *hinh_cv_max() ;
} ;
void DAY_HINH_CN::nhapsl()
{
cout << "So hinh CN = " ;
cin >> n;
h = new HINH_CN[n+1];
for (int i=1;i<=n;++i)
h[i].nhapsl();
}
HINH_CN DAY_HINH_CN::hinh_dt_max()
{
HINH_CN hdtmax;
hdtmax = h[1];
for (int i=2; i<=n; ++i)
if (h[i].dien_tich() > hdtmax.dien_tich() )
hdtmax = h[i];
return hdtmax;
}
HINH_CN *DAY_HINH_CN::hinh_cv_max()
{
int imax = 1;
for (int i=2; i<=n; ++i)
if (h[i].chu_vi() > h[imax].chu_vi() )
imax = i ;
return (h+imax);
}
void main()
{
DAY_HINH_CN d;
HINH_CN hdtmax;
d.nhapsl();
hdtmax = d.hinh_dt_max();
hdtmax.in() ;
HINH_CN *hcvmax=d.hinh_cv_max();
hcvmax->in() ;
getch();
}
Ví dụ 2 minh hoạ:
113 114
+ Thuộc tính (thành phần dữ liệu) của lớp có thể là đối tượng của lớp khác đã định nghĩa bên trên.
+ Phương thức có giá trị trả về kiểu đối tượng
+ Vai trò của con trỏ this (xem phương thức maxdt của lớp TAM_GIAC)
+ Phương thức tĩnh (xem phương thức tao_tg của lớp TAM_GIAC)
Nội dung chương trình là nhập một dẫy các điểm, sau đó tìm tam giác lớn nhất (về diện tích) có đỉnh là các điểm vừa nhập.
Chương trình được tổ chức thành 2 lớp:
+ Lớp DIEM gồm:
- Các thuộc tính: x và y (toạ độ của điểm)
- Các phương thức
void nhapsl() ; // Nhập x, y
void in() ; // In toạ độ
double do_dai(DIEM d2) ; // Tính độ dài đoạn thẳng qua
// 2 điểm (điểm ẩn xác định bởi this và điểm d2)
+ Lớp TAM_GIAC gồm:
- Các thuộc tính:
DIEM d1,d2,d3; // 3 đỉnh của tam giác
- Các phương thức:
void nhapsl(); // Nhập toạ độ 3 đỉnh
void in(); // In toạ độ 3 đỉnh
// Tạo một đối tượng TAM_GIAC từ 3 đối tượng DIEM
static TAM_GIAC tao_tg(DIEM e1, DIEM e2, DIEM e3)
double dien_tich() ; // Tính diện tích
// Tìm tam giác có diện tích max trong 2 tam giác *this và t2
TAM_GIAC maxdt(TAM_GIAC t2);
+ Các vấn đề đáng chú ý trong chương trình là:
- Phương thưc tĩnh tao_tg (sẽ giải thích bên dưới)
- Phương thưc maxdt
+ Thuật toán là:
- Duyệt qua các tổ hợp 3 điểm.
- Dùng phương thức tao_tg để lập tam giác từ 3 điểm
- Dùng phương thức maxdt để chọn tam giác có diện tích lớn hơn trong 2 tam giác: tam giác vừa tạo và tam giác có diện tích max (trong số các tam giác đã tạo)
#include <conio.h>
#include <iostream.h>
#include <math.h>
class DIEM
{
private:
double x,y; // Toa do cua diem
public:
void nhapsl()
{
cout << " Toa do x, y: " ;
cin >> x >> y ;
}
void in()
{
cout << " x = " << x << " y = " << y;
}
double do_dai(DIEM d2)
{
return sqrt(pow(x-d2.x,2) + pow(y-d2.y,2) );
}
115 116
} ;
class TAM_GIAC
{
private:
DIEM d1,d2,d3; // 3 dinh tam giac
public:
void nhapsl();
void in();
static TAM_GIAC tao_tg(DIEM e1, DIEM e2, DIEM e3)
{
TAM_GIAC t;
t.d1=e1; t.d2 = e2; t.d3=e3;
return t;
}
double dien_tich() ;
TAM_GIAC maxdt(TAM_GIAC t2);
} ;
void TAM_GIAC::nhapsl()
{
cout << "
Dinh 1 - " ;
d1.nhapsl();
cout << "
Dinh 2 - " ;
d2.nhapsl();
cout << "
Dinh 3 - " ;
d3.nhapsl();
}
void TAM_GIAC::in()
{
cout << "
Dinh 1: " ; d1.in();
cout << "
Dinh 2: " ; d2.in();
cout << "
Dinh 3: " ; d3.in();
}
double TAM_GIAC::dien_tich()
{
double a,b,c,p,s;
a=d1.do_dai(d2);
b=d2.do_dai(d3);
c=d3.do_dai(d1);
p=(a+b+c)/2;
return sqrt(p*(p-a)*(p-b)*(p-c));
}
TAM_GIAC TAM_GIAC::maxdt(TAM_GIAC t2)
{
if (this->dien_tich() > t2.dien_tich())
return *this ;
else
return t2;
}
void main()
{
DIEM d[50];
int n, i ;
clrscr();
cout << "
So diem= ";
cin >> n;
for (i=1; i<=n; ++i)
{
117 118
cout << "
Nhap diem " << i << " - " ;
d[i].nhapsl();
}
int j, k ;
TAM_GIAC tmax, t;
tmax = TAM_GIAC::tao_tg(d[1],d[2],d[3]);
for (i=1;i<=n-2;++i)
for (j=i+1;j<=n-1;++j)
for (k=j+1;k<=n;++k)
{
t=TAM_GIAC::tao_tg(d[i],d[j],d[k]);
tmax = tmax.maxdt(t);
}
cout << "
Tam giac co dien tich lon nhat: " ;
tmax.in();
cout << "
Dien tich = " << tmax.dien_tich();
getch();
}
Chú ý 1: Để tạo một đối tượng TAM_GIAC từ 3 đối tượng DIEM ta đã dùng phương thức tĩnh:
static TAM_GIAC tao_tg(DIEM e1, DIEM e2, DIEM e3)
{
TAM_GIAC t;
t.d1=e1; t.d2 = e2; t.d3=e3;
return t;
}
Phương thức tĩnh (sẽ nói thêm trong các mục bên dưới) có các đặc điểm sau:
+ Nó giống phương thức thông thường ở chỗ: Trong thân của nó có thể truy nhập tới các thành phần của lớp (cụ thể là lớp TAM_GIAC).
+ Nó khác phương thức thông thường ở chỗ:
- Không có đối ngầm định xác định bởi con trỏ this (như phương thức thông thường). Như vậy phương thức tao_tg có đúng 3 đối.
- Nó không gắn với một đối tượng cụ thể nào của lớp, nên trong lời gọi tới phương thức ảo có thể dùng tên lớp, ví dụ (xem hàm main):
t=TAM_GIAC::tao_tg(d[i],d[j],d[k]);
Chú ý 2: Không thể thay phương thức tĩnh tao_tg bằng hàm, vì trong thân hàm không được truy xuất đến các thuộc tính của lớp TAM_GIAC. Tuy nhiên có một giải pháp khác là dùng khái niệm hàm bạn (friend). Hàm bạn của một lớp có quyền truy nhập đến các thuộc tính của lớp. Trong ví dụ 3 dưới đây ta sẽ xây dựng hàm tao_tg như một hàm bạn của lớp TAM_GIAC.
Chú ý 3: còn một giải pháp nữa là dùng hàm tạo (constructor) sẽ trình bầy trong các chương sau:
Chương trình dưới đây có nội dung giống như ví dụ 2, nhưng thay phương thức tĩnh tao_tg bằng hàm bạn tao_tg.
Ví dụ 3: Minh hoạ cách dùng hàm bạn. Nội dung chương trình giống như trong ví dụ 2.
#include <conio.h>
#include <iostream.h>
#include <math.h>
class DIEM
{
private:
double x,y; // Toa do cua diem
public:
void nhapsl()
{
cout << " Toa do x, y: " ;
cin >> x >> y ;
119 120
}
void in()
{
cout << " x = " << x << " y = " << y;
}
double do_dai(DIEM d2)
{
return sqrt(pow(x-d2.x,2) + pow(y-d2.y,2) );
}
} ;
class TAM_GIAC
{
private:
DIEM d1,d2,d3; // 3 dinh tam giac
public:
void nhapsl();
void in();
friend TAM_GIAC tao_tg(DIEM e1, DIEM e2, DIEM e3)
{
TAM_GIAC t;
t.d1=e1; t.d2 = e2; t.d3=e3;
return t;
}
double dien_tich() ;
TAM_GIAC maxdt(TAM_GIAC t2);
} ;
void TAM_GIAC::nhapsl()
{
cout << "
Dinh 1 - " ;
d1.nhapsl();
cout << "
Dinh 2 - " ;
d2.nhapsl();
cout << "
Dinh 3 - " ;
d3.nhapsl();
}
void TAM_GIAC::in()
{
cout << "
Dinh 1: " ; d1.in();
cout << "
Dinh 2: " ; d2.in();
cout << "
Dinh 3: " ; d3.in();
}
double TAM_GIAC::dien_tich()
{
double a,b,c,p,s;
a=d1.do_dai(d2);
b=d2.do_dai(d3);
c=d3.do_dai(d1);
p=(a+b+c)/2;
return sqrt(p*(p-a)*(p-b)*(p-c));
}
TAM_GIAC TAM_GIAC::maxdt(TAM_GIAC t2)
{
if (this->dien_tich() > t2.dien_tich())
return *this ;
else
return t2;
121 122
}
void main()
{
DIEM d[50];
int n, i ;
clrscr();
cout << "
So diem= ";
cin >> n;
for (i=1; i<=n; ++i)
{
cout << "
Nhap diem " << i << " - " ;
d[i].nhapsl();
}
int j, k ;
TAM_GIAC tmax, t;
tmax = tao_tg(d[1],d[2],d[3]);
for (i=1;i<=n-2;++i)
for (j=i+1;j<=n-1;++j)
for (k=j+1;k<=n;++k)
{
t=tao_tg(d[i],d[j],d[k]);
tmax = tmax.maxdt(t);
}
cout << "
Tam giac co dien tich lon nhat: " ;
tmax.in();
cout << "
Dien tich = " << tmax.dien_tich();
getch();
}
Chú ý: Hàm bạn có thể xây dựng bên trong định nghĩa lớp (như chương trình trên) hoặc có thể khai báo bên trong và xây dựng bên ngoài định nghĩa lớp như sau:
class TAM_GIAC
{
private:
DIEM d1,d2,d3; // 3 dinh tam giac
public:
void nhapsl();
void in();
friend TAM_GIAC tao_tg(DIEM e1,DIEM e2,DIEM e3);
double dien_tich() ;
TAM_GIAC maxdt(TAM_GIAC t2);
} ;
TAM_GIAC tao_tg(DIEM e1, DIEM e2, DIEM e3)
{
TAM_GIAC t;
t.d1=e1; t.d2 = e2; t.d3=e3;
return t;
}
Nhận xét: Không cho phép dùng từ khoá friend khi xây dựng hàm (bên ngoài lớp)
§ 6. Hàm, hàm bạn
6.1. Hàm có các tính chất sau:
123 124
+ Phạm vi của hàm là toàn bộ chương trình, vì vậy hàm có thể được gọi tới từ bất kỳ chỗ nào. Như vây trong các phương thức có thể sử dụng hàm.
+ Đối của hàm có thể là các đối tượng, tuy nhiên có một hạn chế là trong thân hàm không cho phép truy nhập tới thuộc tính của các đối này. Ví dụ giả sử đã định nghĩa lớp:
class DIEM
{
private:
double x,y; // Toa do cua diem
public:
void nhapsl()
{
cout << " Toa do x, y: " ;
cin >> x >> y ;
}
void in()
{
cout << " x = " << x << " y = " << y;
}
};
Dùng lớp DIEM, ta xây dựng hàm tính độ dài của đoạn thẳng đi qua 2 điểm như sau:
double do_dai(DIEM d1, DIEM d2)
{
return sqrt(pow(d1.x-d2.x,2) + pow(d1.y-d2.y,2));
}
Hàm này sẽ bị báo lỗi khi dịch, vì trong thân hàm không cho phép sử dụng các thuộc tính d1.x, d1.y, d2.x, d2.y của các đối tượng d1 và d2 thuộc lớp DIEM.
+ Phạm vi sử dụng của các phương thức (public) là toàn chương trình, vì vậy trong thân hàm có thể gọi tới các phương thức. Ví dụ giả sử đã định nghĩa lớp:
class DIEM
{
private:
double x,y; // Toa do cua diem
public:
void nhapsl()
{
cout << " Toa do x, y: " ;
cin >> x >> y ;
}
void in()
{
cout << " x = " << x << " y = " << y;
}
double do_dai(DIEM d2)
{
return sqrt(pow(x-d2.x,2) + pow(y-d2.y,2) );
}
} ;
Khi đó bằng cách dùng phương thức do_dai, ta có thể viết hàm tính diện tích tam giác có đỉnh là các đối tượng d1, d2, d3 của lớp DIEM như sau:
double dt_tg(DIEM d1, DIEM d2, DIEM d3)
{
double a,b,c,p,s;
a=d1.do_dai(d2);
b=d2.do_dai(d3);
125 126
c=d3.do_dai(d1);
p=(a+b+c)/2;
return sqrt(p*(p-a)*(p-b)*(p-c));
}
Bằng cách dùng hàm dt_tg, có thể tổ chức lại chương trình tìm tam giác có diện tích lớn nhất (ở mục trên) một cách đơn giản hơn( bỏ đi lớp TAM_GIAC) như ví dụ sau.
Ví dụ 1:
#include <conio.h>
#include <iostream.h>
#include <math.h>
class DIEM
{
private:
double x,y; // Toa do cua diem
public:
void nhapsl()
{
cout << " Toa do x, y: " ;
cin >> x >> y ;
}
void in()
{
cout << " x = " << x << " y = " << y;
}
double do_dai(DIEM d2)
{
return sqrt(pow(x-d2.x,2) + pow(y-d2.y,2) );
}
} ;
double dt_tg(DIEM d1, DIEM d2, DIEM d3)
{
double a,b,c,p,s;
a=d1.do_dai(d2);
b=d2.do_dai(d3);
c=d3.do_dai(d1);
p=(a+b+c)/2;
return sqrt(p*(p-a)*(p-b)*(p-c));
}
void main()
{
DIEM d[50];
int n, i,j,k,imax,jmax,kmax ;
clrscr();
cout << "
So diem= ";
cin >> n;
for (i=1; i<=n; ++i)
{
cout << "
Nhap diem " << i << " - " ;
d[i].nhapsl();
}
imax=1; jmax=2; kmax=3;
for (i=1;i<=n-2;++i)
for (j=i+1;j<=n-1;++j)
for (k=j+1;k<=n;++k)
if (dt_tg(d[i],d[j],d[k]) > dt_tg(d[imax],d[jmax],d[kmax]))
{
imax = i ;
jmax = j;
kmax = k;
127 128
}
cout << "
Tam giac co dien tich lon nhat: " ;
cout << "
Dinh 1 - "; d[imax].in();
cout << "
Dinh 2 - "; d[jmax].in();
cout << "
Dinh 3 - "; d[kmax].in();
cout << "
Dien tich = " << dt_tg(d[imax],d[jmax],d[kmax]) ;
getch();
}
Nhận xét: Chương trình trên làm việc trên mảng d kiểu DIEM. Bây giờ nếu ta dùng mảng ngoài thì từ số thứ tự sẽ suy ra phần tử của mảng. Như vây hàm
double dt_tg(DIEM d1, DIEM d2, DIEM d3);
có 3 đối kiểu DIEM có thể thay bằng hàm có 3 đối nguyên:
double dt_tg(int i, int j, int k);
để tính diện tích tam giác có đỉnh là d[i], d[j] và d[k] . ý tưởng này được thể hiện trong ví dụ sau.
Ví dụ 2: Chương trình dùng mảng đối tượng ngoài.
Chú ý: Khai báo mảng đối tượng phải đặt sau định nghĩa kiểu đối tượng (định nghĩa lớp).
#include <conio.h>
#include <iostream.h>
#include <math.h>
double dt_tg(int i, int j, int k); // Khai báo hàm dt_tg
class DIEM
{
private:
double x,y; // Toa do cua diem
public:
void nhapsl();
void in();
double do_dai(DIEM d2);
} ;
// Chú ý: Khai báo mảng kiểu DIEM phải đặt sau định nghĩa
// lớp DIEM
DIEM d[50];
void DIEM::nhapsl()
{
cout << " Toa do x, y: " ;
cin >> x >> y ;
}
void DIEM::in()
{
cout << " x = " << x << " y = " << y;
}
double DIEM::do_dai(DIEM d2)
{
return sqrt(pow(x-d2.x,2) + pow(y-d2.y,2) );
}
double dt_tg(int i, int j, int k)
{
double a,b,c,p,s;
a=d[i].do_dai(d[j]);
b=d[j].do_dai(d[k]);
c=d[k].do_dai(d[i]);
p=(a+b+c)/2;
return sqrt(p*(p-a)*(p-b)*(p-c));
}
void main()
{
int n, i,j,k,imax,jmax,kmax ;
129 130
clrscr();
cout << "
So diem= ";
cin >> n;
for (i=1; i<=n; ++i)
{
cout << "
Nhap diem " << i << " - " ;
d[i].nhapsl();
}
imax=1; jmax=2; kmax=3;
for (i=1;i<=n-2;++i)
for (j=i+1;j<=n-1;++j)
for (k=j+1;k<=n;++k)
if (dt_tg(i,j,k) > dt_tg(imax,jmax,kmax))
{
imax = i ;
jmax = j;
kmax = k;
}
cout << "
Tam giac co dien tich lon nhat: " ;
cout << "
Dinh 1 - "; d[imax].in();
cout << "
Dinh 2 - "; d[jmax].in();
cout << "
Dinh 3 - "; d[kmax].in();
cout << "
Dien tich = " << dt_tg(imax,jmax,kmax);
getch();
}
6.2. Hàm bạn (friend function)
6.2.1. Để một hàm trở thành bạn của một lớp, có 2 cách viết:
Cách 1: Dùng từ khoá friend để khai báo hàm trong lớp và xây dựng hàm bên ngoài như các hàm thông thường (không dùng từ khoá friend). Mẫu viết như sau:
class A
{
private:
// Khai báo các thuộc tính
public:
...
// Khai báo các hàm bạn của lớp A
friend void f1(...);
friend double f2(...);
friend A f3(...) ;
...
} ;
// Xây dựng các hàm f1, f2, f3
void f1(...)
{
...
}
double f2(...)
{
...
}
A f3(...)
{
...
}
131 132
Cách 2: Dùng từ khoá friend để xây dựng hàm trong định nghĩa lớp. Mẫu viết như sau:
class A
{
private:
// Khai báo các thuộc tính
public:
...
// Xây dựng các hàm bạn của lớp A
void f1(...)
{
...
}
double f2(...)
{
...
}
A f3(...)
{
...
}
...
} ;
6.2.2. Tính chất của hàm bạn
Trong thân hàm bạn của một lớp có thể truy nhập tới các thuộc tính của các đối tượng thuộc lớp này. Đây là sự khác nhau duy nhất giữa hàm bạn và hàm thông thường. Chú ý rằng hàm bạn không phải là phương thức của lớp. Phương thức có một đối ẩn (ứng với con trỏ this) và lời gọi của phương thức phải gắn với một đối tượng nào đó (địa chỉ đối tượng này được truyền cho con trỏ this). Lời gọi của hàm bạn giống như lời gọi của hàm thông thường.
Ví dụ sau sẽ so sánh phương thức, hàm bạn và hàm tự do (hàm thông thường). Xét lớp SP (số phức). Hãy so sánh 3 phương án để thực hiện việc cộng 2 số phức:
Phương án 1: Dùng phương thức
class SP
{
private:
double a; // Phần thực
double b; // Phần ảo
public:
SP cong(SP u2)
{
SP u:
u.a = this->a + u2.a ;
u.b = this->b + u2.b ;
return u;
}
} ;
Cách dùng
SP u, u1, u2;
u = u1.cong(u2);
Phương án 2: Dùng hàm bạn
class SP
{
private:
double a; // Phần thực
133 134
double b; // Phần ảo
public:
friend SP cong(SP u1, SP u2)
{
SP u:
u.a = u1.a + u2.a ;
u.b = u1.b + u2.b ;
return u;
}
};
Cách dùng
SP u, u1, u2;
u = cong(u1, u2);
Phương án 3: Dùng hàm tự do
class SP
{
private:
double a; // Phần thực
double b; // Phần ảo
public:
...
} ;
SP cong(SP u1, SP u2)
{
SP u:
u.a = u1.a + u2.a ;
u.b = u1.b + u2.b ;
return u;
}
Phương án này không được chấp nhận, Trình biên dịch sẽ báo lỗi vì trong thân hàm không được quyền truy xuất đến các thuộc tính riêng (private) a, b của các đối tượng u, u1 và u2 thuộc lớp SP.
6.2.3. Một hàm có thể là bạn của nhiều lớp được không? Câu trả lời là được. Khi một hàm là bạn của nhiều lớp, thì nó có quyền truy nhập tới tất cả các thuộc tính của các đối tượng trong các lớp này. Để làm cho hàm f trở thành bạn của các lớp A, B và C ta sử dụng mẫu viết sau:
class B; // Khai báo trước lớp A
class B; // Khai báo trước lớp B
class C; // Khai báo trước lớp C
// Định nghĩa lớp A
class A
{
// Khai báo f là bạn của A
friend void f(...) ;
} ;
// Định nghĩa lớp B
class B
{
// Khai báo f là bạn của B
friend void f(...) ;
} ;
// Định nghĩa lớp C
class C
{
// Khai báo f là bạn của C
friend void f(...) ;
135 136
} ;
// Xây dụng hàm f
void f(...)
{
...
}
Chương trình sau đây minh hoạ cách dùng hàm bạn (bạn của một lớp và bạn của nhiều lớp). Chương trình đưa vào 2 lớp VT (véc tơ), MT (ma trận) và 3 hàm bạn để thực hiện các thao tác trên 2 lớp này:
// Hàm bạn với lớp VT dùng để in một véc tơ
friend void in(const VT &x);
// Hàm bạn với lớp MT dùng để in một ma trận
friend void in(const MT &a);
// Hàm bạn với cả 2 lớp MT và VT dùng để nhân ma trận với véc tơ
friend VT tich(const MT &a,const VT &x);
Nội dung chương trình là nhập một ma trận vuông cấp n và một véc tơ cấp n, sau đó thực hiện phép nhân ma trận với véc tơ vừa nhập.
// Chương trình CT3_09.CPP
#include <conio.h>
#include <iostream.h>
#include <math.h>
class VT;
class MT ;
class VT
{
private:
int n;
double x[20]; // Toa do cua diem
public:
void nhapsl();
friend void in(const VT &x);
friend VT tich(const MT &a,const VT &x) ;
} ;
class MT
{
private:
int n;
double a[20][20];
public:
friend VT tich(const MT &a,const VT &x);
friend void in(const MT &a);
void nhapsl();
} ;
void VT::nhapsl()
{
cout << "
Cap vec to = ";
cin >> n ;
for (int i=1; i<=n ; ++i)
{
cout << "
Phan tu thu " << i << " = " ;
cin >> x[i];
}
}
void MT::nhapsl()
{
cout << "
Cap ma tran = ";
cin >> n ;
137 138
for (int i=1; i<=n ; ++i)
for (int j=1; j<=n; ++j)
{
cout << "
Phan tu thu hang "<< i << " cot " << j << "=" ;
cin >> a[i][j];
}
}
VT tich(const MT &a,const VT &x)
{
VT y;
int n=a.n;
if (n!=x.n)
return x;
y.n = n;
for (int i=1; i<=n; ++i)
{
y.x[i]=0;
for (int j=1; j<=n; ++j)
y.x[i] += a.a[i][j]*x.x[j];
}
return y;
}
void in(const VT &x)
{
cout << "
";
for (int i=1; i<=x.n; ++i)
cout << x.x[i] << " ";
}
void in(const MT &a)
{
for (int i=1; i<=a.n; ++i)
{
cout << "
" ;
for (int j=1; j<=a.n; ++j)
cout << a.a[i][j] << " ";
}
}
void main()
{
MT a; VT x,y;
clrscr();
a.nhapsl();
x.nhapsl();
y=tich(a,x);
clrscr();
cout << "
Ma tran A:";
in(a);
cout << "
Vec to x: " ;
in(x);
cout << "
Vec y = Ax: " ;
in(y);
getch();
}
139 140
§ 7. Phạm vi truy xuất
7.1. Các từ khoá private và public
Các thành phần (thuộc tính và phương thức) của lớp có thể khai báo là private hoặc public theo mẫu:
private:
// Khai báo các thành phần riêng của lớp
public:
// Khai báo các thành phần chung (công cộng)
Chú ý: Các thành phần khai báo mặc định (không dùng các từ khoá private và public) được xem là các thành phần private.
7.2. Các thành phần riêng của lớp chỉ được sử dụng trong phạm vi của lớp (trong thân các phương thức của lớp). Chúng không thể đem ra sử dụng bên ngoài lớp.
+ Một thuộc tính private: Thuộc tính này (của một đối tượng nào đó) chỉ có thể được sử dụng trong thân của các phương thức cùng lớp.
+ Một phương thức private: Chỉ được sử dụng trong thân của các phương thức cùng lớp.
Ví dụ sau minh hoạ cách dùng phương thức private. Xét lớp PS (phân số) với 2 thuộc tính nguyên là t (tử) và m (mẫu). Giả sử cần xây dựng các phương thức để thực hiện các phép toán cộng trừ, nhân, chia phân số. Do các phép toán này cần dùng trong toàn bộ chương trình, nên các phương thức thực hiện các phép toán cần khai báo là public. Để thực hiện các phép tính trên phân số cần dùng đến phép rút gọn phân số. Ta có thể dùng một phương thức private để làm điều này vì việc rút gọn chỉ dùng trong nội bộ lớp.
7.3. Các thành phần công cộng của lớp có phạm vi sử dụng trong toàn chương trình. Như vậy nếu một thuộc tính được khai báo là public, thì nó có thể được truy nhập trong thân của bất kỳ hàm nào trong chương trình.
Ví dụ trong §6 đã chỉ ra phương án dùng một hàm (tự do) để thực hiện phép cộng 2 số phức như sau là sai:
Phương án 3: Dùng hàm tự do
class SP
{
private:
double a; // Phần thực
double b; // Phần ảo
public:
...
} ;
SP cong(SP u1, SP u2)
{
SP u:
u.a = u1.a + u2.a ;
u.b = u1.b + u2.b ;
return u;
}
Tuy nhiên nếu sửa chữa bằng cách khai báo các thuộc tính a và b là public thì lại được.
Nhận xét: Các thuộc tính thường khai báo là private để đảm bảo tính dấu kín, an toàn dữ liệu của lớp.
§ 8. Các phương thức toán tử
8.1. Cách đặt tên
Các phương thức toán tử được xây dựng như các phương thức thông thường, chỉ có khác cách đặt tên. Tên các phương thức toán tử (cũng giống như hàm toán tử) được tạo bằng cách ghép từ khoá operator với một phép toán, ví dụ:
141 142
operator+
operator<<
operator>>
8.2. Con trỏ this
Cũng giống như phương thức thông thường, phương thức toán tử có đối đầu tiên (đối không tường minh) là con trỏ this.
8.3. Toán tử một toán hạng
Các phương thức toán tử một toán hạng: Dùng ngay con trỏ this để biểu thị toán hạng duy nhất này, nên trong phương thức sẽ không có đối tường minh. Ví dụ phương thức toán tử - (đổi dấu) một đối tượng kiểu SP (số phức) có thể viết như sau:
class SP
{
private:
double a; // Phần thực
double b; // Phần ảo
public:
SP operator-();
} ;
SP SP:: operator-()
{
SP u ;
u.a = - this->a ;
u.b = - this->b ;
return u;
}
Cách dùng:
SP u, v;
u = -v;
8.4. Toán tử hai toán hạng
Các phương thức toán tử hai toán hạng: Con trỏ this ứng với toán hạng thứ nhất, nên trong phương thức chỉ cần dùng một đối tường minh để biểu thị toán hạng thứ hai. Ví dụ phương thức toán tử + (cộng) hai đối tượng kiểu SP (số phức) có thể viết như sau:
class SP
{
private:
double a; // Phần thực
double b; // Phần ảo
public:
SP operator+(SP u2);
} ;
SP SP:: operator+(SP u2)
{
SP u ;
u.a = this->a + u2.a ;
u.b = this->b + u2.b ;
return u;
}
Cách dùng:
SP p, p, r;
r = p + q ;
8.5. Lớp DT (Đa thức)
Chương trình sau sẽ định nghĩa lớp DT và đưa vào các phương thức, hàm:
+ Các thuộc tính:
int n ; // bậc đa thức
143 144
double *a ; // trỏ tới vùng nhớ chứa các hệ số đa thức
+ Các phương thức operator+, operator- dùng để đổi dấu các hệ số đa thức
operator+ dùng để cộng 2 đa thức
operator- dùng để trừ 2 đa thức
operator* dùng để nhân 2 đa thức
operator^ dùng để tính giá trị đa thức
operator[] dùng để cho biết bậc và hệ số của đa thức
+ Các hàm bạn:
operator<< dùng để in các hệ số đa thức
operator>> dùng để nhập các hệ số đa thức
+ Hàm (tự do)
double F(DT p, double x) dùng để tính p(x)-giá trị đa thức tại x
+ Nói thêm về phương thức chỉ số và hàm tự do F
- Nếu p là đối tượng của lớp DT, thì hàm chỉ số cho biết:
p[-1] = double(n)
p[i] = a[i] , i=0, 1, ..., n
- Hàm tự do F sẽ dùng phương thức chỉ số để xác định n , các hệ số đa thức và dùng chúng để tính giá trị đa thức.
+ Trong chương trình sử dụng hàm new để cấp phát vùng nhớ chứa hệ số đa thức.
+ Nội dung chương trình gồm:
- Nhập, in các đa thức p, q, r, s
- Tính đa thức: f = -(p + q)*(r - s)
- Nhập các số thực x1 và x2
- Tính f(x1) (bằng cách dùng phương thức operator^)
- Tính f(x2) (bằng cách dùng hàm F)
// Chương trình CT3_10.CPP
#include <conio.h>
#include <iostream.h>
#include <math.h>
class DT
{
private:
int n; // Bac da thuc
double *a; // Tro toi vung nho chua cac he so da thuc
// a0, a1,...
public:
friend ostream& operator<< (ostream& os,const DT &d);
friend istream& operator>> (istream& is,DT &d);
DT operator-();
DT operator+(const DT &d2);
DT operator-(DT d2);
DT operator*(const DT &d2);
double operator^(const double &x); // Tinh gia tri da thuc
double operator[](int i)
{
if(i<0)
return double(n);
else
return a[i];
}
} ;
// Ham tinh gia tri da thuc
double F(DT d,double x)
{
double s=0.0 , t=1.0;
int n;
n = int(d[-1]);
for (int i=0; i<=n; ++i)
{
145 146
s += d[i]*t;
t *= x;
}
return s;
}
ostream& operator<< (ostream& os,const DT &d)
{
os << " - Cac he so (tu ao): " ;
for (int i=0 ; i<= d.n ; ++i)
os << d.a[i] <<" " ;
return os;
}
istream& operator>> (istream& is,DT &d)
{
cout << " - Bac da thuc: " ;
cin >> d.n;
d.a = new double[d.n+1];
cout << "Nhap cac he so da thuc:
" ;
for (int i=0 ; i<= d.n ; ++i)
{
cout << "He so bac " << i << " = " ;
is >> d.a[i] ;
}
return is;
}
DT DT::operator-()
{
DT p;
p.n = n;
p.a = new double[n+1];
for (int i=0 ; i<=n ; ++i)
p.a[i] = -a[i];
return p;
}
DT DT::operator+(const DT &d2)
{
DT d;
int k,i;
k = n > d2.n ? n : d2.n ;
d.a = new double[k+1];
for (i=0; i<=k ; ++i)
if (i<=n && i<=d2.n)
d.a[i] = a[i] + d2.a[i];
else if (i<=n)
d.a[i] = a[i];
else
d.a[i] = d2.a[i];
i=k;
while(i>0 && d.a[i]==0.0) --i;
d.n = i;
return d ;
}
DT DT::operator-(DT d2)
{
return (*this + (-d2));
}
DT DT::operator*(const DT &d2)
147 148
{
DT d;
int k, i, j;
k = d.n = n + d2.n ;
d.a = new double[k+1];
for (i=0; i<=k; ++i) d.a[i] = 0;
for (i=0 ; i<= n ; ++i)
for (j=0 ; j<= d2.n ; ++j)
d.a[i+j] += a[i]*d2.a[j] ;
return d;
}
double DT::operator^(const double &x)
{
double s=0.0 , t=1.0;
for (int i=0 ; i<= n ; ++i)
{
s += a[i]*t;
t *= x;
}
return s;
}
void main()
{
DT p,q,r,s,f;
double x1,x2,g1,g2;
clrscr();
cout <<"
Nhap da thuc P " ; cin >> p;
cout << "
Da thuc p " << p ;
cout <<"
Nhap da thuc Q " ; cin >> q;
cout << "
Da thuc q " << q ;
cout <<"
Nhap da thuc R " ; cin >> r;
cout << "
Da thuc r " << r ;
cout <<"
Nhap da thuc S " ; cin >> s;
cout << "
Da thuc s " << s ;
f = -(p+q)*(r-s);
cout << "
Nhap so thuc x1: " ; cin >> x1;
cout << "
Nhap so thuc x2: " ; cin >> x2;
g1 = f^x1;
g2 = F(f,x2);
cout << "
Da thuc f " << f ;
cout << "
f("<<x1<<") = " << g1;
cout << "
f("<<x2<<") = " << g2;
getch();
}
149
147 148
Bạn đang đọc truyện trên: AzTruyen.Top