【问题标题】:Access to templated lambda from std::function in C++从 C++ 中的 std::function 访问模板化 lambda
【发布时间】:2020-05-06 06:28:30
【问题描述】:

我正在尝试向我的 lambda 添加一些检测以存储一些数据(即当 lambda 被触发时),但我在从 std::function 检索 lambda 时遇到了一些麻烦。 我创建了一个通用仿函数来存储任何 lambda 和我的参数,然后我将这个仿函数存储到 std::function 中。

当我尝试从 std::function 访问我的仿函数时,通过目标方法传递的类型不起作用...我传递的类型似乎不是好的类型,我不知道如何传递不是定义类型的 lambda 类型...

谁能帮助我实现这一目标? 我正在使用 Microsoft (R) C/C++ Optimizing Compiler Version 19.16.27030.1 for x64

谢谢

这是一个例子:

#include <iostream>
#include <functional>
#include <string>

template <typename Signature>
class lambda_wrapper
{
    public:
    lambda_wrapper(Signature&& lambda, const std::string& iName)
        : _lambda(std::move(lambda))
        , _sName(iName)
    {}

    template <typename... Args> inline
        auto operator()(Args&&... args)
    {
        return _lambda(std::forward<decltype(args)>(args)...);
    }

    std::string& getName()
    {
        return _sName;
    }
    private:
    Signature _lambda;
    std::string _sName;
};

template <typename Signature>
lambda_wrapper<Signature> myLambda(Signature&& lambda, const std::string& iName)
{
    return {std::forward<Signature>(lambda), iName};
}

class MyClass
{
    public:
    MyClass()
        : _lambda(myLambda([](int)
    {}, "test"))
    {

    }
    virtual ~MyClass() = default;

    void printVal()
    {
        lambda_wrapper<std::function<void(int)>>* pWrapper = _lambda.target<lambda_wrapper<std::function<void(int)>>>();
        if (pWrapper)
            std::cout << pWrapper->getName();
        else
            std::cout << "NULL" << std::endl;

        std::cout << "Stored type name=" << _lambda.target_type().name() << std::endl;
        std::cout << "Cast type name=" << typeid(lambda_wrapper<std::function<void(int)>>).name() << std::endl;
    }

    private:
    std::function<void(int)> _lambda;
};

int main()
{
    MyClass myClass;
    myClass.printVal();
}

结果如下:

NULL
Stored type name=class lambda_wrapper<class <lambda_de654b4bbc7b5fec872744c55537b509> >
Cast type name=class lambda_wrapper<class std::function<void __cdecl(int)> >

【问题讨论】:

  • lambda 不是std::function
  • "不是类型的lambda类型"是类型,但是没有名字。
  • 为什么在你想要lambda_wrapper&lt;std::function&lt;void&gt;(int)&gt;的地方使用std::function&lt;void(int)&gt;
  • @Jarod42,你是正确的类型,这就是我的意思。我没有使用 lambda_wrapper 类型,因为它更像是一种调试功能,我只想在“调试”模式下使用这种类型,而不是总是重载 lambda。它也是从客户端视图中隐藏类型。

标签: c++ c++11 lambda std-function


【解决方案1】:

_lambda.target 的正确类型在您的示例中无法命名。每个 lambda 表达式都会创建一个具有唯一类型的对象。

您可以扩展 lambda_wrapper 以键入像 std::function 这样的擦除,而不是使用 std::function 并尝试返回 lambda。然后您可以轻松地将行为添加到知道实际类型的部分。

【讨论】:

  • 感谢您的回答,但我不确定我是否理解。你有你所指的类型擦除的例子吗?如果这就是我的想法,恐怕它不符合我的需要。
  • @Vvh13 类型擦除是std::function 的工作原理。问题是我们需要向仍然知道 lambdas 类型的隐藏部分添加行为。所以你可以做的是实现类似std::function 的东西,它也有你想要的额外的东西
猜你喜欢
  • 2021-06-30
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2018-03-11
  • 2015-02-20
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多