让 serde 只为人类可读的序列化程序生成十六进制字符串?
Make serde only produce hex strings for human-readable serialiser?
我目前正在使用 serde-hex。
use serde_hex::{SerHex,StrictPfx,CompactPfx};
#[derive(Debug,PartialEq,Eq,Serialize,Deserialize)]
struct Foo {
#[serde(with = "SerHex::<StrictPfx>")]
bar: [u8;4],
#[serde(with = "SerHex::<CompactPfx>")]
bin: u64
}
fn it_works() {
let foo = Foo { bar: [0,1,2,3], bin: 16 };
let ser = serde_json::to_string(&foo).unwrap();
let exp = r#"{"bar":"0x00010203","bin":"0x10"}"#;
assert_eq!(ser,exp);
// this fails
let binser = bincode::serialize(&foo).unwrap();
let binexp: [u8; 12] = [0, 1, 2, 3, 16, 0, 0, 0, 0, 0, 0, 0];
assert_eq!(binser,binexp);
}
失败:
thread 'wire::era::tests::it_works' panicked at 'assertion failed: `(left == right)`
left: `[10, 0, 0, 0, 0, 0, 0, 0, 48, 120, 48, 48, 48, 49, 48, 50, 48, 51, 4, 0, 0, 0, 0, 0, 0, 0, 48, 120, 49, 48]`,
right: `[0, 1, 2, 3, 16, 0, 0, 0, 0, 0, 0, 0]`', src/test.rs:20:9
因为它已将二进制代码的值扩展为十六进制字符串。
我有很多结构需要用 serde_json 和 bincode 进行序列化。 serde_hex
完全满足我对 JSON 序列化的需求。使用 bincode serde-hex 时,仍然会将数组转换为十六进制字符串,这是不需要的。
我注意到 secp256k1 uses d.is_human_readable()
。
如何让 serde_hex
仅适用于 serde_json
而对 bincode
忽略?
可与 serde's with-attribute 一起使用的函数的实现大部分是样板文件,如下所示。
这仅区分人类可读的格式和其他格式。如果您需要更细粒度的控制,您可以在线程局部变量上分支。
fn serialize_hex<S>(v: &u64, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
if serializer.is_human_readable() {
serde_hex::SerHex::<serde_hex::StrictPfx>::serialize(v, serializer)
} else {
v.serialize(serializer)
}
}
// use like
// #[serde(serialize_with = "serialize_hex")]
// bin: u64
可以通过将 u64
变为通用来改进该代码段。
我目前正在使用 serde-hex。
use serde_hex::{SerHex,StrictPfx,CompactPfx};
#[derive(Debug,PartialEq,Eq,Serialize,Deserialize)]
struct Foo {
#[serde(with = "SerHex::<StrictPfx>")]
bar: [u8;4],
#[serde(with = "SerHex::<CompactPfx>")]
bin: u64
}
fn it_works() {
let foo = Foo { bar: [0,1,2,3], bin: 16 };
let ser = serde_json::to_string(&foo).unwrap();
let exp = r#"{"bar":"0x00010203","bin":"0x10"}"#;
assert_eq!(ser,exp);
// this fails
let binser = bincode::serialize(&foo).unwrap();
let binexp: [u8; 12] = [0, 1, 2, 3, 16, 0, 0, 0, 0, 0, 0, 0];
assert_eq!(binser,binexp);
}
失败:
thread 'wire::era::tests::it_works' panicked at 'assertion failed: `(left == right)`
left: `[10, 0, 0, 0, 0, 0, 0, 0, 48, 120, 48, 48, 48, 49, 48, 50, 48, 51, 4, 0, 0, 0, 0, 0, 0, 0, 48, 120, 49, 48]`,
right: `[0, 1, 2, 3, 16, 0, 0, 0, 0, 0, 0, 0]`', src/test.rs:20:9
因为它已将二进制代码的值扩展为十六进制字符串。
我有很多结构需要用 serde_json 和 bincode 进行序列化。 serde_hex
完全满足我对 JSON 序列化的需求。使用 bincode serde-hex 时,仍然会将数组转换为十六进制字符串,这是不需要的。
我注意到 secp256k1 uses d.is_human_readable()
。
如何让 serde_hex
仅适用于 serde_json
而对 bincode
忽略?
可与 serde's with-attribute 一起使用的函数的实现大部分是样板文件,如下所示。 这仅区分人类可读的格式和其他格式。如果您需要更细粒度的控制,您可以在线程局部变量上分支。
fn serialize_hex<S>(v: &u64, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
if serializer.is_human_readable() {
serde_hex::SerHex::<serde_hex::StrictPfx>::serialize(v, serializer)
} else {
v.serialize(serializer)
}
}
// use like
// #[serde(serialize_with = "serialize_hex")]
// bin: u64
可以通过将 u64
变为通用来改进该代码段。