当前位置:首页 > 科技  > 软件

在Rust中使用Serde的详细指南

来源: 责编: 时间:2024-03-26 09:33:11 261观看
导读在处理HTTP请求时,我们总是需要在一种数据结构(可以是enum、struct等)和一种可以存储或传输并稍后重建的格式(例如JSON)之间来回转换。Serde是一个库(crate),用于高效、通用地序列化和反序列化Rust数据结构。在本文中,将

在处理HTTP请求时,我们总是需要在一种数据结构(可以是enum、struct等)和一种可以存储或传输并稍后重建的格式(例如JSON)之间来回转换。u5m28资讯网——每日最新资讯28at.com

Serde是一个库(crate),用于高效、通用地序列化和反序列化Rust数据结构。在本文中,将详细介绍如何使用Serde对数据结构进行序列化和反序列化操作。u5m28资讯网——每日最新资讯28at.com

让我们从一个简单的结构体Student开始,它的定义如下所示,并进行初始化。u5m28资讯网——每日最新资讯28at.com

use serde::{Serialize, Deserialize};#[derive(Debug, Clone, Serialize, Deserialize)]struct Student {    pub name: String,     pub student_id: String,}let student = Student{name:"tom".to_owned(), student_id:"J19990".to_owned()};

u5m28资讯网——每日最新资讯28at.com

u5m28资讯网——每日最新资讯28at.com

约定

对于上面的示例,如果我们使用serde_json::to_string(&student)将其转换为JSON字符串,那么输出将如下所示。u5m28资讯网——每日最新资讯28at.com

{  "name": "tom",  "student_id": "J19990"}

u5m28资讯网——每日最新资讯28at.com

看起来太棒了!然而,这依赖于发送HTTP请求的内容,很有可能会与Rust中的数据结构有不同的大小写约定。u5m28资讯网——每日最新资讯28at.com

基本上有两种方法可以进行约定:可以重命名字段;也可以对整个结构应用大小写约定。u5m28资讯网——每日最新资讯28at.com

例如,我们实际上希望使用studentId而不是student_id作为字段名。u5m28资讯网——每日最新资讯28at.com

方法1:使用#[serde(rename="")重命名单个字段。u5m28资讯网——每日最新资讯28at.com

