2012-11-19 5 views
18

Il seguente programma non compilare:Lambda di un lambda: la funzione non viene catturata

#include <iostream> 
#include <vector> 
#include <functional> 
#include <algorithm> 
#include <cstdlib> 
#include <cmath> 

void asort(std::vector<double>& v, std::function<bool(double, double)> f) 
{ 
    std::sort(v.begin(), v.end(), [](double a, double b){return f(std::abs(a), std::abs(b));}); 
} 

int main() 
{ 
    std::vector<double> v({1.2, -1.3, 4.5, 2.3, -10.2, -3.4}); 
    for (unsigned int i = 0; i < v.size(); ++i) { 
     std::cout<<v[i]<<" "; 
    } 
    std::cout<<std::endl; 
    asort(v, [](double a, double b){return a < b;}); 
    for (unsigned int i = 0; i < v.size(); ++i) { 
     std::cout<<v[i]<<" "; 
    } 
    std::cout<<std::endl; 
    return 0; 
} 

perché:

error : 'f' is not captured 

Che cosa significa e come risolvere il problema?

risposta

37

Si utilizza il parametro f nella lambda all'interno di asort(), ma non lo si acquisisce. Prova ad aggiungere f all'elenco di acquisizione (modifica [] per leggere [&f]).

+7

O anche catturarlo per riferimento con '[&]'. – Cameron

+0

Sì, hai ragione, ho aggiornato la mia risposta. Per qualche ragione ho letto troppo velocemente e ho pensato che 'f' era un puntatore a funzione, non un oggetto funzione. – cdhowie

7

Si sta effettivamente facendo riferimento a f, che è una variabile nell'ambito esterno, nel proprio lambda. Dovresti catturarlo nella tua lista di cattura (il più semplice è probabilmente il riferimento [& f] o [&] per catturare tutto per riferimento, dato che lo stai usando immediatamente).

In un'altra nota, std :: function ha un sovraccarico mentre esegue la cancellazione dei tipi, nel tuo caso qui potrebbe essere meglio introdurre un tipo di modello.