How can a compiler generated default constructor be more efficient than a self-written one that does nothing but initialize members?

后端 未结 2 1101
野性不改
野性不改 2021-02-20 03:49

Triggered by this answer I was reading in the core guidelines:

C.45: Don’t define a default constructor that only initializes data members; use in-class memb

2条回答
  •  猫巷女王i
    2021-02-20 03:57

    I think that it's important to assume that C.45 refers to constants (example and enforcement):

    Example, bad

    class X1 { // BAD: doesn't use member initializers
        string s;
        int i; public:
        X1() :s{"default"}, i{1} { }
        // ... };
    

    Example

     class X2 {
        string s = "default";
        int i = 1; public:
        // use compiler-generated default constructor
        // ... };
    

    Enforcement

    (Simple) A default constructor should do more than just initialize member variables with constants.

    With that in mind, it's easier to justify (via C.48) why we should prefer in-class initializers to member initializers in constructors for constants:

    C.48: Prefer in-class initializers to member initializers in constructors for constant initializers

    Reason

    Makes it explicit that the same value is expected to be used in all constructors. Avoids repetition. Avoids maintenance problems. It leads to the shortest and most efficient code.

    Example, bad

    class X {   // BAD
        int i;         string s;
        int j; public:
        X() :i{666}, s{"qqq"} { }   // j is uninitialized
        X(int ii) :i{ii} {}         // s is "" and j is uninitialized
        // ... };
    

    How would a maintainer know whether j was deliberately uninitialized (probably a poor idea anyway) and whether it was intentional to give s the default value "" in one case and qqq in another (almost certainly a bug)? The problem with j (forgetting to initialize a member) often happens when a new member is added to an existing class.

    Example

    class X2 {
        int i {666};
        string s {"qqq"};
        int j {0}; public:
        X2() = default;        // all members are initialized to their defaults
        X2(int ii) :i{ii} {}   // s and j initialized to their defaults
        // ... };
    

    Alternative: We can get part of the benefits from default arguments to constructors, and that is not uncommon in older code. However, that is less explicit, causes more arguments to be passed, and is repetitive when there is more than one constructor:

    class X3 {   // BAD: inexplicit, argument passing overhead
        int i;
        string s;
        int j; public:
        X3(int ii = 666, const string& ss = "qqq", int jj = 0)
            :i{ii}, s{ss}, j{jj} { }   // all members are initialized to their defaults
        // ... };
    

    Enforcement

    (Simple) Every constructor should initialize every member variable (either explicitly, via a delegating ctor call or via default
    

    construction). (Simple) Default arguments to constructors suggest an in-class initializer may be more appropriate.

提交回复
热议问题