struct Student {    pub name: String,     #[serde(rename="studentId")    pub student_id: String,}

u5m28资讯网——每日最新资讯28at.com

方法2:使用#[serde(rename_all="camelCase")将大小写约定驼峰形式,应用于整个结构体。u5m28资讯网——每日最新资讯28at.com

#[serde(rename_all = "camelCase")]struct Student {    pub name: String,     pub student_id: String,}

u5m28资讯网——每日最新资讯28at.com

任何一种方法都会给出以下输出:u5m28资讯网——每日最新资讯28at.com

{  "name": "tom",  "studentId": "J19990"}

u5m28资讯网——每日最新资讯28at.com

除了camelCase之外,还可以应用其他的约定。取值为“lowercase, UPPERCASE, PascalCase, camelCase, snake_case, SCREAMING_SNAKE_CASE, kebab-case, SCREAMING-KEBAB-CASE”。u5m28资讯网——每日最新资讯28at.com

u5m28资讯网——每日最新资讯28at.com

u5m28资讯网——每日最新资讯28at.com

Skip

Skip可用于不希望序列化或反序列化的字段。下面是一个简单的例子。让我们给Student添加birth_year和age。u5m28资讯网——每日最新资讯28at.com

struct Student {    pub name: String,     pub student_id: String,    pub birth_year: u32,    pub age: u32,}

u5m28资讯网——每日最新资讯28at.com

我们可能希望动态更新年龄,因此需要对学生birth_year的引用。但是,当我们发送请求时,应该只显示age字段,这可以使用#[serde(skip)]来解决。u5m28资讯网——每日最新资讯28at.com

struct Student {    pub name: String,     pub student_id: String,    #[serde(skip)]    pub birth_year: u32,    pub age: u32,}

u5m28资讯网——每日最新资讯28at.com

通过这样做,我们的JSON对象将变成:u5m28资讯网——每日最新资讯28at.com

{  "name": "tom",  "studentId": "J19990",  "age": 123}

u5m28资讯网——每日最新资讯28at.com

u5m28资讯网——每日最新资讯28at.com

Skip If

最常见的两种使用方法是作用于Option字段和空的vector字段。u5m28资讯网——每日最新资讯28at.com

Option

假设我们有一个middle_name: Option<String>字段,如果我们想在学生没有这个字段的情况下跳过这个字段,我们可以这样做。u5m28资讯网——每日最新资讯28at.com

struct Student {    pub name: String,     pub student_id: String,    #[serde(skip_serializing_if="Option::is_none")]    pub middle_name: Option<String>}

u5m28资讯网——每日最新资讯28at.com

这将为带有或不带有中间名的学生生成如下JSON:u5m28资讯网——每日最新资讯28at.com

// 没有中间名{  "name": "tom",  "studentId": "J19990",}// 有中间名{  "name": "tom",  "studentId": "J19990",  "middleName": "middle"}

u5m28资讯网——每日最新资讯28at.com

Vector

例如,我们为student结构体提供了pets: Vec<String>字段。由于学生不必拥有宠物,它可以是一个空向量。u5m28资讯网——每日最新资讯28at.com

要跳过对空向量的序列化,可以向字段添加以下属性。u5m28资讯网——每日最新资讯28at.com

#[serde(skip_serializing_if="Vec::is_empty")]pub pets: Vec<String>,

u5m28资讯网——每日最新资讯28at.com

有属性和没有属性之间的输出差异如下所示。u5m28资讯网——每日最新资讯28at.com

// 没有属性{  "name": "tom",  "studentId": "J19990",  "pets": []}// 有属性{  "name": "tom",  "studentId": "J19990"}

u5m28资讯网——每日最新资讯28at.com

u5m28资讯网——每日最新资讯28at.com

Flatten

让我们创建一个名为SideInfo的新结构体,并将Student结构体更改为以下内容。u5m28资讯网——每日最新资讯28at.com

#[derive(Debug, Clone, Serialize, Deserialize)]#[serde(rename_all = "camelCase")]struct Student {    pub name: String,     pub student_id: String,    #[serde(skip_serializing_if="Option::is_none")]    pub side_info: Option<SideInfo>}#[derive(Debug, Clone, Serialize, Deserialize, Default)]#[serde(rename_all = "camelCase")]struct SideInfo {    #[serde(skip_serializing_if="Option::is_none")]    pub pets: Option<Vec<String>>,    #[serde(skip_serializing_if="Option::is_none")]    pub address: Option<String>,}

u5m28资讯网——每日最新资讯28at.com

让我们创建一个新的Studentu5m28资讯网——每日最新资讯28at.com

let student = Student{name:"dan".to_owned(), student_id: "1".to_owned(), side_info:Some(SideInfo{address:Some("47 street".to_owned()), ..Default::default()})};

u5m28资讯网——每日最新资讯28at.com

并输出它的JSON字符串:u5m28资讯网——每日最新资讯28at.com

{  "name": "dan",  "studentId": "1",  "sideInfo": {    "address": "47 street"  }}

u5m28资讯网——每日最新资讯28at.com

如你所见,地址字段嵌套在sideInfo中。我们可以通过将属性flatten添加到Student结构体中的sideInfo字段上,来消除嵌套。u5m28资讯网——每日最新资讯28at.com

#[serde(skip_serializing_if="Option::is_none", flatten)]pub side_info: Option<SideInfo>

u5m28资讯网——每日最新资讯28at.com

就会变成:u5m28资讯网——每日最新资讯28at.com

{  "name": "dan",  "studentId": "1",  "address": "47 street"}

u5m28资讯网——每日最新资讯28at.com

u5m28资讯网——每日最新资讯28at.com

枚举上的标签与非标签

假设我们有一个StudentList enum,如下所示:u5m28资讯网——每日最新资讯28at.com

enum StudentList {    Student1(Student),     Student2(Student)}

u5m28资讯网——每日最新资讯28at.com

定义学生名单u5m28资讯网——每日最新资讯28at.com

let student1 = Student{name:"tom".to_owned(), student_id:"J19990".to_owned(), pets: vec![], middle_name:Some("middle".to_owned())};let student2 = Student{name:"dan".to_owned(), student_id:"J19990".to_owned(), pets: vec![], middle_name:Some("middle".to_owned())};let student_list = vec![StudentList::Student1(student1), StudentList::Student2(student2)];

u5m28资讯网——每日最新资讯28at.com

如果我们像现在一样打印出JSON,它将如下所示,它是有标签的,是serde的默认行为。u5m28资讯网——每日最新资讯28at.com

[  {    "Student1": {      "name": "tom",      "studentId": "J19990",      "pets": [],      "middleName": "middle"    }  },  {    "Student2": {      "name": "dan",      "studentId": "J19990",      "pets": [],      "middleName": "middle"    }  }]

u5m28资讯网——每日最新资讯28at.com

如果你希望所有标签都具有相同的名称,例如Student,该怎么办呢?你可能认为可以使用rename_all来实现这一点,但实际上并非如此,应该手动重命名枚举中的每个变体。u5m28资讯网——每日最新资讯28at.com

#[derive(Debug, Clone, Serialize, Deserialize)]enum StudentList {    #[serde(rename="Student")]    Student1(Student),     #[serde(rename="Student")]    Student2(Student)}

u5m28资讯网——每日最新资讯28at.com

输出如下:u5m28资讯网——每日最新资讯28at.com

[  {    "Student": {      "name": "tom",      "studentId": "J19990",      "pets": [],      "middleName": "middle"    }  },  {    "Student": {      "name": "dan",      "studentId": "J19990",      "pets": [],      "middleName": "middle"    }  }]

u5m28资讯网——每日最新资讯28at.com

不加标签

如果我们只想要一个简单的学生数组,而不显示枚举变量名称,该怎么办?我们可以通过向枚举中添加#[serde(untagged)]属性来实现这一点。通过这样做,我们的输出将变成:u5m28资讯网——每日最新资讯28at.com

[  {    "name": "tom",    "studentId": "J19990",    "pets": [],    "middleName": "middle"  },  {    "name": "dan",    "studentId": "J19990",    "pets": [],    "middleName": "middle"  }]

u5m28资讯网——每日最新资讯28at.com

内部标签

枚举的另一种表示形式是内部标签,让我们创建一个包含不同学生类型的新枚举,我们将有班长、副班长和普通学生。u5m28资讯网——每日最新资讯28at.com

#[derive(Debug, Clone, Serialize, Deserialize)]#[serde(tag = "type", rename_all="camelCase")]enum StudentType {    Regular(Student),     Leader(Student),     SubLeader(Student)}

u5m28资讯网——每日最新资讯28at.com

指定serde(tag = "type")将允许我们在内容中使用标签来识别我们正在处理的变体,如下所示:u5m28资讯网——每日最新资讯28at.com

[  {    "type": "leader",    "name": "tom",    "studentId": "J19990",    "pets": [],    "middleName": "middle"  },  {    "type": "regular",    "name": "dan",    "studentId": "J19990",    "pets": [],    "middleName": "middle"  }]

u5m28资讯网——每日最新资讯28at.com

相邻标签

表示标签和内容作为同一对象中的两个字段彼此相邻。将枚举的属性修改如下:u5m28资讯网——每日最新资讯28at.com

#[serde(tag = "type", content = "student", rename_all="camelCase")]

u5m28资讯网——每日最新资讯28at.com

json数据变成:u5m28资讯网——每日最新资讯28at.com

[  {    "type": "leader",    "student": {      "name": "tom",      "studentId": "J19990",      "pets": [],      "middleName": "middle"    }  },  {    "type": "regular",    "student": {      "name": "dan",      "studentId": "J19990",      "pets": [],      "middleName": "middle"    }  }]

u5m28资讯网——每日最新资讯28at.com

可以用Serde做很多很多事情。u5m28资讯网——每日最新资讯28at.com

u5m28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-79291-0.html在Rust中使用Serde的详细指南

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: LG Innotek 宣布进军半导体玻璃基板业务

下一篇: 一图看懂常见 AWS 网络架构图

标签:
  • 热门焦点
  • 6月iOS设备好评榜:第一蝉联榜首近一年

