l值ref和r值ref以及non-ref类型的完美转发可变参数模板?

时间:2019-02-28 02:36:10

标签: c++ c++14 variadic-templates rvalue-reference forwarding-reference

在我的程序中,我有一些模板类,这些模板类大多数是用于特殊用途std :: function <..>的包装器。最小的示例是:

template <typename... Args>
class Foo {

    public:

        explicit Foo(std::function<void(Args&&...)> _function)
            : function_(_function)
        {}

        template<typename... Arguments>
        void Bar(Arguments&&... _args) {
            function_(std::forward<Arguments>(_args)...);
        }

    private:

        std::function<void(Args&&...)> function_;

};

这些模板的实例化通常是l值引用,r值引用或无引用类型的组合。问题在于,当某些参数是非引用类型(例如int或std :: vector)时,调用Bar会导致错误。解决方法是声明一个临时变量,然后将其移入函数调用。

int main(){
    Foo<int> test1([](int x) { });
    const int x = 1;
    test1.Bar(x); // [Error] cannot bind rvalue reference of type 'int&&' to lvalue of type 'const int'

    int tmp = x;
    test1.Bar(tmp); // [Error] cannot bind rvalue reference of type 'int&&' to lvalue of type 'int'
    test1.Bar(std::move(tmp)); // [OK] But I don't want to have to reassign and move every time I use this.

    /* I want perfect forwarding on variables that can be forwarded. */
    /* There are cases when the templates are like this with a combination of l-value ref and r-value ref and non-ref types. */
    Foo<const std::vector<uint8_t>&, std::vector<uint8_t>&&, int> test2([](const std::vector<uint8_t>&, std::vector<uint8_t>&&, int) { });
    test2.Bar(std::vector<uint8_t>(1, 2), std::vector<uint8_t>(1, 2), x); // [Error] cannot bind rvalue reference of type 'int&&' to lvalue of type 'const int'

    return 1;
}

我希望能够将Bar与任何模板参数一起使用,而不必每次都重新分配和std :: move(),而且还完美地传递了ref参数。有没有办法做到这一点?

编辑 在网上浏览了一下之后-问题是std::function<void(Args&&...)> function_;不是一个采用通用引用的函数,而是采用了r-val引用的函数。因此,尝试转发无引用类型会引发错误。

那么问题是,是否可以拥有并存储带有通用引用的std :: function?

1 个答案:

答案 0 :(得分:2)

std::function<void(Args&&...)>中,您实际上希望得到r值引用,您可能需要std::function<void(Args...)>

template <typename... Args>
class Foo {
public:
    explicit Foo(std::function<void(Args...)> _function)
        : function_(_function)
    {}

    template <typename... Arguments>
    void Bar(Arguments&&... _args) {
        function_(std::forward<Arguments>(_args)...);
    }

private:
    std::function<void(Args...)> function_;
};

Demo

如果合适,您可以摆脱std::function

template <typename F>
class Foo {
public:
    explicit Foo(F f) : f(f) {}

    template <typename... Ts>
    auto operator ()(Ts&&... args) const
    -> decltype(f(std::forward<Ts>(args)...))
    {
        return f(std::forward<Ts>(args)...);
    }

private:
    F f;
};

template <typename F>
Foo<F> MakeFoo(F f) { return Foo<F>{f}; }

Demo