casting - 转换相同底层类型的枚举变体

标签 casting rust variant

有没有一种简单的方法可以创建一个enum,只要底层类型匹配,它就可以强制或以其他方式将实例从一个变体转换为另一个变体?

例如,给定这个枚举:

enum CanConvert {
    Vec1(Vec<String>),
    Vec2(Vec<String>),
    NotVec(i32),
}

我希望能够轻松地在 Vec1Vec2 变体之间进行转换:

let v1 = CanConvert::Vec1(["one", "two", "three", "uno", "dos", "tres"]
                          .iter()
                          .map(|&s| s.into())
                          .collect());

let v2 = v1 as CanConvert::Vec2;

上面的代码目前给出了错误not a type,因为枚举变体不是(当前)考虑的类型。即使它编译了,也不清楚应该如何处理 v1NotVec 变体的情况( panic ?)。

这行得通,但它非常冗长,这似乎应该是一个相当普遍的模式:

let v2 = match v1 {
    CanConvert::Vec1(ref underlying) |
    CanConvert::Vec2(ref underlying) => CanConvert::Vec2(underlying.clone()),
    _ => panic!("v1 did not have appropriate underlying type!"),
};

那么有没有一种简单的方法可以在可能的情况下提供变体之间的这种简单类型转换?我还没有编写任何我自己的 Deref 强制转换,但这似乎是我在这里寻找的。 (包裹在某种自定义 derive 中的强制将是理想的。)

编辑:请注意,据我所知,没有任何安全的方法可以引用 原始文件,并具有所需的变体类型;需要一个完整的克隆,因为引用必然有错误的变体标签。但是,似乎可以通过简单地适当更改标签来转换原始enum

最佳答案

首先,由于&T 是一个不可变引用,Deref 的实现不能 改变类型。它只能指向内部(在这种情况下,它可能指向底层的Vec)。

关于转换,ref 在这里引入了低效率,因为它迫使您克隆,如果您获得所有权,您可以简单地移动底层向量:

fn to_v2(v: CanConvert) -> CanConvert {
    match v {
        CanConvert::Vec1(u) | CanConvert::Vec2(u) => CanConvert::Vec2(u),
        _ => panic!("Inappropriate type");
    }
}

关于casting - 转换相同底层类型的枚举变体,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43354262/

相关文章:

c# - 使用辅助转换函数时的约定

redis - 调用使用redis-rs创建redis连接的函数时如何防止生命周期问题

c++ - 如何在变体容器上使用 STL 算法

struct - 放入结构中时,值的生命周期不够长

c++ - 在不知道哪个值处于事件状态的情况下在 std::visit 中获取事件值

types - 将多个变体合并为一个变体

lisp - Common Lisp中的整数除法?

c++ - 为避免有符号/无符号比较而进行的 C 转换是否有意义?

c++ - 如何从 const char* 转换为结构?

architecture - 代理对象或不同方法的设计原则