C++11 lambda as member variable?

后端 未结 5 1193
Happy的楠姐
Happy的楠姐 2021-01-31 07:24

Can lambda\'s be defined as class members?

For example, would it be possible to rewrite the code sample below using a lambda instead of a function object?



        
相关标签:
5条回答
  • 2021-01-31 07:51

    A lambda just makes a function object, so, yes, you can initialize a function member with a lambda. Here is an example:

    #include <functional>
    #include <cmath>
    
    struct Example {
    
      Example() {
        lambda = [](double x) { return int(std::round(x)); };
      };
    
      std::function<int(double)> lambda;
    
    };
    
    0 讨论(0)
  • 2021-01-31 08:01

    A bit late, but I have not seen this answer anywhere here. If the lambda has no capture arguments, then it can be implicitly cast to a pointer to a function with the same arguments and return types.

    For example, the following program compiles fine and does what you would expect:

    struct a {
        int (*func)(int, int);
    };
    
    int main()
    {
        a var;
        var.func = [](int a, int b) { return a+b; };
    }
    

    Of course, one of the main advantages of lambdas is the capture clause, and once you add that, then that trick will simply not work. Use std::function or a template, as answered above.

    0 讨论(0)
  • 2021-01-31 08:08

    "if a lambda can be passed as a function argument then maybe also as a member variable"

    The first is a yes, you can use template argument deduction or "auto" to do so. The second is probably no, since you need to know the type at declaration point and neither of the previous two tricks can be used for that.

    One that may work, but for which I don't know whether it will, is using decltype.

    0 讨论(0)
  • 2021-01-31 08:11
    #include <functional>
    
    struct Foo {
        std::function<void()> bar;
    };
    
    void hello(const std::string & name) {
        std::cout << "Hello " << name << "!" << std::endl;
    }
    
    int test_foo() {
        Foo f;
        f.bar = std::bind(hello, "John");
    
        // Alternatively: 
        f.bar = []() { hello("John"); };
        f.bar();
    }
    
    0 讨论(0)
  • 2021-01-31 08:12

    Templates make it possible without type erasure, but that's it:

    template<typename T>
    struct foo {
        T t;
    };
    
    template<typename T>
    foo<typename std::decay<T>::type>
    make_foo(T&& t)
    {
        return { std::forward<T>(t) };
    }
    
    // ...
    auto f = make_foo([] { return 42; });
    

    Repeating the arguments that everyone has already exposed: []{} is not a type, so you can't use it as e.g. a template parameter like you're trying. Using decltype is also iffy because every instance of a lambda expression is a notation for a separate closure object with a unique type. (e.g. the type of f above is not foo<decltype([] { return 42; })>.)

    0 讨论(0)
提交回复
热议问题