Tests show a strange behavior (c++14
, g++ 4.9.1
, clang 3.5.5
):
To sum it up:
- if
B
provides no other constructors it can useA::A()
- if
B
provides other constructors it cannot useA::A()
, but it usesA::A(whatever arguments)
, which is unexpected behavior (to me at least).
Setup 1:
struct A {
A() {};
A(int) {}; // with or without this overload the result are the same
};
struct B : A {
using A::A;
};
B b0{}; // OK
Setup 2:
struct A {
A() {}; // with a default constructor instead (empty class A)
// the results are the same
};
struct B : A {
using A::A;
B(int){}
};
B b0{}; // no matching constructor
B b1{24}; // OK
Setup 3:
struct A {
A() {};
A(int) {};
};
struct B : A {
using A::A;
B(int, int){}
};
B b0{}; // no matching constructor
B b1{24}; // OK
B b2{24, 42}; // OK
Why is this happening and how can it be "fixed".
I can't tell you the rationale for this, but I can at least tell you that it is standard-mandated:
Since the default
B()
invokes the defaultA()
, you can "fix" it like this:(live demo)
The following wording from the original proposal (n2540) suggests that this fix's ease and symmetry is more or less the driving factor behind the decision, though I still find that to be somewhat unsatisfactory. Oh well.