一、初始化列表和构造函数体初始化区别
- 对于基础类型,使用初始化列表初始化和在构造函数体内初始化是一样的。
{
public:
Date(int year, int month, int day) //
: _year(year)
, _month(month)
, _day(day)
{}
Date(int year, int month, int day) //效果一样
{
_year = year;
_month = month;
_day = day;
}
private:
int _year;
int _month;
int _day;
};
- 对于引用成员变量、const成员变量、必须使用初始化列表,否则编译器报错
因为编译器规则是:先调用父类构造函数、再初始化列表(同时创建对象),最后才是调用自身的构造函数。由于引用成员变量,const成员变量必须对象创建时就初始化,所以必须在初始化列表中; - 自定义类型成员两种初始化方式的区别
{
public:
A(){}
A(int value):value(value){}
private:
int value;
};
class B {
public:
// 方式一
B(int value,A a) {
value = value;
a = a;
}
// 方式二
B(int value,A a) : value(value),a(a){}
private:
A a;
int value
};
方式一自定义类型成员不在初始化列表中,它会先自动调用该类的默认构造函数初始化(如果没有实现默认构造函数,就会报错),然后又调用一次A的赋值构造函数进行赋值操作,相当于调用了两次A的构造函数。
方式二对于a来说只对a调用一次赋值构造函数。
所以对于类类型成员,初始化列表会减少性能开销。
二、QList等等容器引用指针时内存的释放
不管是Qt的QList、QMap、QVector容器,还是c++的vector、map等等容器,都不负责其引用指针的内存管理。看如下代码:
// 定义一个类
class MyClass {
public:
MyClass() {qDebug() << "构造函数"; }
~MyClass() { qDebug() << "析构函数"; }
};
// 测试函数
void test() {
QList<MyClass*> list;
for (int i = 0; i < 3; ++i) {
auto *myClass = new MyClass();
list.push_back(myClass);
}
// 会有内存泄漏
list.clear();
}
上面会造成内存泄漏。因为list.clear()指示将容器QList置空,并不会释放引用的内存。需要如下调用:
// 需要调用qDeleteAll()函数
qDeleteAll(list);
// 或者for循环逐个释放内存
// for (auto *myClass : list) {
// delete myClass;
// }
// 清空list
list.clear();