    作为安兔兔各种榜单里变化最小的那个,2023年6月的iOS好评榜和上个月相比没有任何排名上的变化,仅仅是部分设备好评率的下降,长年累月的用户评价和逐渐退出市场的老款机器让这
  • CSS单标签实现转转logo

    转转品牌升级后更新了全新的Logo,今天我们用纯CSS来实现转转的新Logo,为了有一定的挑战性,这里我们只使用一个标签实现,将最大化的使用CSS能力完成Logo的绘制与动画效果。新logo
  • 三分钟白话RocketMQ系列—— 如何发送消息

    我们知道RocketMQ主要分为消息 生产、存储(消息堆积)、消费 三大块领域。那接下来,我们白话一下,RocketMQ是如何发送消息的,揭秘消息生产全过程。注意,如果白话中不小心提到相关代
  • 10天营收超1亿美元,《星铁》比《原神》差在哪?

    来源:伯虎财经作者:陈平安即便你没玩过《原神》,你一定听说过的它的大名。恨它的人把《原神》开服那天称作是中国游戏史上最黑暗的一天,有粉丝因为索尼在PS平台上线《原神》,怒而
  • 自律,给不了Keep自由!

    来源 | 互联网品牌官作者 | 李大为编排 | 又耳 审核 | 谷晓辉自律能不能给用户自由暂时不好说,但大概率不能给Keep自由。近日,全球最大的在线健身平台Keep正式登陆港交所,努力
  • 签约井川里予、何丹彤,单视频点赞近千万,MCN黑马永恒文希快速崛起!

    来源:视听观察永恒文希传媒作为一家MCN公司,说起它的名字来,可能大家会觉得有点儿陌生,但是说出来下面一串的名字之后,或许大家就会感到震惊,原来这么多网红,都签约这家公司了。根
  • 阿里瓴羊One推出背后,零售企业迎数字化新解

    作者:刘旷近年来随着数字经济的高速发展,各式各样的SaaS应用服务更是层出不穷,但本质上SaaS大多局限于单一业务流层面,对用户核心关切的增长问题等则没有提供更好的解法。在Saa
  • 华为Mate60标准版细节曝光:经典星环相机模组回归

    这段时间以来,关于华为新旗舰的爆料日渐密集。据此前多方爆料,今年华为将开始恢复一年双旗舰战略,除上半年推出的P60系列外,往年下半年的Mate系列也将
  • AI芯片初创公司Tenstorrent获三星和现代1亿美元投资

    Tenstorrent是一家由芯片行业资深人士Jim Keller领导的加拿大初创公司,专注于开发人工智能芯片,该公司周三表示,已经从现代汽车集团和三星投资基金等
Top