web-dev-qa-db-ja.com

ラムダのラムダ:関数はキャプチャされません

次のプログラムはコンパイルされません。

#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;
}

なぜなら:

error : 'f' is not captured

それは何を意味し、どのように問題を解決しますか?

29
Vincent

asort()内のラムダでfパラメーターを使用しますが、キャプチャしません。 fをキャプチャリストに追加してみてください(変更[]を読む[&f])。

54
cdhowie

ラムダでは、外部スコープの変数であるfを効果的に参照しています。キャプチャリストでキャプチャする必要があります(最も単純なのは、参照[&f]を使用するか、[&]を使用してすぐに使用するため、すべてを参照でキャプチャすることです)。

別の注意として、std :: functionは型の消去を実行するため、オーバーヘッドがあります。ここでは、テンプレート型を導入する方が良い場合があります。

8
wendazhou