Учитывая boost::tuple
и std::tuple
, как вы можете конвертировать между ними?Преобразование между std :: tuple и boost :: tuple
Другими словами, как бы вы реализовали следующие две функции?
template <typename... T> boost::tuple<T...> asBoostTuple( std::tuple<T...> stdTuple);
template <typename... T> std::tuple<T...> asStdTuple (boost::tuple<T...> boostTuple);
Кажется, что это просто, но я не мог найти никаких хороших решений.
Что я пробовал?
Я закончил работу с шаблоном программирования. Кажется, что это работает в моей конкретной настройке, но это не кажется правильным (слишком многословным), и я даже не уверен, действительно ли он работает во всех ситуациях (я приду к этому вопросу позже).
Во всяком случае, вот интересная часть:
template<int offset, typename... T>
struct CopyStdTupleHelper
{
static void copy(boost::tuple<T...> source, std::tuple<T...>& target) {
std::get<offset>(target) = std::move(boost::get<offset>(source));
CopyStdTupleHelper<offset - 1, T...>::copy(source, target);
}
static void copy(std::tuple<T...> source, boost::tuple<T...>& target) {
boost::get<offset>(target) = std::move(std::get<offset>(source));
CopyStdTupleHelper<offset - 1, T...>::copy(source, target);
}
};
template<typename... T>
struct CopyStdTupleHelper<-1, T...>
{
static void copy(boost::tuple<T...> source, std::tuple<T...>& target)
{ /* nothing to do (end of recursion) */ }
static void copy(std::tuple<T...> source, boost::tuple<T...>& target)
{ /* nothing to do (end of recursion) */ }
};
std::tuple<T...> asStdTuple(boost::tuple<T...> boostTuple) {
std::tuple<T...> result;
CopyStdTupleHelper<sizeof...(T) - 1, T...>::copy(std::move(boostTuple), result);
return result;
}
boost::tuple<T...> asBoostTuple(std::tuple<T...> stdTuple) {
boost::tuple<T...> result;
CopyStdTupleHelper<sizeof...(T) - 1, T...>::copy(std::move(stdTuple), result);
return result;
}
Интересно, если есть более элегантный способ. Похоже, что очень распространенная операция для переноса существующих API с использованием boost::tuple
в std::tuple
.
Я пытался предоставить вам с минимальным тестового примера, где не хватает только реализация asBoostTuple
и asStdTuple
. Однако для некоторой магии с boost::tuples::null_type
, которую я не совсем понимаю, он не компилируется. Это также является причиной того, что я не уверен, что мое существующее решение может быть в целом применено.
Вот фрагмент кода:
#include <tuple>
#include <boost/tuple/tuple.hpp>
template <typename... T>
boost::tuple<T...> asBoostTuple(std::tuple<T...> stdTuple) {
boost::tuple<T...> result;
// TODO: ...
return result;
}
template <typename... T>
std::tuple<T...> asStdTuple(boost::tuple<T...> boostTuple) {
std::tuple<T...> result;
// TODO: ...
return result;
}
int main() {
boost::tuple<std::string, int, char> x = asBoostTuple(std::make_tuple("A", 1, 'x'));
// ERROR:
std::tuple<std::string, int, char> y = asStdTuple<std::string, int, char>(x);
return x == y;
}
Сообщение об ошибке:
example.cpp:20:38: error: no viable conversion from 'tuple<[3 *
...], boost::tuples::null_type, boost::tuples::null_type,
boost::tuples::null_type, boost::tuples::null_type,
boost::tuples::null_type, boost::tuples::null_type,
boost::tuples::null_type>' to 'tuple<[3 * ...], (no
argument), (no argument), (no argument), (no argument),
(no argument), (no argument), (no argument)>'
...int, char> y = asStdTuple<std::string, int, char>(x);
Я понимаю, реализация, повышающая не основана на переменном число шаблонов, которые должны объяснить null_type
, но все-таки я m не уверен, как избежать этой ошибки компиляции.