【问题标题】:How to use variadic templates as keys to a map of std::type_index?如何将Variadic模板用作STD :: type_index地图的键?
【发布时间】:2018-04-24 21:23:28
【问题描述】:

我正在尝试实现一个可变参数模板化成员函数,它可以接受具有不同数量模板参数或根本没有模板参数的 std::function 对象。

下面是我正在尝试的简化且可编译的示例。我希望能够将任何用户定义的数据对象动态添加到我的经理。然后,我希望能够使用可变参数模板迭代存储在管理器中的任何数据类型。我在哪里评论过 // HowTo? // 是我坚持的。如何使用可变参数模板实现注释掉的代码?

我试图从这篇文章中得出一个解决方案: Possible Solution?

#include <functional>
#include <map>
#include <memory>
#include <typeindex>
#include <vector>

constexpr auto Count = 10;

struct Base
{
    virtual ~Base()
    {
    }
};

template <typename T>
struct Container : public Base
{
    std::vector<T> data;
    std::vector<bool> valid;
};

struct Manager
{
    template <typename T>
    void add()
    {
        Container<T>* container{};

        if(this->containers[typeid(T)] == nullptr)
        {
            auto c = std::make_unique<Container<T>>();
            container = c.get();
            this->containers[typeid(T)] = std::move(c);
        }
        else
        {
            container = static_cast<Container<T>*>(this->containers[typeid(T)].get());
        }

        container->data.push_back(T());
        container->valid.push_back(true);
    }

    template <typename ...Args>
    void each(std::function<void(Args&...)> f)
    {
        // HowTo? // auto oneContainer = static_cast<Container<T>*>(this->containers[typeid(DataOne)].get());
        // HowTo? // auto twoContainer = static_cast<Container<T>*>(this->containers[typeid(DataTwo)].get());

        for(auto i = 0; i < Count; i++)
        {
            // HowTo? // if(oneContainer->valid[i] == true && twoContainer->valid[i] == true)
            // HowTo? // f(oneContainer->data[i], twoContainer->data[i]);
        }
    }

    std::map<std::type_index, std::unique_ptr<Base>> containers;
};

struct DataOne
{
    int value{};
};

struct DataTwo
{
    double value{};
};

int main()
{
    Manager manager;

    for(auto i = 0; i < Count; i++)
    {
        manager.add<DataOne>();
        manager.add<DataTwo>();
    }

    manager.each<DataOne>([](DataOne& a) {
        a.value = 0;
    });

    manager.each<DataTwo>([](DataTwo& a) {
        a.value = 0.0;
    });

    manager.each<DataOne, DataTwo>([](DataOne& a, DataTwo& b) {
        a.value = 0;
        b.value = 0.0;
    });
}

【问题讨论】:

    标签: c++ templates variadic-templates c++17


    【解决方案1】:
    auto containers = std::make_tuple<Container<Args>*...>(
      static_cast<Container<Args>*>(this->containers[typeid(Args)].get())...
    );
    

    ...

        for(auto i = 0; i < Count; i++)
        {
          std::apply([&f](auto*... containers){
            bool valid = (containers&&...) && (containers->valid[i]&&...);
            if (valid)
              f( containers->data[i]... );
          }, containers);
        }
    

    如有错别字请见谅。

    【讨论】:

    • 谢谢!这正是我想要的。我不知道 std::apply() 存在。我需要阅读更多关于 c++17 特性的信息。我唯一的问题是我的编译器是 MSVC 15.4.3 并且似乎还不支持折叠表达式。不过,它看起来确实会在 MSVC 15.5 中得到支持!
    • @asxa86 试试auto tests = { true, (containers &amp;&amp; containers-&gt;value[i])... }; bool valid = all_of( tests.begin(), tests.end() );
    猜你喜欢
    • 2023-03-06
    • 1970-01-01
    • 2018-02-22
    • 2016-07-13
    • 1970-01-01
    • 2016-10-23
    • 2017-02-03
    • 2019-12-19
    • 1970-01-01
    相关资源
    最近更新 更多