Следующий код, скомпилированный в VS2015, вызывает конструктор копирования Foo при захвате foo
void bar(Foo& foo) {
auto f = [foo]() {
};
}
Ранее, ошибочно полагал, что внутри лямбды, foo должна была остаться ссылкой, по аналогии с обычными функциями:
template
void bar(const Foo& foo) {
f
baz(Foo());
Не разобрался в спецификации до конца, правильно ли понял, что при захвате по значению будет создана константная переменная (с помощью конструктора копирования), доступная внутри лямбды, имеющая тип Foo, не зависимо от того является внешняя переменная ссылкой на Foo или экземпляром Foo
update
Что интересно, если вызывать не f
Ответ
Чтобы разобрать Ваш пример, нужно обратиться к стандарту и понимать, что для реализации лямбда-функций компилятор создаёт тип-замыкание. Так вот, когда в списке захвата, некий объект закхватывается по значению, тогда в замыкании создаётся объект такого же типа, какой имеет этот объект, если это не ссылка. Если же это ссылка, то типом в замыкании будет тип, на который эта ссылка ссылается.
Стандарт [expr.prim.lambda]p15:
An entity is captured by copy if it is implicitly captured and the
capture-default is = or if it is explicitly captured with a capture
that is not of the form & identifier or & identifier initializer. For
each entity captured by copy, an unnamed non-static data member is
declared in the closure type. The declaration order of these members
is unspecified. The type of such a data member is the type of the
corresponding captured entity if the entity is not a reference to an
object, or the referenced type otherwise. [ Note: If the captured
entity is a reference to a function, the corresponding data member is
also a reference to a function. — end note ] A member of an anonymous
union shall not be captured by copy
Таким образом, в замыкании Ваш foo будет иметь тип Foo
Комментариев нет:
Отправить комментарий