为什么要调用错误的ctor?

boa*_*der 4 c++ constructor explicit

我的代码按预期工作:

EscapedString es("Abc&def");
EscapedString es2("");
es2 = es;     // es2 == Abc%26def
Run Code Online (Sandbox Code Playgroud)

并且代码不能按预期工作:

EscapedString es("Abc&def");
EscapedString es2 = es;    // es == Abc%2526def
Run Code Online (Sandbox Code Playgroud)

在第二种情况下,即使es是EscapedString,也会调用CTOR2而不是CTOR3.

EscapedString es(EscapedString("Abc?def"));
Run Code Online (Sandbox Code Playgroud)

是对的,但我似乎无法在CTOR3上设置断点,所以我不确定它是否正常工作,或者代码已被优化掉或者它是意外工作的.

课程如下:

class EscapedString : public std::string {
public:
    EscapedString(const char *szUnEscaped) {  // CTOR1
        *this = szUnEscaped;
    }

    EscapedString(const std::string &strUnEscaped) {   // CTOR2
        *this = strUnEscaped;
    }

    explicit EscapedString(const EscapedString &strEscaped) {   // CTOR3
        *this = strEscaped;  // Can't set breakpoint here
    }

    EscapedString &operator=(const std::string &strUnEscaped) {
        char *szEscaped = curl_easy_escape(NULL, strUnEscaped.c_str(), strUnEscaped.length());
        this->assign(szEscaped);
        curl_free(szEscaped);
        return *this;
    }
    EscapedString &operator=(const char *szUnEscaped) {
        char *szEscaped = curl_easy_escape(NULL, szUnEscaped, strlen(szUnEscaped));
        this->assign(szEscaped);
        curl_free(szEscaped);
        return *this;
    }
    EscapedString &operator=(const EscapedString &strEscaped) {
        // Don't re-escape the escaped value
        this->assign(static_cast<const std::string &>(strEscaped));
        return *this;
    }
};
Run Code Online (Sandbox Code Playgroud)

Joh*_*ing 10

通常,EscapedString es2 = es;将调用复制构造函数,但是您明确告诉它不要通过创建复制构造函数explicit:

explicit EscapedString(const EscapedString &strEscaped)
Run Code Online (Sandbox Code Playgroud)

标记的构造函数explicit永远不能通过自动类型转换来调用.它只能被明确地调用,你在这里做了:

EscapedString es(EscapedString("Abc?def"));
Run Code Online (Sandbox Code Playgroud)

这是编译器遇到的情况EscapedString es2 = es;.

首先,编译器会看到它是否可以使用复制构造函数并发现它不能,因为它已被标记explicit.所以它寻找另一个构造函数来调用.由于EscapedString派生自std::string,编译器能够转换esconst std::string&和调用:

EscapedString &operator=(const std::string &strUnEscaped)
Run Code Online (Sandbox Code Playgroud)