As far as I know, in standard C++11 (not C++14), when omitting the return type of a lambda, its return type is deduced to be:
void
in all other cases.Consider now this code:
#include <iostream>
auto closure = [](int x)
{
x++;
return x;
};
int main()
{
int y = closure(10);
std::cout << y << std::endl;
}
This should fall under case 2., however the code compiles as if were C++14 with auto
type deduction, in both g++4.9.2, g++5 and clang++, with -pedantic -Wall -Wextra -std=c++11
. What's going on here? Am I interpreting the standard wrong?
The type and return type of the lambdas are automatically inferred.
If Lambda encounters an error, it returns an exception type, message, and HTTP status code that indicates the cause of the error.
If a return value isn't required, declare the function to have void return type. If a return type isn't specified, the C compiler assumes a default return type of int . Many programmers use parentheses to enclose the expression argument of the return statement.
What is the return type of lambda expression? Explanation: Lambda expression enables us to pass functionality as an argument to another method, such as what action should be taken when someone clicks a button.
Your code is being accepted without any warnings because the original C++11 restriction is considered a defect in the standard, which allows implementations to fix the behavior. See CWG DR975, DR1048 and N3638.
975. Restrictions on return type deduction for lambdas
[Moved to DR status at the April, 2013 meeting as part of paper N3638.]
There does not appear to be any technical difficulty that would require the current restriction that the return type of a lambda can be deduced only if the body of the lambda consists of a single return statement. In particular, multiple return statements could be permitted if they all return the same type.
1048. auto deduction and lambda return type deduction.
...
Notes from the November, 2014 meeting:
CWG agreed that the change embodied in paper N3638 should be considered to have been a DR against C++11.
In summary, DR975 proposed modifying the rules for return type deduction for lambda expressions to allow multiple return statements.
DR1048 identifies a discrepancy where the rules for deducing the return type for normal functions using the placeholder type auto
differs slightly from the rules proposed in DR975. Specifically, return type deduction for normal functions would discard top-level cv-qualifiers in all cases, where as those for lambda expressions would preserve cv-qualifiers for class types.
N3638 resolves this issue, amongst others.
I doubt there's any way to revert to the original behavior short of finding a compiler version that shipped with C++11 lambda support prior to the implementation of the DR above.
Some C++14 rules are available in C++11 mode, when the compiler writers considered it too complicated to implement both rules at once.
This is what I find in the C++ Draft Standard N3337:
If a lambda-expression does not include a lambda-declarator, it is as if the lambda-declarator were (). If a lambda-expression does not include a trailing-return-type, it is as if the trailing-return-type denotes the following type:
— if the compound-statement is of the form
{ attribute-specifier-seqopt
return
expression ; }the type of the returned expression after lvalue-to-rvalue conversion (4.1), array-to-pointer conversion (4.2), and function-to-pointer conversion (4.3);
— otherwise,
void
.[ Example:
auto x1 = [](int i){ return i; }; // OK: return type is int auto x2 = []{ return { 1, 2 }; }; // error: the return type is void (a // braced-init-list is not an expression)
— end example ]
The standard seems to indicate that:
Then the return type is deduced from the expression. Otherwise the return type is void
.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With