web-dev-qa-db-ja.com

C ++でクロージャはありますか?

私はネット上の閉鎖について読んでいた。 C++にクロージャー用の組み込み機能があるのか​​、それともC++でクロージャーを実装できる方法があるのか​​疑問に思いました。

47
Parvinder Singh
39
Apeirogon Prime

クロージャが、埋め込まれた永続的な隠された分離できないコンテキスト(メモリ、状態)を持つ関数への参照として理解している場合、はい:

class add_offset {
private:
    int offset;
public:
    add_offset(int _offset) : offset(_offset) {}
    int operator () (int x) { return x + offset; }
}

// make a closure
add_offset my_add_3_closure(3);

// use cloure
int x = 4;
int y = my_add_3_closure(x);
std::cout << y << std::endl;

次のものはその状態を変更します:

class summer
{
private:
    int sum;
public:
    summer() : sum(0) {}
    int operator () (int x) { return sum += x; }
}

// make a closure
summer adder;
// use closure
adder(3);
adder(4);
std::cout << adder(0) << std::endl;

内部状態は外部から参照(アクセス)できません。

定義方法に応じて、クロージャーには複数の関数への参照を含めることができます。また、2つのクロージャーは同じコンテキストを共有できます。つまり、2つの関数は同じ永続的な...状態を共有できます。

クロージャーとは、自由変数を含まないことを意味します-これは、プライベート属性とパブリックメソッドのみを持つクラスに匹敵します。

18
Zrin

はい、これは、ファンクターを使用せずに状態を持つ関数を実装する方法を示しています。

#include <iostream>
#include <functional>


std::function<int()> make_my_closure(int x){
    return [x]() mutable {   
        ++x;
        return x;   
    };
}

int main()
{
    auto my_f = make_my_closure(10);

    std::cout << my_f() << std::endl; // 11
    std::cout << my_f() << std::endl; // 12
    std::cout << my_f() << std::endl; // 13

     auto my_f1 = make_my_closure(1);

    std::cout << my_f1() << std::endl; // 2
    std::cout << my_f1() << std::endl; // 3
    std::cout << my_f1() << std::endl; // 4

    std::cout << my_f() << std::endl; // 14
}

未定義の動作を導入したmutableキーワードを忘れていました(clangバージョンはガベージ値を返していました)。実装されたとおり、クロージャは正常に動作します(GCCおよびclangで)

14
Setepenre

はい、C++ 11には lambdas という名前のクロージャーがあります。

C++ 03では、ラムダの組み込みサポートはありませんが、 Boost.Lambda 実装があります。

7
Rost

閉鎖の意味に依存すると思います。私がいつも使用してきた意味は、ある種のガベージコレクションを意味します(参照カウントを使用して実装できると思いますが)。参照をキャプチャして参照オブジェクトを保持する他の言語のラムダとは異なり、C++ラムダは値をキャプチャするか、参照されるオブジェクトがnotキープアライブです(そして、参照は簡単にぶら下がる可能性があります)。

7
James Kanze