- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
这段代码完全按照我的意愿序列化了一个 32 字节的数组:
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq, PartialOrd, Ord)]
struct Hash([u8; 32]);
let hash = Hash([1u8; 32]);
let hash_bin = bincode::serialize(&hash).unwrap();
assert_eq!(hash_bin, [1u8; 32]);
它是如何工作的?
根据 https://serde.rs/impl-serializer.html有一个 serialize_bytes()
函数,但 bincode 版本会在数据前添加一个长度。
fn serialize_bytes(self, v: &[u8]) -> Result<()> {
O::IntEncoding::serialize_len(self, v.len())?;
self.writer.write_all(v).map_err(Into::into)
}
代码(以上)调用 Serialize 中的什么函数,将 32 个字节序列化为它们本身,没有长度前缀?
上下文:我正在为一个类型实现一个客户序列化,我希望它(在某些情况下)序列化字节数组,以便 bincode 将它们编码为没有长度前缀的字节。这是一个问题,因为调用 serialize_bytes()
添加了一个长度前缀。
我想了解默认情况下字节数组是如何序列化的,因为我不知道要调用哪个方法来代替 serialize_bytes()
来获取没有长度前缀的字节。
最佳答案
[u8; N]
和 [u8]
序列化?简而言之,这里是如何serde
1.0.151 实现了每个方法。 serialize_bytes
实际上不是 serde
的一部分所以它被视为一个序列。
// [T; N] is serialized as a tuple. However, this is only implemented for N 0 to 32 inclusively.
let mut seq = try!(serializer.serialize_tuple(N));
for e in self {
try!(seq.serialize_element(e));
}
seq.end()
// [T] is serialized as a sequence.
serializer.collect_seq(self)
方法 serialize_tuple
和 collect_seq
由您正在使用的特定序列化程序实现。
一个常见的问题是 serde
只实现 Serialize
/Deserialize
对于长度不超过 32 的数组。最简单的方法是使用像 serde_with
这样的 crate 。它添加了额外的序列化/反序列化实现,您可以附加到您的结构。这是从他们的文档中摘录的示例:
#[serde_as]
#[derive(Deserialize, Serialize)]
struct Arrays<const N: usize, const M: usize> {
#[serde_as(as = "[_; N]")]
constgeneric: [bool; N],
#[serde_as(as = "Box<[[_; 64]; N]>")]
nested: Box<[[u8; 64]; N]>,
#[serde_as(as = "Option<[_; M]>")]
optional: Option<[u8; M]>,
}
Serialize
执行序列化实际上很容易。 Serde 没有数组的概念,所以我们需要在 serialize_tuple
之间进行选择。或 serialize_seq
.在引擎盖下,唯一的区别是 serialize_seq
可能没有已知长度,所以我们可以选择 serialize_tuple
.
pub fn serialize<S, T, const N: usize>(this: &[T; N], serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
T: Serialize,
{
let mut seq = serializer.serialize_tuple(N)?;
for element in this {
seq.serialize_element(element)?;
}
seq.end()
}
Deserialize
另一方面,反序列化变得有点复杂。我们需要定义一个访问者,然后指定应该如何访问每个元素。我写了一个例子来说明如何在数组的一般情况下完成它,但这不是最佳解决方案,因为它首先反序列化到堆栈上。我还必须使用 unsafe
代码一次只初始化数组一个元素,但是unsafe
如果T: Default
,代码很容易被删除或者如果扩展数据结构像 Vec<T>
而是使用。通常,这更旨在作为对序列实现反序列化的指南。
pub fn deserialize<'de, D, T, const N: usize>(deserializer: D) -> Result<[T; N], D::Error>
where
D: Deserializer<'de>,
T: 'de + Deserialize<'de>,
{
deserializer.deserialize_seq(ArrayVisitor { _phantom: PhantomData })
}
struct ArrayVisitor<'de, T, const N: usize> {
_phantom: PhantomData<&'de [T; N]>,
}
impl<'de, T, const N: usize> Visitor<'de> for ArrayVisitor<'de, T, N>
where
T: Deserialize<'de>,
{
type Value = [T; N];
fn expecting(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "array of length {}", N)
}
fn visit_seq<A>(self, mut seq: A) -> Result<Self::Value, A::Error>
where
A: SeqAccess<'de>,
{
let mut array: MaybeUninit<[T; N]> = MaybeUninit::uninit();
for index in 0..N {
// Get next item as Result<Option<T>, A::Error>. Since we know
// exactly how many elements we should receive, we can flatten
// this to a Result<T, A::Error>.
let next = seq.next_element::<T>()
.and_then(|x| x.ok_or_else(|| Error::invalid_length(N, &self)));
match next {
Ok(x) => unsafe {
// Safety: We write into the array without reading any
// uninitialized memory and writes only occur within the
// array bounds at multiples of the array stride.
let array_base_ptr = array.as_mut_ptr() as *mut T;
ptr::write(array_base_ptr.add(index), x);
},
Err(err) => {
// Safety: We need to manually drop the parts we
// initialized before we can return.
unsafe {
let array_base_ptr = array.as_mut_ptr() as *mut T;
for offset in 0..index {
ptr::drop_in_place(array_base_ptr.add(offset));
}
}
return Err(err)
},
}
}
// Safety: We have completely initialized every element
unsafe { Ok(array.assume_init()) }
}
}
如果有人好奇如何derive(Deserialize)
适用于结构,我建议查看此 Rust Playground我在这里扩展了宏,然后清理了输出以使其更易于阅读。了解序列化/反序列化的工作原理确实有助于揭开这个过程的神秘面纱。
关于rust - serde/bincode 如何序列化字节数组?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67168397/
我正在尝试创建一个包含 int[][] 项的数组 即 int version0Indexes[][4] = { {1,2,3,4}, {5,6,7,8} }; int version1Indexes[
我有一个整数数组: private int array[]; 如果我还有一个名为 add 的方法,那么以下有什么区别: public void add(int value) { array[va
当您尝试在 JavaScript 中将一个数组添加到另一个数组时,它会将其转换为一个字符串。通常,当以另一种语言执行此操作时,列表会合并。 JavaScript [1, 2] + [3, 4] = "
根据我正在阅读的教程,如果您想创建一个包含 5 列和 3 行的表格来表示这样的数据... 45 4 34 99 56 3 23 99 43 2 1 1 0 43 67 ...它说你可以使用下
我通常使用 python 编写脚本/程序,但最近开始使用 JavaScript 进行编程,并且在使用数组时遇到了一些问题。 在 python 中,当我创建一个数组并使用 for x in y 时,我得
我有一个这样的数组: temp = [ 'data1', ['data1_a','data1_b'], ['data2_a','data2_b','data2_c'] ]; // 我想使用 toStr
rent_property (table name) id fullName propertyName 1 A House Name1 2 B
这个问题在这里已经有了答案: 关闭13年前。 Possible Duplicate: In C arrays why is this true? a[5] == 5[a] array[index] 和
使用 Excel 2013。经过多年的寻找和适应,我的第一篇文章。 我正在尝试将当前 App 用户(即“John Smith”)与他的电子邮件地址“jsmith@work.com”进行匹配。 使用两个
当仅在一个边距上操作时,apply 似乎不会重新组装 3D 数组。考虑: arr 1),但对我来说仍然很奇怪,如果一个函数返回一个具有尺寸的对象,那么它们基本上会被忽略。 最佳答案 这是一个不太理
我有一个包含 GPS 坐标的 MySQL 数据库。这是我检索坐标的部分 PHP 代码; $sql = "SELECT lat, lon FROM gps_data"; $stmt=$db->query
我需要找到一种方法来执行这个操作,我有一个形状数组 [批量大小, 150, 1] 代表 batch_size 整数序列,每个序列有 150 个元素长,但在每个序列中都有很多添加的零,以使所有序列具有相
我必须通过 url 中的 json 获取文本。 层次结构如下: 对象>数组>对象>数组>对象。 我想用这段代码获取文本。但是我收到错误 :org.json.JSONException: No valu
enter code here- (void)viewDidLoad { NSMutableArray *imageViewArray= [[NSMutableArray alloc] init];
知道如何对二维字符串数组执行修剪操作,例如使用 Java 流 API 进行 3x3 并将其收集回相同维度的 3x3 数组? 重点是避免使用显式的 for 循环。 当前的解决方案只是简单地执行一个 fo
已关闭。此问题需要 debugging details 。目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and the
我有来自 ASP.NET Web 服务的以下 XML 输出: 1710 1711 1712 1713
如果我有一个对象todo作为您状态的一部分,并且该对象包含数组列表,则列表内部有对象,在这些对象内部还有另一个数组listItems。如何更新数组 listItems 中 id 为“poi098”的对
我想将最大长度为 8 的 bool 数组打包成一个字节,通过网络发送它,然后将其解压回 bool 数组。已经在这里尝试了一些解决方案,但没有用。我正在使用单声道。 我制作了 BitArray,然后尝试
我们的数据库中有这个字段指示一周中的每一天的真/假标志,如下所示:'1111110' 我需要将此值转换为 boolean 数组。 为此,我编写了以下代码: char[] freqs = weekday
我是一名优秀的程序员,十分优秀!