In §[except.throw]
, the standard says that throwing an exception copy-initializes the exception object from the throw expression
Throwing an exception copy-initializes (11.6, 15.8) a temporary object, called the exception object
Why then does the following code compile on C++17?
class Exception {
public:
Exception() = default;
Exception(Exception&&) = delete;
Exception(const Exception&) = delete;
};
int main() {
throw Exception{};
return 0;
}
(https://wandbox.org/permlink/R3WfzfnBAORTLVSy)
Copy initialization does not include any case (from what it seems like to me) that is eligible for prvalue elision. Why then does the above code compile in C++17?
From a recent draft standard
11.6 [dcl.init]/15
So
T x = T();
is an example of copy-initialization. So is throwing an exception and many other cases.What copy-initialization does is defined in other parts of 11.6 (along with other forms of initialization). The relevant section on prvalues in initializers is:
11.6 [dcl.init]/17.6.1
this also known as is guaranteed elision. If the initializer expression is a prvalue expression of matching type, the prvalue expression is used to directly construct the target of the initialization.