有如下一个简单的复数类:
classClxComplex
{
public:
ClxComplex(doubledReal=0.0,doubledImage=0.0){m_dReal=dReal;dImage=dImage;}
doubleGetReal()const{returnm_dReal;}
doubleGetImage()const{returnm_dImage;}
private:
doublem_dReal;
doublem_dImage;
};
我们知道,下面的3行代码是等价的:
ClxComplexlxTest=2.0;
ClxComplexlxTest=ClxComplex(2.0);
ClxComplexlxTest=ClxComplex(2.0,0.0);
其实,对于前两行来说,编译器都是把它们转换成第3行的代码来实现的。因为我们写了构造函数,编译器就按照我们的构造函数来进行隐式转换,直接把一个 double数值隐式转换成了一个ClxComplex的对象。可是,有些时候,我们不希望进行隐式转换,或者隐式转换会造成错误。比如下面的一个简化的 字符串类:
classClxString
{
public:
ClxString(intiLength);
ClxString(constchar*pString);
~ClxString();
private:
char*m_pString;
};
ClxString::ClxString(intiLength)
{
if(iLength>0)
m_pString=newchar[iLength];
}
ClxString::ClxString(constchar*pString)
{
m_pString=newchar[strlen(pString)];
strcpy(m_pString,pString);
}
ClxString::~ClxString()
{
if(m_pString!=NULL)
deletem_pString;
}
我们可以用字符串的长度来初始化一个ClxString的对象,但是我们却不希望看到下面的代码:
ClxStringlxTest=13;//等同于ClxStringlxTest=ClxString(13);
这会给阅读代码造成不必要的歧义。
还有,我们知道下面的代码是用字符串A来初始化一个ClxString的对象:
ClxStringlxTest="A";//等同于ClxStringlxTest=ClxString("A");
可是,如果有人写成:
ClxStringlxTest='A';//等同于ClxStringlxTest=ClxString(65);
那上面的代码就会初始化一个长度为65(字母A的ASCII码值,在C和C++中,字符是以ASCII值存储的)的字符串。
当然,上面的情况都不是我们希望看到的。在这个时候我们就要用到显示构造函数了。
将构造函数声明成explicit就可以防止隐式转换。
下面是使用显示构造函数的ClxString:
classClxString
{
public:
explicitClxString(intiLength);
ClxString(constchar*pString);
~ClxString();
private:
char*m_pString;
};
在这种情况下,要想用字符串的长度来初始化一个ClxString对象,那就必须显示的调用构造函数:
ClxStringlxTest=ClxString(13);
而下面这些代码将不能通过编译。
ClxStringlxTest=13;
ClxStringlxTest='A';
分享到:
相关推荐
在 C++ 中,如果某个类型未声明它本身,则编译器将自动为该类型生成默认构造函数、复制构造函数、复制赋值运算符和析构函数。这些函数称为特殊成员函数,它们使 C++ 中的简单用户定义类型的行为如同 C 中的结构。也...
C++构造函数详解及显式调用构造函数.doc
面向对象程序设计,即C++语言,类。构造函数的显示调用+构造函数的重载,文件里面有详细的注释。
C++中拷贝构造函数的定义 有一个参数的类型是其类类型的构造函数是为拷贝构造函数。如下: X::X( const X& x); Y::Y( const Y& y, int =0 ); //可以是多参数形式,但其第二个即后继参数都有一个默认值 ...
下面小编就为大家带来一篇浅谈C++ Explicit Constructors(显式构造函数)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
。。。
一、C++中拷贝构造函数的定义: 有一个参数的类型是其类类型的构造函数是为拷贝构造函数。 如下所示: X::X( const X& x); Y::Y( const Y& y, int =0 ); //可以是多参数形式,但其第二个即后继参数都有一个默认值 ...
构造函数初始化列表以一个冒号开始,接着是以逗号分隔的数据成员列表,每个数据成员后面跟一个放在括号中的初始化式。例如: class CExample { public: int a; float b; //构造函数初始化列表 CExample(): a(0...
如果没有显式的构造函数,编译器会给一个默认的构造函数,并且该默认的构造函数仅仅在没有显式地声明构造函数情况下创建。 构造原则如下: 1. 如果子类没有定义构造方法,则调用父类的无参数的构造方法。 ...
复制构造函数和复制赋值运算符 从 C++ 11 中开始,该语言支持两种类型的分配:复制赋值和移动赋值。 在本文中,“赋值”意味着复制赋值,除非有其他显式声明。 赋值操作和初始化操作都会导致对象被复制。 赋值:在将...
如果在类中没有显式地声明构造函数,那么编译器会自动创建一个默认的构造函数;并且这个默认的构造函数仅仅在没有显式地声明构造函数的情况下才会被创建创建。 构造函数与父类的其它成员(成员变量和成员方法)不同...
构造与析构在C++中,有一种特殊的成员函数,它的名字和类名相同,没有返回值,不需要用户显式调用(用户也不能调用),而是在创建对象时自动执行。这种特殊的成员函数就是构造函数(Constructor)。 初始化对象的...
15.4.5 构造函数和析构函数中的虚函数 497 15.5 继承情况下的类作用域 497 15.5.1 名字查找在编译时发生 498 15.5.2 名字冲突与继承 498 15.5.3 作用域与成员函数 499 15.5.4 虚函数与作用域 500 15.6 纯虚函数 502 ...
3.3. 显式构造函数 3.4. 拷贝构造函数 3.5. 结构体 VS. 类 3.6. 继承 3.7. 多重继承 3.8. 接口 3.9. 运算符重载 3.10. 存取控制 3.11. 声明顺序 3.12. 编写简短函数 译者 (YuleFox) 笔记 4. 来自 Google ...
标准C++中的字符串类取代了C标准C函数库头文件中的字符数组处理函数。 C++中用来做控制态输入输出的iostream类库替代了标准C中的stdio函数库。 C++中的try/catch/throw异常处理机制取代了标准C中的setjmp()和...
而且,C# 结构不支持继承,也不支持显式默认构造函数(默认情况下提供一个)。 2、数组:在 C++ 中,数组只是一个指针。在 C# 中,数组是包含方法和属性的对象。例如,可通过 Length 属性查询数组的大小。C# 数组还...
7.1.2 构造函数的显式调用 183 7.1.3 类类型成员变量 190 7.2 其他工具 193 7.2.1 const参数修饰词 193 7.2.2 内联函数 198 7.2.3 静态成员 199 7.2.4 嵌套类和局部类定义 201 7.3 向量—标准模板库预览 202...