如何解决Rust Bincode枚举二进制序列化
我对Rust还是很陌生,并且在我的项目中使用bincode。枚举变量的编码是我在与现有服务器接口时一直试图解决的问题。旧项目是用C编写的,并且定义了“ fshort-enums”的比较选项。
在旧的C项目中,我将定义一个枚举,例如:
enum test_enum {
Start,Init,Complete,}
然后将这些数据放入数据包时,它将占用1个字节,如下所示:
0x00,0x01,0x02
如果我将枚举定义为:
enum test_enum {
Start,Complete = 65535,}
然后,当将此数据放入数据包时,每个变体将占用2个字节,如下所示:
0x00 0x00,0x01 0x00,0x02 0x00
现在,我找到了一种方法来配置bincode
枚举编码,使其从默认的4个字节更改为适合该值的大小,如下所示:
#[derive(Serialize,Deserialize)]
enum TestEnum {
Start,}
let x = TestEnum::Complete;
if let Ok(bytes) = bincode::options().with_varint_encoding().serialize(&x) {
println!("{:?}",bytes); // [2]
}
问题1:如何强制bincode
使所有变体占用2个字节?
问题2:如何强制bincode
应用枚举变量值? (在此示例中,TestEnum::Complete
的值为2而不是65535)。
解决方法
Options::with_varint_encoding
可能不是您想要的,因为它只对小于256的整数使用1个字节。您可以使用Options::with_fixint_encoding
来编码与Rust类型具有相同精度的整数,但您无需指定它,因为它已经是默认值。
Serde具有结构性;它并不真正在乎内存中数据的实际表示方式。您分配给枚举的数值将丢失,并且bincode
将仅看到变体的名称。
您可以通过配置serde
来将枚举转换为整数,然后再将其公开给bincode
序列化程序来解决此问题:
#[derive(Serialize,Deserialize,Copy,Clone)]
#[serde(into = "u16")]
enum TestEnum {
Start,Init,Complete = 65535,}
impl From<TestEnum> for u16 {
fn from(value: TestEnum) -> u16 {
value as u16
}
}
然后您的代码给出了预期的结果:
let x = TestEnum::Complete;
if let Ok(bytes) = bincode::serialize(&x) {
println!("{:?}",bytes); // [255,255]
}
,
我确实使用了@Peter Hall提供的答案,但是我不得不为所有已定义的枚举(有很多枚举)添加一个实现,以便能够对它们进行序列化,而第二个实现则可以反序列化。因此,我继续进行调查,找到了对我来说似乎更有用的替代方法。现在,我不再使用每个枚举的实现,而是使用crates.io的num_enum板条箱,并向枚举添加一些属性。因此,代码如下所示:
use num_enum::{IntoPrimitive,TryFromPrimitive};
#[derive(Serialize,Clone)]
#[serde(into = "u16",try_from = "u16")]
#[repr(u16)]
enum TestEnum {
Start,}
我只是认为这可能会在将来对其他人有所帮助。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。