构造函数
- 在类中的public里面定义。
- 完成对象的初始化动作,主要用于创建对象时为对象的成员属性赋值,构造函数由编译器自动调用。
- 创建对象时调用构造函数和析构函数。
- 格式:函数名与类名相同,无返回类型,无void,可以有参数,可以重载。
构造函数显式调用
Stock food = Stock("World Cabbage", 250, 1.25);
构造函数隐式调用
Stock food("World Cabbage", 250, 1.25);
new构造函数
Stock *p= new Stock("World Cabbage", 250, 1.25);
对象的地址赋给指针p,对象无名称。
拷贝构造函数
- 格式:
类名称(const 类名称 &对象)
{
......
}
旧对象初始化新对象的时候,才会调用拷贝构造函数,如下面所示:
StringBad * p = new StringBad(motto);
/ calls StringBad(const StringBad &)
使用motto初始化一个匿名对象,并且将新对象的地址赋给p指针
下面不会调用拷贝构造函数
- 深拷贝:类中包含了使用new初始化的指针成员,应该定义一个拷贝构造函数,以复制指向的数据,而不是指针。
- 浅拷贝:逐个复制非静态成员,有指针时引发异常。
构造函数分类
按参数类型:
1.有参构造
2.无参构造
- 设计类时,通常应提供对所有类成员做隐式初始化的默认构造函数,即设计方法有:要么提供构造函数参数的默认值,要么进行重载。
- 例子:
Stock(const string & co = "Error", int n = 0, double pr = 0.0);
或
Stock::Stock() // default constructor
{
company = "no name";
shares = 0;
share_val = 0.0;
total_val = 0.0;
}
注意事项
Stock first("Concrete Conglomerate");
/ calls constructor调用接收参数的构造函数
Stock second();
/declares a function声明一个返回值类型为Stock的函数
Stock third;
/ calls default constructor隐式调用默认构造函数
Stock stock2 = Stock ("Boffo Objects", 2, 2.0);
/ 指定对象的初始化,可能会创建临时变量。
stock1 = Stock("Nifty Foods", 10, 50.0);
/ 赋值语句,在赋值语句中使用构造函数会导致在赋值前创建一个临时变量。
按类型分类:
1.普通构造函数
2.有参构造函数
explicit关键字
- 声明为explicit的构造函数不能在隐式转换中使用。例子:不能用:Data data = 10;而要用:Data data(10);
malloc,free
- malloc不会调用构造函数,free不会调用析构函数。
析构函数
- 在类中的public里面定义。
- 由编译器自动调用,在对象结束的时候完成清理工作,主要用于对象销毁前系统自动调用,执行一些清理工作。
- 函数名是在类名称前面加“~”组成,无返回类型,无void,无参数,无重载。
总结
- 对于构造函数:用户一般要实现:无参构造,有参构造,拷贝构造,析构。
- 类中使用指针时,在构造函数中申请空间,在析构函数中释放空间。
- 调用拷贝构造函数时,不会调用有/无参构造。
- 调用有参构造函数,不会屏蔽默认拷贝构造函数。
浅拷贝和深拷贝
- 浅拷贝:使两个对象的成员变量的值相同,但当类中有指针时,析构函数做释放内存时会导致错误。
- 深拷贝:当类中有指针时,要自定义拷贝构造函数,自行的给指针动态分配空间,然后析构函数做释放处理。