本篇文章包含的内容

  • 1 结构体
    • 1.1 定义和初始化结构体
    • 1.2 Tuple Struct
    • 1.3 结构体方法(Rust 面向对象)
    • 1.4 关联函数
  • 2 枚举
    • 2.1 定义和使用枚举
    • 2.2 将数据附加到枚举的变体中
    • 2.3 Option 枚举
    • 2.4 模式匹配
      • 2.4.1 match语句
      • 2.4.2 if let语句


1 结构体

1.1 定义和初始化结构体

定义结构体必须显式指定结构体成员变量的类型。实例化结构体时需要对结构体的每个成员变量赋值,使用下面的写法定义和初始化结构体:

fn main() {let user1 = User {email: String::from("example@example.com"),username: String::from("Nikky"),active: true,sign_in_count: 556,};
}struct User {username: String,email: String,sign_in_count: u64,active: bool,
}

获取或者修改结构体实例中某个字段的值的方式和其他语言类似,使用点标记法即可,例如user1.email。上面例子中的user1默认也是不可变的,如果要修改某个字段,需要将这个结构体声明为可变的,即let mut user1 = User {...}。需要注意,如果结构体实例是可变的,那么其中的每个字段都是可变的,Rust不允许存在一部分字段不变,另一部分字段可变的结构体。

理所当然的,结构体也可以是函数的返回值:

fn build_user(email: String, username: String) -> User {User {email: emailusername: usernameactive: true,sign_in_count: 556,}
}

上面的例子中,初始化变量名和字段的变量名相同,那么初始化时可以用下面的方法简写(又是一个语法糖):

fn build_user(email: String, username: String) -> User {User {email,username,active: true,sign_in_count: 556,}
}

当你想通过基于某个结构体实例来初始化另一个结构体实例时,可以使用struct更新语法:

	let user1 = User {email: String::from("example@example.com"),username: String::from("Nikky"),active: true,sign_in_count: 556,};let user2 = User {email: String::from("another_example@example.com"),username: String::from("Tom"),..user1};

上面的例子中定义emailusername的类型为String而不是&str,使得该struct实例拥有其所有的数据(所有权),并且只要结构体实例是有效的,那么其中的数据也一定有效。struct中也可以存放引用,但是必须使用生命周期。生命周期保证只要struct实例是有效的,那么其中的引用也一定有效。生命周期的概念之后才会涉及。

自定义的struct往往没有Display方法,所以无法被println!()直接输出,如果要使用println!()输出,可以采用下面的写法:

#[derive(Debug)]	//
struct Rectangle {width: u32,length: u32,
}fn main() {let rect = Rectangle {width: 30,length: 50,};println!("{}", area(&rect));println!("{:?}", rect);     // 多行打印使用{:#?}
}fn area(rect: &Rectangle) -> u32 {rect.width * rect.length
}

需要注意,打印结构体时必须使用#[derive(Debug)]注解,使得Rectangle派生于Debug这个trait。

1.2 Tuple Struct

Rust中允许我们定一个类似tuple的struct,它就是Tuple Struct,它有一个整体的结构体名,但是其中的元素可以没有名字。适用于想给整个tuple起名,让其不同于其他的Tuple,但是又不需要给其中的每个元素起名的情况。

fn main() {struct Color(i32, i32, i32);struct Point(i32, i32, i32);let black = Color(0, 0, 0);let origin = Point(0, 0, 0);     // black和origin数据类似,但是是不同的类型
}

1.3 结构体方法(Rust 面向对象)

在C/C++中,class说白了就是拥有很多方法的struct,那么Rust中是否可以为结构体添加方法呢?当然可以。看下面这个例子:

