我有一个包含字节数组的 struct ,我想在二进制中序列化和反序列化,但它只适用于数组up to 32 elements.

下面是我的最小示例代码

main.rs:

#[macro_use]
extern crate serde_derive;
extern crate serde;
extern crate bincode;

use bincode::{serialize, deserialize, Infinite};

const BYTECOUNT: usize = 32; // 33 and more does not work, I need 128
type DataArr = [u8; BYTECOUNT];

#[derive(Serialize, Deserialize, Debug)]
struct Entry {
    number: i64,
    data: DataArr
}

fn main() {
    let mut my_entry = Entry { number: 12345, data: [0; BYTECOUNT] };
    my_entry.data[4] = 42;

    // Convert the Entry to binary.
    let serialized: Vec<u8> = serialize(&my_entry, Infinite).unwrap();
    println!("serialized = {:?}", serialized);

    // Convert the binary representation back to an Entry.
    let deserialized: Entry = deserialize(&serialized).unwrap();
    println!("deserialized = {:?}", deserialized);
}

Cargo.toml:

[package]
name = "array_serialization_test"
version = "0.1.0"

[dependencies]
serde = "*"
serde_derive = "*"
bincode = "*"

输出:

serialized = [57, 48, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 42, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
deserialized = Entry { number: 12345, data: [0, 0, 0, 0, 42, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0] 

如何使其适用于数组中的128个元素?我可以手动扩展用户代码中的array_impls!吗?有没有其他办法?

我认为这个问题不同于How do I map a C struct with padding over 32 bytes using serde and bincode?,因为我实际上需要数组的内容,因为它不仅仅用于填充.此外,我想知道我是否可以扩展我的代码array_impls!.

推荐答案

目前Serde无法提供适用于every数组大小的SerializeDeserialize IMPL.这是被封锁在const generics,这是正在工作,并有望在2018年晚些时候在夜间降落.

现在,您可以定义自己的"大数组"助手,它可以序列化和反序列化 crate 中使用的任何特定大小的array.要使用大数组帮助器的字段需要标记为#[serde(with = "BigArray")],否则Serde将查找不存在的SerializeDeserialize impl.

#[macro_use]
extern crate serde_derive;

extern crate serde;
extern crate bincode;

mod big_array;
use big_array::BigArray;

const BYTECOUNT: usize = 128;
type DataArr = [u8; BYTECOUNT];

#[derive(Serialize, Deserialize)]
struct Entry {
    number: i64,
    #[serde(with = "BigArray")]
    data: DataArr
}

fn main() {
    let mut my_entry = Entry { number: 12345, data: [0; BYTECOUNT] };
    my_entry.data[4] = 42;

    // Convert the Entry to binary.
    let serialized: Vec<u8> = bincode::serialize(&my_entry).unwrap();
    println!("serialized = {:?}", serialized);

    // Convert the binary representation back to an Entry.
    let deserialized: Entry = bincode::deserialize(&serialized).unwrap();
    println!("deserialized = {} {:?}", deserialized.number, &deserialized.data[..]);
}

大数组助手可以在src/big_array.rs中定义如下.如果你想拥有它的话,也许这本身就是一个很好的 crate !

use std::fmt;
use std::marker::PhantomData;
use serde::ser::{Serialize, Serializer, SerializeTuple};
use serde::de::{Deserialize, Deserializer, Visitor, SeqAccess, Error};

pub trait BigArray<'de>: Sized {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
        where S: Serializer;
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
        where D: Deserializer<'de>;
}

macro_rules! big_array {
    ($($len:expr,)+) => {
        $(
            impl<'de, T> BigArray<'de> for [T; $len]
                where T: Default + Copy + Serialize + Deserialize<'de>
            {
                fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
                    where S: Serializer
                {
                    let mut seq = serializer.serialize_tuple(self.len())?;
                    for elem in &self[..] {
                        seq.serialize_element(elem)?;
                    }
                    seq.end()
                }

                fn deserialize<D>(deserializer: D) -> Result<[T; $len], D::Error>
                    where D: Deserializer<'de>
                {
                    struct ArrayVisitor<T> {
                        element: PhantomData<T>,
                    }

                    impl<'de, T> Visitor<'de> for ArrayVisitor<T>
                        where T: Default + Copy + Deserialize<'de>
                    {
                        type Value = [T; $len];

                        fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
                            formatter.write_str(concat!("an array of length ", $len))
                        }

                        fn visit_seq<A>(self, mut seq: A) -> Result<[T; $len], A::Error>
                            where A: SeqAccess<'de>
                        {
                            let mut arr = [T::default(); $len];
                            for i in 0..$len {
                                arr[i] = seq.next_element()?
                                    .ok_or_else(|| Error::invalid_length(i, &self))?;
                            }
                            Ok(arr)
                        }
                    }

                    let visitor = ArrayVisitor { element: PhantomData };
                    deserializer.deserialize_tuple($len, visitor)
                }
            }
        )+
    }
}

big_array! {
    40, 48, 50, 56, 64, 72, 96, 100, 128, 160, 192, 200, 224, 256, 384, 512,
    768, 1024, 2048, 4096, 8192, 16384, 32768, 65536,
}

Rust相关问答推荐

按下按钮时如何在服务器中创建文件并在本地下载?

什么样的 struct 可以避免使用RefCell?

常量泛型和类型枚举箱有重叠的用途吗?

这种获取-释放关系是如何运作的?

定义采用更高级类型泛型的性状

integer cast as pointer是什么意思

如何为rust trait边界指定多种可能性

Tokio_Postgres行上未显示退回特性的生存期,且生命周期 不够长

如何将单个 struct 实例与插入器一起传递到Rust中的映射

如何在函数中返回自定义字符串引用?

通过RabbitMQ取消铁 rust 中长时间运行的人造丝任务的策略

如何在Rust中缩短数组

为什么`tokio::main`可以直接使用而不需要任何导入?

如何强制匹配的返回类型为()?

从字节数组转换为字节元组和字节数组时,为什么 Transmute 会对字节重新排序?

tokio::spawn 有和没有异步块

Rust 中指向自身的引用如何工作?

如何使用 Rust Governor 为每 10 秒 10 个请求创建一个 RateLimiter?

发生移动是因为 `data` 的类型为 `Vec`,它没有实现 `Copy` 特性

在 Rust 中组合特征的不同方法是否等效?