Implicit constructor argument conversion in C++11

前端 未结 3 385
礼貌的吻别
礼貌的吻别 2021-01-21 07:57

Lets concider following code:

class A{
public:
  A(int x){}
};

class B{
public:
  B(A a){};
};


int main() {
  B b = 5;
  return 0;
}

And whi

3条回答
  •  不知归路
    2021-01-21 08:24

    Just to be clear:

    B b = 5;
    

    is "copy initialisation" not assignment. (See http://www.gotw.ca/gotw/036.htm).

    In this case, you are asking the compiler to perform two implicit user-defined conversions first, i.e. int -> A, A -> B before a temporary B object is passed to the copy constructor for B b. The compiler is allowed to elide the temporary object but semantically you are still asking the language to make two jumps across types.

    All implicit behaviour in programming languages is inherently scary. For the sake of a little syntactic sugar, we are asking c++ to "do some magic to make it just work". Unexpected type conversions can wreck havoc in large complex programmes. Otherwise, every time you wrote a new function or a new class, you would have to worry about all the other types and functions it could affect, with the side -effects rippling across your code. Would you really want implicit conversions from int -> apple -> horse -> horse_power -> aeroplane?

    For that reason, c++ only allows a single implicit user-defined conversion:

    12.3 Conversions [class.conv]

    1 Type conversions of class objects can be specified by constructors and by conversion functions. These conversions are called user-defined conversions and are used for implicit type conversions (clause 4), for initialization (8.5), and for explicit type conversions (5.4, 5.2.9).

    4 At most one user-defined conversion (constructor or conversion function) is implicitly applied to a single value.

    You are better off either with an explicit cast or "direct initialisation" both of which make it clear to the compiler and collaborators exactly what you are trying to do. Either the traditional or the new uniform initialisation syntax works:

    B b(5);
    B b{5};
    B b = {5};
    

提交回复
热议问题