1

我已经从这个答案改编了一些代码来处理目标变体是源变体的子集的情况,如下所示:

template <class... Args>
struct variant_cast_proxy
{
    std::variant<Args...> v;

    template <class... ToArgs>
    operator std::variant<ToArgs...>() const
    {
        return std::visit(
            [](auto&& arg) -> std::variant<ToArgs...> { 
                if constexpr (std::is_convertible_v<decltype(arg), std::variant<ToArgs...>>)
                    return arg;
                else
                    throw std::runtime_error("bad variant cast");
            },
            v
        );
    }
};

template <class... Args>
auto variant_cast(const std::variant<Args...>& v) -> variant_cast_proxy<Args...>
{
    return { v };
}

struct A {};
struct B {};
struct C {};
struct D {};
struct E {};
struct F {};

int main() {

    std::variant<A, B, C, D> v1 = B();
    std::variant<B,C> v2;
    v2 = variant_cast(v1);
}

上述方法有效,但我希望它能够处理在编译时可以检测到错误转换的情况。以上处理了运行时所有错误的转换,但运行时和编译时错误都是可能的。如果 v 持有 C 类型的值,则将 v类型std::variant<A,B,C>转换为std::variant<A,B>应该在运行时失败,但是例如

std::variant<A, B, C, D> v1 = B();
std::variant<E,F> v2;
v2 = variant_cast(v1)

甚至不应该编译。

我相信这可以通过 std::enable_if 完成,但我不确定它似乎需要测试可变参数包的集合包含,我不知道该怎么做。

4

2 回答 2

1

我认为可转换是一个错误的问题......除非你真的想能够像 a 一样投射variant<int, long>到 a variant<string, double>。我认为更好的检查是源中的每种类型都variant出现在目标中variant

为此,您可以使用Boost.Mp11来简化此检查:

template <class... Args>
struct variant_cast_proxy
{
    std::variant<Args...> v;

    template <class... ToArgs,
        class V = std::variant<ToArgs...>,
        std::enable_if_t<
            // every type in the source variant is present in the destination
            (mp_contains<V, Args>::value && ...)
            // and the destination id all distinct
            && mp_is_set<V>::value
            , int> = 0>
    operator std::variant<ToArgs...>() const
    {
        return std::visit([&](auto const& arg){ return V(arg); }, v);
    }
};
于 2020-04-05T17:54:45.230 回答
1

您可以添加static_assert检查任何可能持有的变体是否可转换:

static_assert((std::is_convertible_v<Args, std::variant<ToArgs...>> || ...),
    "No possible variant that could be converted exists");

或者,如果您想要 SFINAE,您可以在模板参数中执行此操作:

    // extracted into helper function
    template <class... ToArgs>
    static constexpr bool is_convertible() noexcept {
        return (std::is_convertible_v<Args, std::variant<ToArgs...>> || ...);
    }

    template<class... ToArgs, std::enable_if_t<is_convertible<ToArgs...>(), int> = 0>
    operator std::variant<ToArgs...>() const
    {
        // ...
    }
于 2020-04-05T17:46:49.677 回答