关键字explicit
修饰构造方法的关键字,加上了,就告诉编译器,不可以隐式初始化对象;不加就可以隐式初始化对象;
下面的代码是可以正常编译执行的,但是加了关键字explicit,编译就会错我,因为Test t = 100;是隐式初始化对象,但是如果加上强制类型转换后,就不会有错误了。
强制类型转换:Test t = (Test)100;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
class Test{
public :
Test( int d):data(d){ //explicit
cout << "C:" << this << endl;
}
}
int main(){
Test t = 100;
}
拷贝构造函数如果加上了 explicit ,下面的语句就无法编译通过;不加可以。
class Test{
public :
//拷贝构造函数
explicit Test( const Test &t){
data = t.data;
}
int getData(){
return data;
}
private :
int data;
};
void test(Test x){
}
int main(){
Test t2(t1); //调用拷贝构造函数
//Test t3 = t2;//编译不过
//test(t2);//编译不过
}
|
触发拷贝构造函数的4种方式
1,Test t2(t1);//调用拷贝构造函数
2,声明的同时就赋值Test t3 = t2会调用拷贝构造函数;但是注意下面这种不会调用拷贝构造函数。
Test t3;
t3 = t2;//会调用=的重载方法
3,方法的参数是对象类型test(t2);
4,方法的返回值是对象类型。原因:对象tmp在方法结束后就被释放掉了,要返回到函数外,必须要复制tmp.
但是用gdb看了一下在return处并没有调用拷贝构造函数,所以test方法结束后,tmp也没有被释放,调用test方法的t5的内存地址和tmp是一样的。个人猜测:老版本的gcc编译器可能会在return处调用拷贝构造函数,但是新的编译器(gcc 4.8.5-20)为了提高效率,避免了一次多余的拷贝。
1
2
3
4
5
6
7
|
void test(Test x){ //进入函数的时点会调用拷贝构造函数
int value;
value = x.getData();
Test tmp(value);
return tmp; //return的时点会调用拷贝构造函数
}
Test t5 = test(t1);
|
原文链接:https://www.cnblogs.com/xiaoshiwang/archive/2018/08/17/9491296.html