2013-02-22 3 views
5

У меня есть что-то вроде:Вложенные лямбда Захват в C++

// think of Synonym as a set/vector of values 
// the purpose of this function is to filter out elements from the 2 synonyms/sets, 
// that are not related (similar to SQL inner join) - modifier modifies vars 
void Clauses::modifies(Synonym& modifiers, Synonym& modifiedVars, UnaryPredicate isModifies) { 
    // filter out any modifiers that does not modify (is related to) any of the variables in modifiedVar (left join) 
    modifiers.removeIf([modifiedVars, &isModifies](int line) -> bool { 
     return modifiedVars.none([line, &isModifies](int v) -> bool { 
      return isModifies(line, v); 
     }); 
    }); 

    // filter out any candidate modifiedVars that is not modified by any modifiers (right join) 
    modifiedVars.removeIf([modifiers, &isModifies](int varIndex) -> bool { 
     return modifiers.none([varIndex, &isModifies](int line) -> bool { 
      return isModifies(line, varIndex); 
     }); 
    }); 

    // result is an something like an SQL inner join 
} 

Проблема заключается в Visual Studio жалуется, что:

Error 1 error C3480: 'PQL::Clauses::`anonymous-namespace'::<lambda1>::isModifies': a lambda capture variable must be from an enclosing function scope h:\dropbox\sch\cs3202\spa_cpp\spa\pql.cpp 78 
Error 2 error C2665: 'PQL::Clauses::`anonymous-namespace'::<lambda3>::<lambda3>' : none of the 2 overloads could convert all the argument types h:\dropbox\sch\cs3202\spa_cpp\spa\pql.cpp 78 
... 

Первоначально код не проходит предикаты/условия как ссылки, но где-то читал Я думал, что мне это нужно, но это ничего не меняет

modifiers.removeIf([modifiedVars, isModifies] ... 

UPDATE: Я использую VS2010 для этого проекта

+0

Кажется, это ошибка Visual Studio. GCC и Clang принимают такой захват. –

+0

Какая версия Visual Studio? [Вложенные функции лямбда теряют область действия] (http://connect.microsoft.com/VisualStudio/feedback/details/537366/c-nested-lambda-functions-lose-scope) выглядит как ошибка. –

+0

@JesseGood Ah snap, я просто понял, что меня нет дома, и здесь я использую VS2010, а не последнюю версию VS2012 CTP. –

ответ

5

Если вы используете Visual Studio 2010, ваш код может вызывать ошибку, которая не позволяет вам фиксировать переменную во вложенной лямбда.

Попробуйте использовать режим захвата по умолчанию (например, [&]) в качестве обходного пути.

Эта ошибка исправлена ​​в VS2012.

+0

Собственно, что означает '[&]'? Я вижу это в нескольких местах, но не понимаю, что это на самом деле делает. –

+1

Это означает, что захватить все по ссылке, а '=' означает захват всего по значению – ixSci

+0

Это означает, что вы неявно захватываете все объекты, на которые вы ссылаетесь в теле лямбда через ссылку. – MadScientist

5

Это, похоже, ошибка Visual C++, поскольку GCC и Clang принимают этот захват. Вот обходной путь:

modifiedVars.removeIf([modifiers, &isModifies](int varIndex) -> bool { 
    auto& isModifiesRedirect = isModifies; 
    return modifiers.none([varIndex, &isModifiesRedirect ](int line) -> bool { 
     return isModifiesRedirect (line, varIndex); 
    }); 

Примечание: Я мог только проверить это на VS2010. Он может быть исправлен в VS2012. Возможно, вы захотите рассмотреть возможность поиска в Microsoft Connect и отправку новой ошибки, если это уже не известная проблема.

Смежные вопросы