CHƯƠNG 11: ĐỐI TƯỢNG VÀ LỚP
• Nội dung
Đối tượng
Lớp
Con trỏ this
Hàm thiết lập
Hàm hủy bỏ
Hàm thiết lập sao chép
Các thành phần tĩnh
Hàm bạn và lớp bạn
Đối tượng
• Đối tượng là một khái niệm trong lập trình hướng đối tượng biểu thị sự liên kết
giữa dữ liệu và các thủ tục (gọi là phương thức) thao tác trên dữ liệu
• ĐỐI TƯỢNG = DỮ LIỆU + PHƯƠNG THỨC
Lớp
Lớp là một mô tả nhóm các đối tượng có cùng bản chất, ngược lại mỗi một đối
tượng là một biểu hiện cụ thể cho những mô tả trừu tượng đó.
• khai báo lớp
• class <tên lớp>
• {
• private:
• <khai báo các thành phần riêng trong từng đối tượng>
• public:
• <khai báo các thành phần công cộng trong từng đối tượng>
• };
• <định nghĩa các hàm thành phần chưa được định nghĩa bên trong khai báo lớp>
Các thành phần của lớp có thể là thành phần dữ liệu (thuộc tính) hoặc hàm thành
phần (phương thức).
Thành phần dữ liệu được khai báo giống như khai báo biến
Hàm thành phần được khai báo giống như khai báo hàm trong C. Có hai
cách định nghĩa một hàm thành phần: định nghĩa trong lớp và ngoài lớp.
Tất cả các thành phần được liệt kê sau từ khóa private hoặc protected chỉ
được truy nhập bởi các thành phần bên trong lớp đó. Còn tất cả các thành
phần được liệt kê sau từ khóa public có thể được truy nhập trong bất kỳ
hàm nào.
• Ví dụ 1:
• Class Diem
• {
• private:
• int x, y; // Các thành phần dữ liệu riêng
• public:
• //Các hàm thành phần công cộng
• void Nhap();
• void Xuat();
• };
• //Định nghĩa các hàm thành phần bên ngoài lớp
• void Diem::Nhap()
• { cout <<“Nhap toa do:”;
• cin >> x >> y;
• }
• void Diem::Xuat()
• { cout << x <<“,” << y <<“\n”;
• }
• //Hàm chính
• void main()
• {
• Diem a; //Khai báo một đối tượng a thuộc lớp Diem
• a.Nhap(); //Đối tượng a thực hiện lời gọi hàm thành phần Nhap()
• a.Xuat();
• }
• Chú ý: Lời gọi hàm thành phần luôn có một và chỉ một tham số ngầm định
là đối tượng thực hiện lời gọi hàm. Như vậy các thành phần dữ liệu viết
trong các hàm thành phần được hiểu là thuộc đối tượng dùng làm tham số
ngầm định trong lời gọi hàm.
Con trỏ this
• C++ sử dụng con trỏ đặc biệt this trong các hàm thành phần. Con trỏ này
luôn trỏ tới đối tượng dùng làm tham số ngầm định trong lời gọi hàm thành
phần.
• Ví dụ 2: Các hàm thành phần Nhap() và Xuat() có thể viết theo cách khác
như sau
• void Diem::Nhap()
• {
• cout <<“Nhap toa do:”;
• cin >> this->x >> this->y;
• }
• void Diem::Xuat()
• {
• cout << this->x <<“,” << this->y <<“\n”;
• }
Hàm thiết lập
Hàm thiết lập là một hàm đặc biệt được gọi tự động mỗi khi có một đối
tượng được khai báo.
Chức năng của hàm thiết lập là để khởi tạo giá trị cho các thành phần dữ
liệu hoặc xin cấp phát vùng nhớ cho các thành phần dữ liệu động.
Hàm thiết lập được khai báo giống như một hàm thành phần với tên trùng
tên với lớp, không có giá trị trả về và không cần khai báo void.
Có thể có nhiều hàm thiết lập trong cùng một lớp (chồng hàm thiết lập).
Khi một lớp có nhiều hàm thiết lập, việc tạo ra các đối tượng phải kèm theo
các tham số phù hợp với một trong các hàm thiết lập đã khai bào.
Khi người sử dụng không khai báo tường minh bất kỳ một hàm thiết lập
nào cho lớp thì trình biên dịch tự động phát sinh cho lớp một hàm thiết lập
ngầm định không tham số, hàm này không thực hiện bất cứ nhiệm vụ nào
ngoài việc “lấp chỗ trống”.
• Ví dụ 3:
• Class Diem
• {
• private:
• int x, y;
• public:
• . . .
• Diem(); //Khai báo hàm thiết lập không đối số
• Diem(int xx); //Khai báo hàm thiết lập một đối số
• Diem(int xx, int yy); //Khai báo hàm thiết lập hai đối số
• };
• //Định nghĩa các hàm thành phần
• Diem::Diem()
• {
• cout << “Goi ham thiet lap khong doi so\n”;
• x = y = 0;
• }
• Diem::Diem(int xx)
• {
• cout << “Goi ham thiet lap mot doi so\n”;
• x = xx; y = 0;
• }
• Diem::Diem(int xx, int yy)
• {
• cout << “Goi ham thiet lap hai doi so\n”;
• x = xx; y = yy;
• }
• //Hàm chính
• void main()
• {
• Diem a; //Gọi hàm thiết lập không đối số
• a.Xuat();
• Diem b(2); //Gọi hàm thiết lập một đối số
• b.Xuat();
• Diem c(2, -4); //Gọi hàm thiết lập hai đối số
• c.Xuat();
• }
• Ví dụ 4:
• Class Diem
• {
• private:
• int x, y;
• public:
• . . .
• Diem(int xx = 0, int yy = 0); /*Khai báo hàm thiết lập hai đối số có
giá trị ngầm định*/
• };
• Diem::Diem(int xx, int yy)
• { cout << “Goi ham thiet lap hai doi so\n”;
• x = xx; y = yy;
• }
• void main()
• {
• // giống ví dụ 3
• }
Hàm hủy bỏ
Hàm hủy bỏ là một hàm đặc biệt được gọi tự động khi đối tượng tương ứng
bị xóa khỏi bộ nhớ.
Hàm hủy bỏ được khai báo giống như một hàm thành phần với tên bắt đầu
bằng dấu ~ và tiếp theo là tên lớp tương ứng, không có đối số và không có
giá trị trả về.
Một lớp chỉ có duy nhất một hàm hủy bỏ.
Chức năng của hàm hủy bỏ thường dùng để giải phóng vùng nhớ động mà
đối tượng đang quản lý.
Khi người sử dụng không khai báo tường minh một hàm hủy bỏ nào thì
chương trình dịch sẽ tự động sản sinh ra một hàm hủy bỏ ngầm định, hàm
này không làm gì cả ngoài việc “lấp chỗ trống”.
• Ví dụ 5:
• void fct(); //Nguyên mẫu hàm của một hàm tự do
• class Test
• {
• private:
• int n;
• public:
• Test(int nn); //Khai báo hàm thiết lập một đối số
• ~Test(); //Khai báo hàm hủy bỏ
• };
• //Định nghĩa các hàm thành phần
• Test::Test(int nn)
• { n = nn;
• cout << “Goi ham thiet lap voi n = “ << n << \n”;
• }
• Test::~Test()
• {
• cout << “Goi ham huy bo voi n = “ << n << \n”;
• }
• //Định nghĩa hàm tự do
• void fct(int r)
• {
• Test u(2*r) //Gọi hàm thiết lập 1 đối số
• }
• //Hàm chính
• void main()
• {
• Test a(1); //Gọi tới hàm thiết lập một đối số
for(int i = 1; i <= 2; i++)
• fct(i); //gọi hàm tự do fct()
• }
Sự cần thiết của hàm thiết lập và hàm hủy bỏ
• Trên thực tế với các lớp không có thành phần dữ liệu động chỉ cần sử dụng
hàm thiết lập và hủy bỏ ngầm định là đủ. Hàm thiết lập và hủy bỏ do người
lập trình tạo ra rất cần thiết khi các lớp có chứa thành phần dữ liệu động.
Khi tạo đối tượng hàm thiết lập xin cấp phát một bộ nhớ động, do đó hàm
hủy bỏ sẽ giải phóng vùng nhớ động đã được cấp phát trước đó.
• Ví dụ 6:
• class Vector
• {
• private:
• int n; //số chiều
• float *p; //con trỏ tới vùng nhớ tọa độ
• public:
• Vector(); Vector(int nn); ~vector();
• void Nhap(); void Xuat();
• };
• //Định nghĩa các hàm thành phần
• Vector::vector()
• { cout << “Goi ham thiet lap khong doi\n”;
• cout << “Nhap so chieu:”; cin >> n;
• p =new float[n];
• }
• Vector::vector(int nn)
• { cout << “Goi ham thiet lap mot doi\n”;
• n = nn;
• p =new float[n];
• }
• Vector::~Vector()
• { cout << “Goi ham huy bo\n”;
• delete[]p;
• p = NULL;
• }
• void Vector::Nhap()
• {
• for(int i = 0; i < n; i++)
• {
• cout << “Toa do thu “ << i <<“:”;
• cin >> p[i];
• }
• }
• void Vector::Xuat()
• {
• for(int i = 0; i < n; i++)
• cout << p[i] << “\t”
• cout << “\n”;
• }
• //Hàm chính
• void main()
• {
• Vector a; //Gọi hàm thiết lập không đối
• a.Nhap();
• a.Xuat();
• Vector b(3); //Gọi hàm thiết lập một đối
• b.Nhap();
• b.Xuat();
• }
• Chú ý: Nếu chỉ sử dụng hàm thiết lập mặc định thì đối tượng a được tạo ra
bởi các lệnh khai báo sẽ chưa có bộ nhớ để chứa các tọa độ. Như vậy đối
tượng chưa hoàn chỉnh và chưa dùng được.
Hàm thiết lập sao chép
Hàm thiết lập được gọi khi khai báo và khởi tạo nội dung một đối tượng
thông qua một đối tượng khác gọi là hàm thiết lập sao chép. Nhiệm vụ của
hàm thiết lập sao chép là tạo ra một đối tượng giống hệt một đối tượng đã
có trước đó.
Dạng khai bao hàm thiết lập sao chép:
<Tên lớp>(<Tên lớp> & <Tên đối số>);
Ngoài tình huống trên đây, còn có hai tình huống cần dùng hàm thiết lập
sao chép: truyền đối tượng cho hàm bằng tham trị hoặc hàm trả về một đối
tượng giống hệt một đối tượng cùng lớp đã có trước đó.
Giống như hàm thiết lập ngầm định, nếu không được khai báo tường minh,
sẽ có một hàm thiết lập sao chép ngầm định do chương trình dịch cung cấp.
Hàm này chỉ thực hiện thao tác tối thiểu: sao chép giá trị của các thành
phần dữ liệu trong đối tượng nguồn cho các thành phần dữ liệu tương ứng
trong đối tượng đích.
• Ví dụ 7:
• class Diem
• {
• private:
• . . .
• public:
• . . .
• Diem(Diem &u); //Khai báo hàm thiết lập sao chép
• float KC(Diem u);
• };
• //Định nghĩa hàm thiết lập sao chép
• Diem::Diem(Diem &u)
• {
• cout << “Goi ham thiet lap sao chep\n”;
• x = u.x; y = u.y;
• }
• Ví dụ 7:
• void main()
• {
• Diem a(2, -6);
• a.Xuat();
• Diem b = a; //Gọi hàm thiết lập sao chép
• b.Xuat();
• Diem c(-6, 8);
• float d = a.KC(c); //Gọi hàm thiết lập sao chép
• cout << “khoang cach tu a toi c la “ << d << “\n”;
• }
• Chú ý: Hàm thiết lập sao chép được khai báo tường minh trong ví dụ trên
có chức năng tương tự như hàm thiết lập sao chép mặc định. Nói chung,
với các lớp không khai báo thành phần dữ liệu động thì chỉ cần sử dụng
hàm thiết lập sao chép ngầm định là đủ. Tuy nhiên với những lớp có chứa
thành phần dữ liệu động không được dùng hàm thiết lập sao chép ngầm
định mà phải gọi hàm thiết lập sao chép tường minh.
• Ví dụ 8:
• class Vector
• {
• private:
• . . .
• public:
• . . .
• Vector(Vector &u); //Hàm thiết lập sao chép
• int GetN() { return n;}
• float Nhan(Vector u); //Tích vô hướng hai vector
• };
• //Định nghĩa các hàm thành phần
• Vector::Vector(Vector &u)
• { cout << “Goi ham thiet lap sao chep của lop Vector\n”;
• n = u.n;
• p = new int [n]; //Cấp phát vnđ bằng kích thước có trong đt nguồn
• //Gán nội dung vnđ trong đt nguồn sang đt đích
• for(int i =0; i < n; i++)
• p[i] = u.p[i];
• }
• float Vector::Nhan(Vector u)
• { float res = 0;
• for(i = 0; i < n; i++)
• res += p[i] * u.p[i];
• return res;
• }
• //Hàm chính
• void main()
• { . . .
• Vector c = a; //Gọi hàm thiết lập sao chép
• c.Xuat();
• if(b.GetN() == c.GetN()) {
• float kq = b.Nhan(c); //Gọi hàm thiết lập sao chép
• cout << “b * c = “ << kq <<“\n”;
• }
• else cout << “Khong thuc hien duoc phép toán b * c\n”;
• }
• Chú ý:
Việc truyền tham số đối tượng cho hàm bằng giá trị đòi hỏi phải có hàm
thiết lập sao chép. Ngoài ra nó cũng làm cho chương trình chạy chậm vì
phải mất thời gian sao chép một lượng lớn dữ liệu. Do đó người ta thường
chọn cách truyền tham chiếu.
Cần phân biệt giữa hàm thiếp lập sao chép và phép gán. Phép gán thực hiện
việc sao chép nội dung từ đối tượng này sang đối tượng khác, do vậy cả hai
đối tượng trong phép gán đều đã tồn tại. Ngược lại hàm thiết lập thực hiện
đồng thời hai nhiệm vụ: tạo đối tượng và thực hiện việc sao chép nội dung
từ một đối tượng đã có sang đối tượng mới tạo ra đó.
• Ví dụ 9:
• void main()
• {
• . . .
• b = a; //Phép gán
• Vector c = a; //Gọi tới hàm thiết lập sao chép
• }
Hàm bạn và lớp bạn
• Các hàm hay lớp nếu được khai báo là bạn của lớp thì có thể truy nhập tới
mọi thành phần private của lớp.
• Ví dụ 10: Xây dựng một hàm nhân MaTran với Vector
• class MaTran; //Khai báo trước lớp MaTran
• class Vector
• { . . .
• public:
• . . .
• //Khai báo hàm tự do Nhan() là bạn của lớp vector
• friend Vector Nhan(MaTran u, Vector v);
• };
• class MaTran
• { . . .
• public:
• . . .
• //Khai báo hàm tự do Nhan() là bạn của lớp MaTran
• friend Vector Nhan(MaTran u, vector v);
• };
• Vector Nhan(Matran u, Vector v); //Khai báo nguyên mẫu
• //Định nghĩa hàm tự do Nhan()
• Vector Nhan(MaTran u, Vector v)
• { Vector res(u.m);
• for(int i = 0; i < u.m; i++) {
• res.p[i] = 0;
• for(int j = 0; j < v.n; j++)
• res.p[i] += u.p[i][j] * v.p[j];
• }
• return res;
• }
• void main()
• { Vector a(3); a.Xuat();
• Matran b(2, 3); b.Xuat();
• if(b.GetN() == a.GetN()) {
• Vector c = Nhan(b, a); c.Xuat();
• }
• else cout << “b * a khong thuc hien duoc\n”;
• }
Hàm thiết lập và đối tượng thành phần
Một lớp có thành phần dữ liệu là đối tượng lớp khác được gọi là lớp bao
• Ví dụ 12:
• class tamGiac
• {
• private:
• Diem A, B, C;
• public:
• TamGiac(int x1, int y1, int x2, int y2, int x3, int y3);
• . . .
• };
• A, B, C là các đối tượng thành phần
• TamGiac là lớp bao
• Diem là lớp thành phần (của TamGiac)