1. 隐式转换和拷贝构造之间的一点实例分析
#include <stdio.h> #include <iostream> #include <vector> #include <map> using namespace std; class A { public: A(int i): m_a(i) { printf("constructor\n"); } A(const A &a) { printf("copy constructor\n"); m_a = a.m_a; } protected: int m_a; }; int _tmain(int argc, _TCHAR* argv[]) { A a = 1; getchar(); return 0; }运行结果是:
constructor分析:从以上结果可以发现通过隐式转换A a = 1相当于执行了A a(1)。这与我预测的先把1转换为一个A类型的临时变量调用一次构造函数,然后临时变量再赋值给a调用一次拷贝构造函数的想法不一样,出现这种结果应该是编译器编译的时候发现存在隐式转换于是直接把A a = 1处理成A a(1)了,这样可以省掉一些中间不必要的开销。
2. 重载的赋值操作符、拷贝构造函数与隐式转换
#include <stdio.h> class A { public: A(int i): m_a(i) { printf("constructor\n"); } A(const A &a) { printf("copy constructor\n"); m_a = a.m_a; } A& operator = (const A& a) { printf("operator =\n"); this->m_a = a.m_a; return *this; } protected: int m_a; }; int _tmain(int argc, _TCHAR* argv[]) { A a(1); a = 1; getchar(); return 0; }
运行结果如下:
constructor constructor operator =分析:首先是定义a的时候调用了一次构造函数,第二次构造函数是由于 a = 1的时候因为隐式转换把1转换为A类型的临时变量导致的,最后是赋值操作
3. 如何避免隐式转换
#include <stdio.h> class A { public: A(int i): m_a(i) { printf("constructor\n"); } A(const A &a) { printf("copy constructor\n"); m_a = a.m_a; } protected: int m_a; }; int _tmain(int argc, _TCHAR* argv[]) { A a = 1; getchar(); return 0; }
分析:通过给引发隐式转换的构造函数加上explicit关键字来阻止隐式转换的发生,上面代码编译不能通过:error C2440: “初始化”: 无法从“int”转换为“A”。
4.重载+操作符:
#include <stdio.h> class A { public: explicit A(int i) : m_a(i) { printf("constructor\n"); } A(const A &a) { printf("copy constructor\n"); m_a = a.m_a; } A& operator = (const A& a) { printf("operator =\n"); this->m_a = a.m_a; return *this; } A operator + (const A& a) { printf("Generic\n"); return A(this->m_a + a.m_a); } A operator + (const int a) { printf("int\n"); return A(this->m_a + a); } double operator + (const double a) { printf("double\n"); return this->m_a + a; } protected: int m_a; }; int _tmain(int argc, _TCHAR* argv[]) { A a(1); printf("\n"); A b = a + 1; printf("\n"); double c = a + 2.0; //出错:double c = a + 2; printf("\n"); A d = a + b; getchar(); return 0; }
运行结果如下:
constructor int constructor double Generic constructor分析:从以上结果可知,类A可以重载多个operator + 操作,传入参数和返回值可以根据自己的需要去设置。