I'm not sure if it's a bug of the GCC compiler or the intended behavior of noexcept
.
Consider the following example:
struct B {
B(int) noexcept { }
virtual void f() = 0;
};
struct D: public B {
using B::B;
D() noexcept(noexcept(D{42})): B{42} { }
void f() override { }
};
int main() {
B *b = new D{};
}
If the noexcept
is removed, it compiles.
Anyway, as it is in the example, I got this error from GCC v5.3.1:
test.cpp:8:31: error: invalid use of incomplete type ‘struct D’
D() noexcept(noexcept(D{42})): B{42} { }
^
As far as I know, struct D
is not an incomplete type, but inheriting constructors are involved in the statement and it looks like the compiler is actually considering the completeness of the base struct B
more than of D
.
Is that the intended behavior or is it legal code?
For the sake of clarity:
See this link to the bugzilla for the GCC compiler for further details.
Currently, the bug is still unconfirmed. I'll update the question as soon as possible.
Your code is legal, even though GCC claims otherwise. It takes offense at this funny-looking declaration:
The outermost
noexcept
is a noexcept specifier, stating thatD::D()
is noexcept if and only if its constant-expression argument evaluates to true. The innernoexcept
is a noexcept operator that checks at compile time whether its argument expression, which is not actually evaluated, throws no exceptions. BecauseD::D(int)
is noexcept (inherited from B), this should be true.cppreference.com explicitly notes that using the operator inside the specifier is allowed (emphasis added):
Now, the class should be considered complete within the noexcept specifier due to §9.2.2 of the Standard (bold emphasis added):
§15.4.1 defines an exception-specification as the following grammar:
So GCC should not reject your code.