Search code examples
c++templatesc++17variadic-templatespacked

How do I complete this example?


Hello I'm trying an example and I don't know how to follow. I'm trying to set a map (or some other structure) that saves all the same class/type together. For that, my approach was to take all parameters on the constructor of my class (ResourceManager_t) and everytime that I'm adding a new resource to our map, check if that resource can be added, because was accepted on Ctor.

But I don't know how to fill that std::vector<> to provide any type/class. I've tried with std::any but that doesn't seems to be a right solution.

#include <iostream>
#include <memory>
#include <unordered_map>
#include <vector>
#include <type_traits>
#include <cstdint>
#include <utility>

template <typename...> struct is_one_of;
template <typename F> struct is_one_of<F> { static constexpr bool value = false; };

template <typename F, typename S, typename... T>
struct is_one_of<F, S, T...>
{
    static constexpr bool value = std::is_same<F, S>::value
        || is_one_of<F, T...>::value;
};

template <typename...> struct is_unique;
template <> struct is_unique<> { static constexpr bool value = true; };

template<typename F, typename... T>
struct is_unique<F, T...>
{
    static constexpr bool value = is_unique<T...>::value
        && !is_one_of<F, T...>::value;
};


template<typename... types_t>
struct ResourceManager_t
{
    static constexpr inline bool areAllUnique = is_unique<types_t...>::value;
    static_assert(areAllUnique);
    
    // Ctor.
    ResourceManager_t()
    {
        constexpr auto size = 1 + sizeof...(types_t);
        m_resources.reserve(size);
    }

    template <typename type_t>
    void addResource([[maybe_unused]] type_t add)
    {
        static constexpr auto oneOf = is_one_of<type_t, types_t...>::value;
        static_assert(oneOf);
        const auto name = typeid(type_t).name();
        m_resources[name].emplace_back(add);
    }

    void printMap()
    {
        std::cout << "---------------\n";
        for(auto const& [s, vi] : m_resources)
        {
            std::cout << "[" << s << ": ";
            for(auto const& v : vi)
                std::cout << static_cast<s>(v) << " ";
            std::cout << "]\n";
        }

        std::cout << "---------------\n";
    }

private:
    std::unordered_map<std::string, std::vector</*TODO*/>> m_resources{};
};

int main(int, char *[])
{
    ResourceManager_t<int, char, uint32_t, uint8_t> rm{};
    rm.addResource(1);
    rm.addResource('a');
    rm.addResource(uint8_t{3U});
    rm.addResource(uint8_t{6U});
    // rm.addResource(3.F);

    rm.printMap();

    return 0;
}

I know that my question is not very clear but I've tried a lot of things. Example that I want

Map:
[int    , {1, 2, 3}]
[char   , {'a', 'b', 'c']
[uint8_t, {1U, 2U, 3U}]
// int, char, uint8_t can be changed with an int, or whatever that identifies my value.

If you want a compile/execute example, I've done it with std::vector<int> just to check all other parameters https://godbolt.org/z/hrb1afjxv


Solution

  • It seems to me that you're looking for a std::tuple<std::vector<types_t>...>.

    The following is a full compiling C++17 example, with some simplifications

    #include <tuple>
    #include <vector>
    #include <cstdint>
    #include <utility>
    #include <iostream>
    #include <type_traits>
    
    template <typename T0, typename ... Ts>
    struct is_one_of : public std::bool_constant<(std::is_same_v<T0, Ts> || ...)>
     { };
    
    template <typename...>
    struct are_unique : public std::true_type
     { };
    
    template <typename T0, typename ... Ts>
    struct are_unique<T0, Ts...>
       : public std::bool_constant<not is_one_of<T0, Ts...>::value
                                && are_unique<Ts...>::value>
     { };
    
    template <typename ... Ts>
    struct ResourceManager
     {
       static_assert(are_unique<Ts...>::value, "not unique types");
    
       private:
          std::tuple<std::vector<Ts>...> m_resources{};
    
       public:
          ResourceManager ()
           { }
    
          template <typename T0>
          void addResource (T0 add)
           {
             // the static assert is superflous: you get an error from std::get
             // in case of wrong type
             static_assert(is_one_of<T0, Ts...>::value, "wrong type in add");
    
             std::get<std::vector<T0>>(m_resources).emplace_back(std::move(add));
           }
    
          void printMap ()
           {
             auto l = [](auto const & vect)
              {
                std::cout << "[" << typeid(decltype(vect[0])).name() << ": ";
                for (auto const & value : vect)
                   std::cout << value << " ";
                std::cout << "]\n";
              };
    
             std::cout << "---------------\n";
             std::apply([&](auto ... vects){ (l(vects), ...); }, m_resources);
             std::cout << "---------------\n";
           }
     };
    
    int main ()
    {
        ResourceManager<int, char, std::string, std::uint8_t> rm;
    
        rm.addResource(1);
        rm.addResource('a');
        rm.addResource(std::uint8_t{3U});
        rm.addResource<std::uint8_t>(6U);
        // rm.addResource(3.F);
        rm.addResource(std::string{"str1"});
        rm.addResource<std::string>("longestStringForYou");
    
        rm.printMap();
    }