struct Rectangle {width: u32,length: u32,
}impl Rectangle {fn area(&self) -> u32 {		// 借用self,不需要其所有权self.width * self.length}
}fn main() {let rect = Rectangle {width: 30,length: 50,};println!("{}", rect.area());
}

Rust的方法是在struct(或者enumtrait对象)的上下文中使用impl关键字(implementation)定义的。方法的第一个参数总是self

Rust会自动引用或者自动解引用,在调用方法时,rect.area()就相当于(&rect).area(),Rust会自动在变量前添加&&mut或者*。以便于实例匹配方法的签名。

一个impl里可以定义多个方法,也可以在多个impl块中定义,方法除了自身以外也可以有其他参数:

struct Rectangle {width: u32,length: u32,
}impl Rectangle {fn area(&self) -> u32 {self.width * self.length}fn can_hold(&self, other: &Rectangle) -> bool {self.width > other.width && self.length >= other.length}
}fn main() {let rect1 = Rectangle {width: 30,length: 50,};let rect2 = Rectangle {width: 10,length: 20,};println!("{}", rect1.area());println!("{}", rect1.can_hold(&rect2));
}

1.4 关联函数

impl块中也可以定义函数,这个函数不是对象的方法,但是它又与struct有一定的关联,我们把这种函数称为关联函数。通常使用关联函数作为对象的构造器,例如常用的String::from("")函数构造一个String。

#[derive(Debug)]
struct Rectangle {width: u32,length: u32,
}impl Rectangle {fn area(&self) -> u32 {self.width * self.length}fn can_hold(&self, other: &Rectangle) -> bool {self.width >= other.width && self.length >= other.length}fn square(size: u32) -> Rectangle {Rectangle {width: size,length: size,}}
}fn main() {let rect1 = Rectangle {width: 30,length: 50,};let rect2 = Rectangle {width: 10,length: 20,};let s = Rectangle::square(20);println!("{}", rect1.area());println!("{}", rect1.can_hold(&rect2));println!("{:?}", s);
}

2 枚举

2.1 定义和使用枚举

枚举(Enum)允许我们使用确定的可能值定义一个类型。枚举的可能的值称为变体。枚举的最基础作用其实就是提高代码的可读性,这和C/C++的枚举没有本质的区别。

enum IpAddrKind {V4,V6,
}fn main() {let four_ip = IpAddrKind::V4;let six_ip = IpAddrKind::V6;route(four_ip);route(six_ip);route(IpAddrKind::V4);
}fn route(ip_kind:IpAddrKind) {// some code...
}

自然地,枚举可以是结构体的成员变量:

enum IpAddrKind {V4,V6,
}struct IpAddr {ip_addr_kind: IpAddrKind,ip_addr: String,
}fn main() {let home = IpAddr {ip_addr: String::from("127.0.0.1"),ip_addr_kind: IpAddrKind::V4,};let loopback = IpAddr {ip_addr: String::from("::1"),ip_addr_kind: IpAddrKind::V6,};
}

2.2 将数据附加到枚举的变体中

枚举的变体中可以添加一些附加数据,这样可以让我们不用定义额外的结构体存储其他的信息:

enum IpAddrKind {V4(u8, u8, u8, u8),V6(String),
}fn main() {let home = IpAddrKind::V4(192, 168, 0, 1);let loopback = IpAddrKind::V6(String::from("::1"));
}

标准库的IpAddr就使用了类似的设计,不过标准库中枚举IpAddr中嵌入的是结构体struct。说明我么可以在枚举中嵌入任意的数据类型,甚至嵌入另外的枚举。枚举也可以定义方法,枚举方法的第一个参数也是self,调用依然采用.进行调用。看下面这个例子:

enum Message {Quit,Move {x: i32, y: i32},      // 关联一个匿名结构体Write(String),ChangeColor(i32, i32, i32),
}impl Message {fn call(&self) {// some code ...}
}fn main() {let q = Message::Quit;let m = Message::Move {x: 1, y: 2};let w = Message::Write(String::from("hello"));let c = Message::ChangeColor(0, 128, 0);q.call();
}

2.3 Option 枚举

Option枚举位于标准库中,它是预导入(Prelude)的。它主要是为了解决其他语言中一个值可能存在,并且可能是空值null的情况。Rust为了解决null的弊端,直接摒弃了null,取而代之的是Option枚举,使得开发者要想使用null,则必须同时处理值存在和不存在两种情况。

Option在标准库中的定义如下所示,T是泛型参数。Option枚举是预导入的,它的两个变体也是预导入的,所以程序中可以直接使用。

// 标准库中的定义
enum Option<T> {Some(T),None,
}
let some_number = Some(1);
let some_string = Some("hello");let absent_number: Option<i32> = None;

2.4 模式匹配

2.4.1 match语句

match是Rust中的一个强大的控制流运算符,它允许一个值与一系列模式进行依次匹配,这个“模式”可以是子面值,变量名或者是通配符,匹配成功后执行对应的代码块。

enum Coin {Penny,Nickel,Dime,Quarter,
}fn value_in_cents(coin: Coin) -> u8 {match coin {Coin::Penny => 1,Coin::Nickel => 5,Coin::Dime => 10,Coin::Quarter => 25,}
}fn main() {}

匹配到的模式可以关联被匹配对象的部分值,利用这个特性(语法糖?),我么可以方便地提取枚举中的值:

#[derive(Debug)]
enum UsState {Alabama,Alaska,
}enum Coin {Penny,Nickel,Dime,Quarter(UsState),       // 关联枚举数据
}fn value_in_cents(coin: Coin) -> u8 {match coin {Coin::Penny => 1,Coin::Nickel => 5,Coin::Dime => 10,Coin::Quarter(state) => {println!("Quarter from: {:?}", state);25},}
}fn main() {value_in_cents(Coin::Quarter(UsState::Alabama));value_in_cents(Coin::Quarter(UsState::Alaska));
}

在这里插入图片描述

Option枚举可以和match语句结合处理当数据为空的情况:

fn plus_one(number: Option<i32>) -> Option<i32> {match number {None => None,Some(i) => Some(i + 1)	// 返回也必须是Option<i32>}
}fn main() {let five = Some(5);let six = plus_one(five);let none = plus_one(None);
}

需要注意,match必须穷举所有的可能。如果没有穷举所有的可能编译器就会报错,但是有时候确实只需要处理其中一部分数据,我么需要一个default,Rust使用下划线通配符表示未提及的情况:

fn main() {let v = 1u8;match v {1 => println!("one!"),3 => println!("three!"),_ => {},    // 或者 _ => () }
}

2.4.2 if let语句

if let语句相当于match语句只需要处理一种情况时的语法糖,后面可以加else,写法如下:

fn main() {let v = Some(3);if let Some(i) = v {      // 注意这里是 =,并且被匹配变量必须写在后面println!("the number is {}", i);} else {println!("others");}
}

虽然看起来直接写if更简单,但是if let本质是模式匹配,除了控制流,它还有另一个重要的功能:提取枚举携带的值。所以它和普通的控制流语句if还是不同的。

在这里插入图片描述


  原创笔记,码字不易,欢迎点赞,收藏~ 如有谬误敬请在评论区不吝告知,感激不尽!博主将持续更新有关嵌入式开发、FPGA方面的学习笔记。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/93374.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/93374.shtml
英文地址,请注明出处:http://en.pswp.cn/web/93374.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++——分布式

文章目录一、什么是分布式&#xff1f;核心特点为什么需要分布式&#xff1f;分布式 vs 集中式常见分布式场景挑战与难点二、 简述下CAP理论2.1 简述2.2 详细三、 简述下分布式中的2PC2.1 详细3.2 简述三 、简述下Raft协议3.1 详细3.2 简述四 grpc框架4.1 RPC&#xff08;Remot…

Redis面试精讲 Day 20:Redis大规模部署性能调优

【Redis面试精讲 Day 20】Redis大规模部署性能调优 开篇 欢迎来到"Redis面试精讲"系列第20天&#xff01;今天我们将深入探讨Redis在大规模部署场景下的性能调优策略&#xff0c;这是高级工程师和架构师面试必考的核心知识点。本文将从操作系统配置、Redis参数调优…

[微服务]ELK Stack安装与配置全指南

目录 一、ELK相关介绍 1.1 什么是ELK Stack 1.2 ELK核心组件与功能 1.3 ELK优势 1.4 ES数据库结构对比SqlServer 二、安装ELK 2.1 window安装 2.2 Docker下环境搭建 2.2.1 安装7.16.3版本ElasticSearch 2.2.2 安装7.16.3版本Kibana : 2.2.3 安装8.0.0版本ElasticSea…

java项目怎么实现用户行为分析、漏斗转化、数据可视化报表。

在 Java 项目中实现用户行为分析、漏斗转化和数据可视化报表是一个系统性的工作&#xff0c;需要从数据采集、存储、分析到展示的完整链路设计。以下是一个可行的实现方案&#xff1a;1. 整体架构设计建议采用分层架构&#xff1a;数据采集层&#xff1a;收集用户行为数据数据存…

缓存元数据损坏操作步骤(lvmcache修复)

现象为:机械盘丢失cvol-cmeta卷如图所示,lvm逻辑卷中缺失缓存的lvm,这边以只读cache为例日志现象报错信息为:lvmcache_cvol failed manual repair required!lvmcache_cvol failed: manual repair required! 这类报错&#xff0c;本质上是 LVM cache 池&#xff08;cache-pool&…

使用CMAKE-GUI生成Visual Studio项目

使用CMAKE-GUI生成Visual Studio项目第一种&#xff0c;如果我们想把以Cmake构建的项目移植VS上&#xff0c;就可以使用Cmake来生成.sln文件 准备生成的目录文件先准备好我们要打包的源代码等文件&#xff08;放在resource下&#xff09;使用cmake-gui工具来构建&#xff08;命…

20道DOM相关前端面试题

DOM 相关面试题及答案 什么是 DOM&#xff1f;DOM 树的结构是怎样的&#xff1f; DOM&#xff08;文档对象模型&#xff0c;Document Object Model&#xff09;是 HTML/XML 文档的编程接口&#xff0c;将文档结构化为树形节点集合&#xff0c;允许程序动态访问和修改文档内容、…

CVE-2021-4300漏洞复现

Adminer是一个PHP编写的开源数据库管理工具&#xff0c;支持MySQL、MariaDB、PostgreSQL、SQLite、MS SQL、Oracle、Elasticsearch、MongoDB等数据库。在其版本1.12.0到4.6.2之间存在一处因为MySQL LOAD DATA LOCAL导致的文件读取漏洞。 一、伪造服务器 利用mysql-fake-serve…

【LeetCode题解】LeetCode 35. 搜索插入位置

【题目链接】 35. 搜索插入位置 【题目描述】 【题解】 通过题目可以知道这是一道经典的二分查找的题目&#xff0c;对于二分查找的题目&#xff0c;根据需要查找的两个边界点&#xff0c;分为两个不同的模板&#xff0c;如下图所示。 这道题要求在数组中查找目标值并返回其索…

RK3568 NPU RKNN(五):RKNN-ToolKit-lite2板端推理

文章目录1、前言2、目标3、安装RKNN-ToolKit-lite23.1、安装环境3.2、安装RKNN-ToolKit-lite23.3、验证4、完整的测试程序5、运行测试程序6、程序拆解7、总结1、前言 本文仅记录本人学习过程&#xff0c;不具备教学指导意义。 2、目标 之前提到过&#xff0c;RKNN-Toolkit2-…

二分查找。。

1 二分查找二分查找前提是数组有序。先令&#xff0c;left 0 , right 7mid (right left) / 2;如果mid的值大于要查找的值&#xff0c;则right mid - 1&#xff1b;如果小于&#xff0c;left mid 1&#xff1b;如果mid的值等于要查找的值&#xff0c;查找成功。重复步骤2…

Spring Ai 如何配置以及如何搭建

Spring Ai 如何配置以及如何搭建 解释什么是Spring ai 首先&#xff0c;我们用Spring ai 其实不是去了解他的LLM,以及底层用的一些东西&#xff0c;Spring AI,提供给我们的其实是对各种大模型快速调用&#xff0c;提供了大模型API的作用&#xff0c;Spring AI 的核心定位是提…

FCC认证三星XR头显加速全球量产,微美全息AI+AR技术引领智能眼镜硬件创新

据悉&#xff0c;三星(SSNGY.US)XR头显Project Moohan目前已获得美国FCC认证&#xff0c;FCC认证表明该款头显即将上市&#xff0c;之前三星财报会议也表明确认将于今年年底推出XR头显。此前有报道称&#xff0c;该设备将采用索尼旗舰级 OLEDoS 显示屏&#xff0c;像素密度高达…

洛谷P1595讲解(加强版)+错排讲解

前言接我原先的文章&#xff0c;因为一场考试&#xff0c;让我对这道题记忆深刻注&#xff1a;&#xff08;因为那道题&#xff0c;所以80分&#xff09;正文1.分析题目题目&#xff1a;某人写了 n 封信和 n 个信封&#xff0c;如果所有的信都装错了信封。求所有信都装错信封共…

提升化工制造质量的 7 种方法

尽管化工制造属于制造业的一个子类别&#xff0c;但它是一个广泛的范畴&#xff0c;涵盖了基础化学品、树脂和合成纤维、农药和化肥、涂料和粘合剂&#xff0c;甚至消费类化合物&#xff08;如肥皂和清洁化学品&#xff09;等所有领域。尽管这些细分领域差异巨大&#xff0c;但…

从“数据垄断”到“全民共建”:Dataparts如何重构智能时代的数据流通规则?

从“数据垄断”到“全民共建”&#xff1a;Dataparts如何重构智能时代的数据流通规则&#xff1f;在杭州某科技园区的会议室里&#xff0c;一场关于“AI大模型训练数据”的讨论正在激烈进行。某头部AI企业的技术总监指着屏幕上的“对话场景零件库”说&#xff1a;“过去我们花3…

31 HTB Union 机器 - 中等难度

第一阶段 侦查nmap扫描oxdfparrot$ nmap -p- --min-rate 10000 -oA scans/nmap-alltcp 10.10.11.128 Starting Nmap 7.80 ( https://nmap.org ) at 2021-11-19 08:29 EST Nmap scan report for 10.10.11.128 Host is up (0.092s latency). Not shown: 65534 filtered ports POR…

【数据分享】上市公司创新韧性数据(2007-2023)

数据介绍核心看点&#xff1a; 在复杂多变的市场环境中&#xff0c;企业如何通过创新维持竞争力&#xff1f;创新韧性是衡量企业在外部冲击下保持创新活力的关键指标。本文分享2007-2023年上市公司创新韧性数据&#xff0c;为研究企业抗风险能力提供核心支持。数据概览数据名称…

服务器配置开机自启动服务

一、配置启动文件sudo vim /etc/systemd/system/smartailab-backend.service sudo vim /etc/systemd/system/reall3d-frontend.servicesudo vim /etc/systemd/system/Culture_Liquor-backend.servicevim /etc/systemd/system/Culture_Liquor-backend.service内容&#xff1a;[U…

Ubuntu 25.04更新了哪些内容揭秘

2025年4月,Canonical正式推出Ubuntu 25.04 版本,代号"Plucky Puffin(勇敢的海鹦)"。此次发布围绕AI算力强化、桌面交互革新与跨架构支持三大核心方向展开,为开发者、创作者及企业用户带来多项突破性升级。 一、核心系统更新 systemd v257.4带来了重要的上游更新…