如何为大于32的数组实现serde :: Deserialize?

6

我需要使用33字节的数组。看起来包括32个元素及以下的数组可以正常序列化,但是33字节的数组会导致:

error[E0277]: the trait bound `[u8; 33]: _::_serde::Deserialize<'_>` is not satisfied
  --> src/main.rs:54:2
   |
54 |     foo: [u8; 33]
   |     ^^^ the trait `_::_serde::Deserialize<'_>` is not implemented for `[u8; 33]`
   |
   = help: the following implementations were found:
             <&'a [u8] as _::_serde::Deserialize<'de>>
             <[T; 0] as _::_serde::Deserialize<'de>>
             <[T; 10] as _::_serde::Deserialize<'de>>
             <[T; 11] as _::_serde::Deserialize<'de>>
           and 30 others
   = note: required by `_::_serde::de::MapAccess::next_value`

他们显然决定32位是一个停止的好地方。我该如何为33个元素的数组添加支持?


1
一个简单的解决方案是使用 serde-big-array 库。 - L. Riemer
@L.Riemer 谢谢,但 serde-big-array 不支持33个元素,只支持40、48、50等。我会看看是否能以某种方式将其 lib.rs 复制到我的项目中。 - fadedbee
我的错。是的,你只需要在big_array!宏调用中添加33!可能会使用mcartons solution来完成这个操作。它基本上是相同的代码。 - L. Riemer
1个回答

8
你不能为你不拥有的类型实现一个你不拥有的特质,但不幸的是,在这种特殊情况下,你并不需要这样做。

对于序列化,支持大数组相当容易。

struct S {
   #[serde(serialize_with = "<[_]>::serialize")]
   arr: [u8; 256],
}

源代码

这将把数组强制转换为切片并对其进行序列化。


对于反序列化,它会更加复杂一些。
serde 的作者建议采用以下方法。
#[macro_use] extern crate serde_derive;

extern crate serde; extern crate serde_json;

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

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, }

#[derive(Serialize, Deserialize)]
struct S {
    #[serde(with = "BigArray")]
    arr: [u8; 64], }

fn main() {
    let s = S { arr: [1; 64] };
    let j = serde_json::to_string(&s).unwrap();
    println!("{}", j);
    serde_json::from_str::<S>(&j).unwrap(); }

来源

这是在serde-big-array crate中发布的。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接