Chào cả nhà
Trong bài viết này chúng ta sẽ đi tìm hiểu một hàm thành viên rất quan trọng của lớp đó chính là hàm khởi tạo và hàm huỷ.
Hàm khởi tạo (Constructor)
Hàm khởi tạo là gì?
Hàm khởi tạo là một hàm thành viên đặc biệt của một lớp. Nó sẽ được tự động gọi đến khi một đối tượng của lớp đó được khởi tạo.
Sự khác biệt giữa hàm tạo và hàm thành viên thông thường
Một hàm tạo sẽ khác những hàm thông thường ở những điểm sau:
- Có tên trùng với tên lớp
- Không có kiểu dữ liệu trả về ( kể cả kiểu
void
) - Tự động được gọi khi một đối tượng thuộc lớp được tạo ra
- Nếu chúng ta không khai báo một hàm tạo, trình biên dịch C++ sẽ tự động tạo một hàm tạo mặc định cho chúng ta (sẽ là hàm không có tham số nào và có phần thân trống).
Hàm tạo có thể rất hữu ích để thiết lập các giá trị khởi tạo cho các biến thành viên cụ thể.
Ví dụ đơn giản về hàm khởi tạo:
class sinhvien { private: string ten; int tuoi; public: sinhvien(); // Đây là hàm khởi tạo ~sinhvien(); };
Các loại hàm khởi tạo
Hàm khởi tạo về cơ bản sẽ được chia làm 3 loại:
- Hàm khởi tạo không tham số (Cũng có thể gọi là hàm tạo mặc định – Default Constructor )
- Hàm khởi tạo có tham số ( Parameterized Constructor )
- Hàm khởi tạo sao chép ( Copy Constructor )
Hàm khởi tạo không tham số ( Default Constructor )
Hàm tạo loại này sẽ không truyền vào bất kì một đối số nào
class sinhvien { private: string ten; int tuoi; public: sinhvien() { this->ten = ""; this->tuoi = 0; } ~sinhvien(); };
Như trong ví dụ trên, hàm tạo sinhvien() không hề có đối số nào được truyền vào.
Theo ý kiến riêng của mình thì thông thường trong hàm loại này mình sẽ gán cho tất cả các thuộc tính về giá trị mặc định.
Trong ví dụ trên:
- Thuộc tính ten thuộc kiểu string mình sẽ đưa về mặc định là một chuối rỗng
""
. - Thuộc tính tuoi thuộc kiểu int mình sẽ đưa về mặc định là 0.
Hàm khởi tạo có tham số ( Parameterized Constructor )
Với loại hàm tạo này ta có thể truyền đối số cho chúng. Thông thường, các đối số này giúp khởi tạo một đối tượng khi nó được tạo.
Để khai báo một hàm khởi tạo có tham số chỉ cần thêm các tham số vào nó giống như cách bạn thêm tham số bất kỳ hàm nào khác.Khi bạn xác định phần thân của hàm tạo, hãy sử dụng các tham số để khởi tạo đối tượng.
class sinhvien { private: string ten; int tuoi; public: sinhvien(string param_ten, int param_tuoi) { this->ten = param_ten; this->tuoi = param_tuoi; } ~sinhvien(); };
Sau khi khai báo hàm trong lớp, ta có thể dễ dàng dùng nó bằng cách truyền tham số trong khi khởi tạo đối tượng.
int main() { sinhvien obj("lap Trinh Khong Kho", 5); // Ta truyền luôn tham số trong khi khới tạo đối tượng }
Lưu ý:
- Khi một đối tượng được khai báo trong hàm khởi tạo có tham số, các giá trị ban đầu phải được truyền dưới dạng đối số cho hàm tạo.
- Cách khai báo đối tượng bình thường có thể sẽ gây lỗi.
Điều này có nghĩa là bình thường để khai báo một đối tượng bạn sẽ khai báo bằng cú pháp:
sinhvien obj;
Nhưng do hàm khởi tạo là hàm có tham số nên cú pháp sẽ phải là:
sinhvien obj("Lap Trinh Khong Kho", 5);
- Các hàm khởi tạo có thể được gọi một cách rõ ràng hoặc ngầm định.
sinhvien obj = sinhvien("Lap Trinh Khong Kho", 5); // Đây là cách rõ ràng sinhvien obj("Lap Trinh Khong Kho", 5); // Đây là cách ngầm định
Nhưng thông thường để tiết kiệm code thì chúng ta hay sử dụng các ngầm định hơn.
Công dụng của hàm khởi tạo có tham số
- Nó được sử dụng để khởi tạo các thành phần dữ liệu khác nhau của các đối tượng khác nhau với các giá trị khác nhau khi chúng được tạo.
- Nó được sử dụng để nạp chồng các hàm khởi tạo.
Nạp chồng? Có thể hiểu đơn giản là ta sẽ có nhiều hơn một hàm khởi tạo trong cùng một lớp. Và phần này thì sẽ được mình trình bài trong bài sau nhé.
Hàm khởi tạo sao chép ( Copy Constructor )
Hàm khởi tạo sao chép là gì?
Hàm khởi tạo sao chép là một hàm tạo mà tạo một đối tượng bằng việc khởi tạo nó với một đối tượng của cùng lớp đó, mà đã được tạo trước đó.
Một hàm khởi tạo sao chép sẽ có nguyên mẫu chung như sau:
ClassName(const ClassName &old_obj) { // Code }
Trong đó Classname
là tên của lớp, old_obj
là đối tượng cũ sẽ lấy làm gốc để sao chép sang đối tượng mới
Ví dụ đơn giản về hàm khởi tạo sao chép:
/* Code by KingNNT */ #include <bits/stdc++.h> using namespace std; class Point { private: int x, y; public: Point(int x1, int y1) { x = x1; y = y1; } // Hàm khởi tạo sao chép Point(const Point &p2) { x = p2.x; y = p2.y; } int getX() { return x; } int getY() { return y; } }; int main() { Point p1(10, 15); // Hàm khởi tạo có tham số thông thường Point p2 = p1; // hàm khởi tạo sao chép được gọi ở đây cout << "p1.x = " << p1.getX() << ", p1.y = " << p1.getY() << endl; cout << "p2.x = " << p2.getX() << ", p2.y = " << p2.getY() << endl; return 0; }
Sau khi chạy chương trình ta sẽ có kết quả:
p1.x = 10, p1.y = 15 p2.x = 10, p2.y = 15
Một hàm khởi tạo sao chép sẽ được gọi khi nào?
Hàm khởi tạo sao chép sẽ được gọi khi:
- Khi một đối tượng của lớp được trả về bằng một giá trị.
- Khi một đối tượng của lớp được truyền đối số dưới dạng tham số của một hàm.
- Khi một đối tượng được tạo ra dựa trên một đối tượng khác cùng lớp.
- Khi trình biên dịch tạo một đối tượng tạm thời.
Tuy nhiên trên thực tế thì không chắc chắn rằng hàm khởi tạo sao chép sẽ được gọi trong tất cả 4 trường hợp ở phía trên. Vì C++ tiêu chuẩn sẽ cho phép trình biên dịch tối ưu hoá bản sao trong một số trường hợp nhất định.
Một ví dụ cho điều này là: Ví dụ về tối ưu hoá giá trị trả về ( Có thể gọi tắt là RVO). Xem tại đây
Lưu ý:
Nếu một hàm tạo sao chép không được định nghĩa trong một lớp, trình biên dịch sẽ tự nó định nghĩa nó. Vì thế phải thật lưu ý nếu lớp có các biến con trỏ hoặc có sử dụng cấp phát bộ nhớ động thì nên viết lại hàm.
Chia sẻ nhỏ một chút là mình đã từng mắc lỗi tại đây do khi sử dụng cấp phát bộ nhớ động mà không viết lại hàm khởi tạo sao chép do đó dẫn đến việc truy cập sai ô nhớ.
Hàm huỷ (Deconstructor)
Hàm huỷ là gì?
Hàm huỷ cũng là một hàm thành viên đặc biệt giống như hàm tạo, nó được dùng để phá huỷ hoặc xoá một đối tượng trong lớp.
Hàm huỷ sẽ được gọi khi nào?
Hàm hủy được gọi tự động khi một đối tượng thoát khỏi phạm vi của nó (Scope):
- Một chức năng kết thúc.
- Chương trình kết thúc.
- Một khối chứa các biến cục bộ kết thúc.
- Một toán tử
delete
được gọi
Hàm huỷ khác những hàm thành viên bình thường ở đâu?
- Cũng giống với hàm tạo, hàm huỷ có tên trùng với tên của lớp, nhưng điểm khác biệt ở đây là sẽ có thêm
~
ở đầu. - Hàm huỷ là một hàm không có đối số truyền vào, và cũng không trả về giá trị ( kể cả
void
)
class sinhvien { private: string ten; int tuoi; public: sinhvien() { this->ten = ""; this->tuoi = 0; } ~sinhvien() // Đây là hảm huỷ { this->ten = ""; this->tuoi = 0; } }
Có thể có nhiều hơn một hàm huỷ ở trong cùng một lớp không?
Câu trả lời ở đây là không nhé.
Khác với hảm khởi tạo, hàm huỷ có thể có một và chỉ một mà thôi.
Khi nào thì ta cần tự định nghĩa một hàm huỷ?
Với C++ thì nếu ta không khai báo một hàm huỷ, trình biên dịch cũng sẽ tự định nghĩa một hàm huỷ. Thông thường thì hàm huỷ này hoạt động khá tốt, nhưng khi bài toán có sử dụng con trỏ, hoặc cấp phát bộ nhớ động thì ban nên khai báo một hàm huỷ riêng để tránh rỏ rỉ bộ nhớ.
Nhưng với bản thân mình thông thường mình vẫn sẽ khai báo một hàm huỷ cho dù có dùng con trỏ hay cấp phát động hay không, và trong hàm huỷ đó mình sẽ đưa các thuộc tính của lớp về giá trị mặc định ( giống với hàm khởi tạo không tham số ).
Một hàm huỷ có thể là một hàm ảo hay không ?
Tất nhiên là có rồi. Và bạn có thể xem chi tiết: Tại đây
Bài viết của mình xin được kết thúc tại đây. Mọi thể comment ở bên dưới nếu thấy bất cứ điều gì không chính xác, hoặc đơn giản là muốn chia sẻ thêm những kiến thức tới mọi người. Cảm ơn mọi người. Chào tạm biệt và hẹn gặp lại !
Để lại một bình luận