• C++中的显式构造函数


     有如下一个简单的复数类:

    class ClxComplex
    {
    public:
        ClxComplex(double dReal = 0.0, double dImage = 0.0) { m_dReal = dReal; dImage = dImage; }
    
        double GetReal() const { return m_dReal; }
        double GetImage() const { return m_dImage; }
    
    private:
        double m_dReal;
        double m_dImage;
    };

        我们知道,下面的3行代码是等价的:

    ClxComplex lxTest = 2.0;
    ClxComplex lxTest = ClxComplex(2.0);
    ClxComplex lxTest = ClxComplex(2.0, 0.0);

        其实,对于前两行来说,编译器都是把它们转换成第3行的代码来实现的。因为我们写了构造函数,编译器就按照我们的构造函数来进行隐式转换,直接把一个double数值隐式转换成了一个ClxComplex的对象。可是,有些时候,我们不希望进行隐式转换,或者隐式转换会造成错误。比如下面的一个简化的字符串类:

    class ClxString
    {
    public:
        ClxString(int iLength);
        ClxString(const char *pString);
        ~ClxString();
    
    private:
        char *m_pString;
    };
    
    ClxString::ClxString(int iLength)
    {
        if (iLength > 0)
            m_pString = new char[iLength];
    }
    
    ClxString::ClxString(const char *pString)
    {
        m_pString = new char[strlen(pString)];
        strcpy(m_pString, pString);
    }
    
    ClxString::~ClxString()
    {
        if (m_pString != NULL)
            delete m_pString;
    }

        我们可以用字符串的长度来初始化一个ClxString的对象,但是我们却不希望看到下面的代码:

    ClxString lxTest = 13;  // 等同于ClxString lxTest = ClxString(13);
        这会给阅读代码造成不必要的歧义。

        还有,我们知道下面的代码是用字符串A来初始化一个ClxString的对象:

    ClxString lxTest = "A";  // 等同于ClxString lxTest = ClxString("A");
        可是,如果有人写成:
    ClxString lxTest = 'A';  // 等同于ClxString lxTest = ClxString(65);
        那上面的代码就会初始化一个长度为65(字母A的ASCII码值,在C和C++中,字符是以ASCII值存储的)的字符串。

        当然,上面的情况都不是我们希望看到的。在这个时候我们就要用到显示构造函数了。
        将构造函数声明成explicit就可以防止隐式转换。
        下面是使用显示构造函数的ClxString:

    class ClxString
    {
    public:
        explicit ClxString(int iLength);
        ClxString(const char *pString);
        ~ClxString();
    
    private:
        char *m_pString;
    };

        在这种情况下,要想用字符串的长度来初始化一个ClxString对象,那就必须显示的调用构造函数:

    ClxString lxTest = ClxString(13);
        而下面这些代码将不能通过编译。
    ClxString lxTest = 13; 
    ClxString lxTest = 'A';
  • 相关阅读:
    开发进度二
    开发进度一
    大道至简阅读笔记01
    用户模板和用户评价
    第九周总结
    第八周总结
    NABCD项目分析
    第七周总结
    第六周总结
    构建之法阅读笔记03
  • 原文地址:https://www.cnblogs.com/weekbo/p/9620706.html
Copyright © 2020-2023  润新知