Просто наткнулся на это, когда я пытался по существу решить ту же проблему.Можно даже заставить работать без , полагаясь на тип rebind
, специфичный для std::allocator
- единственное требование состоит в том, что тип значения отскока является первым параметром шаблона соответствующих классов.Это относится ко всем соответствующим классам STL (std::vector
, std::set
, std::list
и т. Д., А также, например, std::less
и std::allocator
).
A до C ++ 11Решение будет выглядеть так:
template <class Container, class NewType>
struct rebind;
template <class ValueType, template <class> class Container, class NewType>
struct rebind<Container<ValueType>, NewType>
{
typedef Container<NewType> type;
};
template <class ValueType, class A, template <class, class> class Container, class NewType>
struct rebind<Container<ValueType, A>, NewType>
{
typedef Container<NewType, typename rebind<A, ValueType>::type> type;
};
template <class ValueType, class A, class B, template <class, class, class> class Container, class NewType>
struct rebind<Container<ValueType, A, B>, NewType>
{
typedef Container<NewType, typename rebind<A, ValueType>::type, typename rebind<B, ValueType>::type> type;
};
// Continue for more parameters (A, B, C, ...)
C ++ 11 делает его немного проще:
template <class Container, class NewType>
struct rebind;
template <class ValueType, class... Args, template <class...> class Container, class NewType>
struct rebind<Container<ValueType, Args...>, NewType>
{
typedef Container<NewType, typename rebind<Args, NewType>::type...> type;
};
Для поддержки std::array
можно добавить следующее:
template <class ValueType, std::size_t N, template <class, std::size_t> class Container, class NewType>
struct rebind<Container<ValueType, N>, NewType>
{
typedef Container<NewType, N> type;
};
Результат может быть использован практически с любым типом STL:
#include <iostream>
#include <typeinfo>
#include <vector>
#include <set>
#include <deque>
#include <queue>
#include <list>
#include <array>
#include "rebind.h"
// Make it all a bit more compact
#define REBIND_DEMO(container, new_type) \
do { \
container test; \
rebind<decltype(test), new_type>::type test2; \
std::cout << typeid(test).name() << "\n"; \
std::cout << typeid(test2).name() << "\n"; \
} while (0)
int main()
{
REBIND_DEMO(std::set<float>, double);
REBIND_DEMO(std::list<float>, double);
REBIND_DEMO(std::deque<float>, double);
REBIND_DEMO(std::queue<float>, double);
typedef std::array<float, 4> TestArray;
REBIND_DEMO(TestArray, double);
REBIND_DEMO(std::unordered_set<float>, double);
return 0;
}
Выполнение этого и передача вывода через c++filt -t
в системе Linux дает вам
std::set<float, std::less<float>, std::allocator<float> >
std::set<double, std::less<double>, std::allocator<double> >
std::list<float, std::allocator<float> >
std::list<double, std::allocator<double> >
std::deque<float, std::allocator<float> >
std::deque<double, std::allocator<double> >
std::queue<float, std::deque<float, std::allocator<float> > >
std::queue<double, std::deque<double, std::allocator<double> > >
std::array<float, 4ul>
std::array<double, 4ul>
std::unordered_set<float, std::hash<float>, std::equal_to<float>, std::allocator<float> >
std::unordered_set<double, std::hash<double>, std::equal_to<double>, std::allocator<double> >