У меня есть вариант ScalarVar
using ScalarVar = std::variant<int, std::string>;
И вариант Var
, который сам может быть ScalarVar
или std::vector
из ScalarVar
s
using Var = std::variant<ScalarVar, std::vector<ScalarVar>>;
Я хочу сделать функцию template<typename T, typename Variant> T Get(const Variant& var);
, чтобы при задании варианта, который не включает внутренние варианты, он действовал так же, как std::get<T>
, то есть он будет возвращать значение T
, если Variant
в настоящее время содержит T
или если задан вариант, содержащий другие варианты, он будет рекурсивно получать содержащийся тип, пока не будет найден не вариант, а затем вернет его.
Вот моя лучшая попытка сделать это:
#include <iostream>
#include <variant>
#include <string>
#include <typeindex>
#include <vector>
#include <map>
template<typename T, typename... T2>
struct is_variant { static inline constexpr bool value = false; };
template<typename... T>
struct is_variant<std::variant<T...>> { static inline constexpr bool value = true; };
template<typename T, typename Variant>
T Get(const Variant& var) {
static_assert (is_variant<Variant>::value == true, "Template parameter Variant must be a std::variant");
auto inner = std::visit([](const auto& i){ return i; }, var);
if constexpr(is_variant<typeof(inner)>::value) {
return Get<T>(inner);
}
else return inner;
}
int main()
{
using ScalarVar = std::variant<int, std::string>;
using Var = std::variant<ScalarVar, std::vector<ScalarVar>>;
ScalarVar s = 5;
std::cout << Get<int>(s) << std::endl;
return 0;
}
Это должно просто вернуть T
, если T
не является std::variant
, и вернуть std::get<InnerT>
, если T
- это std :: вариант, который содержит тип T
.
Но я получаю очень сложную ошибку компиляции из g cc для строки:
std::cout << Get<int>(s) << std::endl
/usr/include/c++/9/variant:1005: error: invalid conversion from ‘std::__success_type<std::__cxx11::basic_string<char> >::type (*)(Get(const Variant&) [with T = int; Variant = std::variant<int, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >]::<lambda(const auto:22&)>&&, const std::variant<int, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)’ {aka ‘std::__cxx11::basic_string<char> (*)(Get(const Variant&) [with T = int; Variant = std::variant<int, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >]::<lambda(const auto:22&)>&&, const std::variant<int, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)’} to ‘int (*)(Get(const Variant&) [with T = int; Variant = std::variant<int, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >]::<lambda(const auto:22&)>&&, const std::variant<int, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)’ [-fpermissive]
1005 | { return _Array_type{&__visit_invoke}; }
| ^
| |
| std::__success_type<std::__cxx11::basic_string<char> >::type (*)(Get(const Variant&) [with T = int; Variant = std::variant<int, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >]::<lambda(const auto:22&)>&&, const std::variant<int, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&) {aka std::__cxx11::basic_string<char> (*)(Get(const Variant&) [with T = int; Variant = std::variant<int, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >]::<lambda(const auto:22&)>&&, const std::variant<int, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)}
Как мне получить требуемое поведение?