一、构造函数特点:
1、构造函数也是函数,其函数名和类名相同 2、构造函数无返回值 3、构造函数可以重载 4、构造函数创建对象时自动调用 注:当设计一个类时,如果没有手动实现一个构造函数,那么编译器会自动生成一个无参的构造函数。
二、拷贝构造函数
1、拷贝构造函数定义
它只有一个参数,参数类型是本类的引用。 如果设计类的人不写拷贝构造函数,编译器会自动生成一个默认的拷贝构造函数。 例子:
#include<iostream>
using namespace std;
class Complex{
public:
double real, imag;
Complex(double r,double i){
real = r; imag = i;
}
Complex(const Complex & c){
real = c.real; imag = c.imag;
cout<<"Copy Constructor called"<<endl ;
}
};
int main(){
Complex cl(1, 2);
Complex c2 (cl);
cout<<c2.real<<","<<c2.imag;
return 0;
}
输出结果: Copy Constructor called 1,2
2、拷贝构造函数被调用的三种情况
(1)当用一个对象去初始化另一个对象时,会调用拷贝构造函数 例:
Complex c2(c1);
Complex c2 = c1;
(2)如果函数F的参数是类A的对象,那么当F被调用时,类A的拷贝构造函数将被调用。
#include<iostream>
using namespace std;
class A{
public:
A(){};
A(A & a){
cout<<"Copy constructor called"<<endl;
}
};
void F(A a){ }
int main(){
A a;
F(a);
return 0;
}
程序输出结果: Copy constructor called 这是因为F函数的形参a在初始化时调用了拷贝构造函数。 (3)如果函数的返回值是类A的对象,则函数返回时,类A的拷贝构造函数被调用
#include<iostream>
using namespace std;
class A {
public:
int v;
A(int n) { v = n; };
A(const A & a) {
v = a.v;
cout << "Copy constructor called" << endl;
}
};
A Func() {
A a(4);
return a;
}
int main() {
cout << Func().v << endl;
return 0;
}
程序的输出结果是: Copy constructor called 4
4、深拷贝与浅拷贝
(1)浅拷贝 浅拷贝只是对指针的拷贝,拷贝后两个指针指向同一内存空间。 浅拷贝后果 1、在进行堆空间释放时,会导致多次释放,即:调用析构函数时,会造成同一份资源析构两次。也就是同一块内存delete两次,会导致内存泄露。 2、浅拷贝两个指针指向同一块内存,任何一方变动都会影响 另一方 3、
int main()
{
Test t1;
Test t2 = t1;
t1.freeP();
t2.freeP();
return 0;
}
(2)深拷贝 深拷贝不但对指针进行拷贝,而且对指针指向的内容进行拷贝,经拷贝后的指针是指向两个不同地址的指针。 深拷贝实现
Test(const Test& other)
{
_x = other._x;
p = new int;
*p = *(other.p);
}
结论: 如果一个类成员变量没有指针,不需要申请堆空间时,那么直接使用默认的拷贝构造函数,反之,则要手动进行深拷贝实现。
三、析构函数
在C++中,为了更好在对象被销毁时,做好清理和释放的工作,引入析构函数。 特点: 1、析构函数名与类名相同 2、析构函数不能重载,一个类只能有一个析构函数 3、析构函数无参 4、析构函数在对象被销毁时,自动调用 5、每一个对象被销毁时,就会自动调用一次析构函数 6、如果类设计者没有实现析构函数,编译器会提供一个默认的析构函数。
参考: http://c.biancheng.net/view/151.html
|