c++ - 使用模板中的参数包创建对象

标签 c++ templates constructor variadic-templates smart-pointers

我想创建模板函数,它会根据模板类型名和参数包创建对象。

我创建了一个函数,该函数应该根据模板的类型名称创建对象,并且我还想将参数包传递给该模板,以便将参数传递给构造函数。这是正确的吗?:

template<typename TComponent, typename... Args>
    void CreateComponent(Args... args) 
    {
        std::shared_ptr<TComponent> component = std::make_shared<TComponent>(args ...);
    }

我还想像这样将这些参数传递给另一个函数:

template<typename TComponent, typename... Args>
    void AddComponent(Args... args) 
    {
          m_world->AddComponent<TComponent, Args>(m_id, args...);
    }

但是编译器返回错误“'args'参数包必须在此上下文中展开”

是否有可能实现我想要实现的目标?

最佳答案

But compiler returns an error " 'args' parameter pack must be expanded in this context"

是的:你忘了扩展类型

m_world->AddComponent<TComponent, Args...>(m_id, args...);
// ...................................^^^

正如 Jarod42 所指出的,根据情况,您可以避免显式显示 Args... 扩展

m_world->AddComponent<TComponent>(m_id, args...);
// no more Args...

并让编译器通过 args... 推断类型(但我们应该看到 AddComponent() 定义)。

无论如何,我没有在您的 CreateComponents() 函数中看到错误,但正如 François Andrieux 在评论中正确指出的那样,您没有使用完美转发。

在答案中解释这个论点太过分了,但是这样一来,您就放弃了移动语义优势(也就是说:您可能会制作一些不必要的拷贝)。

下面是你实现完美转发的CreateComponents()函数

template <typename TComponent, typename ... Args>
void CreateComponent (Args && ... args) 
 { // .....................^^ forwarding reference added
   std::shared_ptr<TComponent> component
     = std::make_shared<TComponent>(std::forward<Args>(args)...);
 } // ..............................^^^^^^^^^^^^^^^^^^^^^^^^

关于c++ - 使用模板中的参数包创建对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56263804/

相关文章:

C++迭代器需要删除吗?

c++ - DirectX9 C++ 实时重新着色顶点数据

templates - Golang setCookie() 在模板 Execute() 之后失败

c++ - 变量初始化和构造函数

JavaScript - 使用原型(prototype)的构造函数

c++ - 使用 std::vector 移动和复制语义

c++ - 获取 QLineEdit 的文本边距

c++ - Mac OS X 中的 ACL - 查找 GUID

c++ - 可变数量的构造函数参数取决于整数模板

c++ - 返回类型为 T 的函数模板无法编译