Rust 学习笔记 - 详解数据类型
前言
任何一门编程语言几乎都脱离不了:变量、基本类型、函数、注释、循环、条件判断,这是一门编程语言的语法基础,只有当掌握这些基础语法及概念才能更好的学习 Rust。
标量类型(Scalar Types)
在 Rust 中,标量类型代表单个值,Rust 有四种基本的标量类型:整型、浮点型、布尔类型和字符类型。
整数类形
整数是没有小数部分的数字,可以是正数、负数或零。整数类型分为两大类:有符号和无符号。有符号整数可以存储包括负数在内的值,而无符号整数只能存储零和正数。
fn main() {let x: i32 = -123; // 有符号整数let y: u32 = 456; // 无符号整数
}
整数类型参照:
| 长度 | 有符号整数 | 无符号整数 |
|---|---|---|
| 8-bit | i8 | u8 |
| 16-bit | i16 | u16 |
| 32-bit | i32 | u32 |
| 64-bit | i64 | u64 |
| 128-bit | i128 | u128 |
| arch | isize | usize |
isize 和 usize 两种整数类型的位长度取决于所运行的平台,如果是 32 位架构的处理器将使用 32 位长度整型 (即 i32 / u32),64 位架构的处理器上使用 64 为长度整型 (即 i64 / u64)。
整数类型具有明确的大小(取值范围),每个有符号整数类型可以存储 − 2 n − 1 -2^{n-1} −2n−1 到 2 n − 1 − 1 2^{n-1} - 1 2n−1−1,其中 n 为数据类型的位数,每个无符号整数类型可以存储 0 到 2 n − 1 2^{n} - 1 2n−1。
举个🌰:
i8可以存储 − 2 8 − 1 -2^{8-1} −28−1 到 2 8 − 1 − 1 2^{8-1} - 1 28−1−1 的数字,即: -128 到 127。u8可以存储 0 到 2 8 − 1 2^{8} - 1 28−1 的数字,即: 0 到 255。
字面量表示:
整数字面量可以通过不同的进制来表示,包括十进制、十六进制、八进制、二进制,以及字节(只适用于 u8)。
| 进制 | 前缀 | 示例 |
|---|---|---|
| 十进制 | 无 | 98_222 |
| 十六进制 | 0x | 0xff |
| 八进制 | 0o | 0o77 |
| 二进制 | 0b | 0b111_0000 |
| 字节(u8 专用) | b | b'A' |
注意:在 Rust 中,数字可以使用 _ 作为可视分隔符来提高可读性,编译器在处理数值时会忽略这些 _。
浮点类型
浮点数是用来处理带有小数部分的数值。Rust 提供了两种基本的浮点类型,分别是 f32 和 f64,其中 f32 是单精度浮点类型,f64 是双精度浮点类型。Rust 中的浮点类型遵循 IEEE-754 标准。
fn main() {let x: f32 = 3.14; // f32 单精度浮点数let y: f64 = 2.71828; // f64 双精度浮点数let z = 2.71828; // 默认是 f64 双精度浮点数
}
f32 类型
f32 类型的浮点数是单精度浮点数,占用 32 位(4 字节)的内存空间。它的范围大约是 1.4E-45 到 3.4E+38,精度大约有 6-7 位十进制数。f32 类型的浮点数对于需要较高性能但不需要非常高精度的场景来说是一个不错的选择,因为它在处理速度和内存使用上都比 f64 更加高效。
f64 类型
f64 类型的浮点数是双精度浮点数,占用 64 位(8 字节)的内存空间。它的范围大约是 4.9E-324 到 1.8E+308,精度大约有 15-16 位十进制数。相比 f32,f64 提供了更大的数值范围和更高的精度,适合对数值精度要求较高的计算任务。由于在现代 CPU 架构中,f64 的性能通常也非常高效,所以 Rust 默认的浮点类型就是 f64。
浮点数的字面量表示法
浮点数可以使用字面量表示法来表示,例如:
- 直接给出小数:
3.14 - 使用科学记数法:
2.5e10或2.5E10表示2.5 × 10¹⁰ - 在数字中使用下划线以提高可读性:
1_000.75_001
特点和使用注意事项
- 浮点数类型具备加 (
+)、减 (-)、乘 (*)、除 (/)、求余 (%) 等算术运算能力。 - 浮点数在运算时可能会出现“舍入错误”,这是因为许多实数不能被精确表示为
f32或f64类型的数值。这类舍入错误是所有使用 IEEE-754 标准的浮点数表示法的编程语言的通病。 - 浮点数比较需要特别注意,直接比较两个浮点数是否相等 (==) 可能会因为舍入误差导致不符合预期的结果。在涉及浮点数比较的时候,通常需要判断两个数的差值是否足够小。
布尔类型
布尔(Boolean)类型用 bool 表示,它是最简单的类型,只有两个值:true 和 false。布尔类型通常用于执行逻辑操作,条件判断和控制流程(例如,if 条件语句和循环控制)
fn main() {let t: bool = true;let f: bool = false;
}
基础
- 类型名为
bool。 - 取值范围仅为
true或false。 - 布尔值用一个字节(
1 byte或8 bits)来存储
字面量表示法
true表示逻辑真。false表示逻辑假。
操作
布尔类型支持多种逻辑操作,包括但不限于:
- 逻辑与 (
&&): 当两个操作数都为true时,返回true;否则返回false。 - 逻辑或 (
||): 只要一个操作数为true,就返回true;如果两个都为false,则返回false。 - 逻辑非 (
!): 如果操作数为true,返回false;如果为false,返回true。
let a = true;
let b = false;let and_result = a && b; // 返回 false
let or_result = a || b; // 返回 true
let not_result = !a; // 返回 false
字符类型
字符类型(char)用于表示单个 Unicode 标量值,这意味着它可以表示比 ASCII 更广泛的字符集。在 Rust 中,字符(char)是通过单引号(')来表示的,而字符串(String 或 &str)是通过双引号(")来表示的。
fn main() {let c: char = 'z';let z: char = 'ℤ'; // Unicode值let heart_eyed_cat = '😻';
}
基础
- 类型名为
char。 char类型在 Rust 中是四个字节的大小,即 32 位(与 UTF-32 编码相同)。这是因为char需要能表示任意一个 Unicode 标量值,其范围从U+0000到U+D7FF和U+E000到U+10FFFF。- 每个字符是单独的
char类型实例,并且占用 4 个字节的存储空间。这与其他一些语言中的字符类型(例如 C/C++ 中的char)不同,其通常是基于 ASCII 并且仅占用 1 个字节。
字面量表示法
- 普通字符:
'a','Z','7'等。 - 特殊字符:可以使用转义序列表示,如换行符
'\n',制表符'\t',单引号'\'',反斜杠'\\'等。 - Unicode 字符:使用
\u{}转义和大括号内的十六进制数值来表示,如'好'可以用'\u{597D}'表示。
操作
char类型的值可以直接参与比较操作(==,!=,<,>, 等)。char类型拥有多种方法用于检查字符的属性(例如is_alphabetic,is_numeric等)。
let c1 = 'A';
let c2 = '\u{597D}'; // 表示 "好"if c1.is_alphabetic() {println!("{} 是字母", c1);
}if c2.is_numeric() {println!("{} 是数字", c2);
}
复合类型(Compound Types)
复合类型可以将多个值组合成一个类型。Rust 主要有两种复合类型:元组(Tuple)和数组(Array)。
元组(Tuple)
元组是可以包含多个不同类型值的一种集合。元组的长度固定,一旦声明,它的长度不会改变。
声明元组
let tup: (i32, f64, char) = (500, 6.4, 'y');
在这个例子中,tup 是一个元组,包含了一个 i32 类型的整数、一个 f64 类型的浮点数,以及一个 char 类型的字符。
元组解构
元组可以被解构(destructured),为其内部的每个值匹配一个变量名称。
let tup = (500, 6.4, 1);
let (x, y, z) = tup;
println!("The value of y is: {}", y); // 输出 6.4
通过索引访问元组元素
也可以使用索引访问元组中的元素。
let five_hundred = tup.0;
let six_point_four = tup.1;
let y = tup.2;
数组(Array)
数组是有多个相同类型值组成的集合。与元组一样,数组的长度也是固定的,Rust 的数组一旦声明,它的长度不能改变。
声明数组
let a = [1, 2, 3, 4, 5];
这里,a 是一个整型数组,包含五个整数。
带类型注解的数组
你也可以在声明数组时添加类型注解,指明数组中的元素类型以及数组的长度。
let a: [i32; 5] = [1, 2, 3, 4, 5];
初始化数组
如果数组中的每个元素都是相同的值,可以使用以下方式进行初始化。
let a = [3; 5]; // 等同于 let a = [3, 3, 3, 3, 3];
这里 a 是一个整型数组,包含五个都是数字3的元素。
访问数组元素
通过索引来访问数组中的元素。
let first = a[0]; // 访问第一个元素
let second = a[1]; // 访问第二个元素
在访问时,如果索引超出了数组的边界,Rust 会在编译时或运行时(取决于索引是否为常量表达式)抛出错误,这是 Rust 的安全性特性之一。
选择元组还是数组?
- 当想要在一个复合类型中包含多个不同类型的值时,应当使用元组。
- 当需要一个包含多个相同类型值的集合时,应当使用数组。
元组非常适合用于有结构的数据,而数组非常适合用于有相同数据类型要求的连续数据序列。通过使用元组和数组,可以创建出符合你需要的各种数据结构。
自定义类型(Custom Types)
自定义类型主要指的是通过使用结构体(struct)和枚举(enum)来创建的数据类型。这两种类型允许开发者定义和使用更丰富且符合业务逻辑的数据结构。
结构体(Struct)
结构体是将零个或多个不同类型的数据聚合到一个复合类型中的一种方式。他们在概念上类似于其他语言中的类(但没有继承功能),是用于创建自定义数据类型的集合。
声明结构体
struct User {username: String,email: String,sign_in_count: u64,active: bool,
}
在这个例子中,User 结构体包含了四个字段,每个字段都有指定的类型。
实例化结构体
要使用结构体,你需要创建其实例并为其字段提供相应的值。
let user1 = User {email: String::from("admin@example.com"),username: String::from("ziyang"),active: true,sign_in_count: 1,
};
可以通过派生特性 #[derive(Debug)] 来允许结构体实例在使用 println! 宏时使用 {:?} 或 {:#?} 格式化输出。
枚举(Enum)
枚举允许定义一个类型,它可以是几个不同的变体中的一个。枚举在那些一次只能有一个值从多个可能的值中选取的情况下特别有用。
声明枚举
enum IpAddrKind {V4,V6,
}
这里,IpAddrKind 枚举有两个变体:V4 和 V6。
枚举也可以关联数据。
enum IpAddr {V4(String),V6(String),
}
甚至每个枚举变体关联的数据都可以有不同类型。
enum Message {Quit,Move { x: i32, y: i32 },Write(String),ChangeColor(i32, i32, i32),
}
枚举类的使用
let four = IpAddrKind::V4;
let six = IpAddrKind::V6;
用 match 控制流运算符来处理枚举有助于确保你考虑到了所有可能的情况。
Never (!)
Rust 中有一个表示"永不返回"的特殊类型,称作 ! 类型,也就是所谓的 “Never” 类型。这个类型用于那些不返回任何值的函数,或者那些不会正常返回,因为它们会无限循环或者结束当前进程的函数。
Never 类型用于表示永不返回的函数。这有助于 Rust 进行更严格的类型检查和控制流分析。
示例:
fn forever() -> ! {loop {// 无限循环,永不返回}
}
在这个示例中,forever 函数有一个 ! 返回类型,表示此函数将永远不会返回一个值。
Never 类型 (!) 的用途
-
控制流运算符:
!类型主要与 Rust 中的match表达式一起用于保证所有可能情况都已处理。如果match的一个分支结束于一个永不返回的函数,Rust 知道不需要返回值。这就是所谓的"穷尽性检查"(exhaustiveness checking)。 -
错误处理:
经常与panic!宏一起使用,它会使当前线程崩溃,并可以带有一个错误消息。由于panic!永远不会返回,它的返回类型是!。
结语
本章深入讲解了 Rust 中的标量类型、复合类型、自定义类型以及特殊的 never 类型。此外,还有指针类型、动态大小类型、函数类型等其他重要的数据类型将在后续文章中陆续进行逐一介绍。
相关文章:
Rust 学习笔记 - 详解数据类型
前言 任何一门编程语言几乎都脱离不了:变量、基本类型、函数、注释、循环、条件判断,这是一门编程语言的语法基础,只有当掌握这些基础语法及概念才能更好的学习 Rust。 标量类型(Scalar Types) 在 Rust 中ÿ…...
构建本地yum源
下载repo数据文件 根据需要修改下载路径和reposync参数 #!/bin/bashlocal_path/repo/remote/rhel9 enabled_repos$(yum repolist enabled | awk NR>3{print $1}) tempfile$(mktemp -t reposync.XXXX)check() {echo "目标目录剩余空间: $(df -h ${local_path} | awk …...
常用的正则表达式,收藏必备!!!
正则表达式是一种强大的文本模式匹配工具,用于在字符串中查找、替换和验证特定模式的文本。下面是一些常用的正则表达式示例: 匹配Email地址: ^[a-zA-Z0-9._%-][a-zA-Z0-9.-]\.[a-zA-Z]{2,}$匹配URL: ^(https?|ftp)://[^\s/$.?#…...
js---webAPI
01 声明变量 js组成: DOM:操作网页内容的,开发页面内容特效和实现用户交互 BOM: DOM树:将 HTML 文档以树状结构直观的表现出来,我们称之为文档树或 DOM 树 文档树直观的体现了标签与标签之间的关系 CSS获取元素的方法 document.querySele…...
git的常用命令有哪些?
Git 是一个流行的分布式版本控制系统,用于跟踪文件的变化、协作开发和管理代码。以下是一些常用的 Git 命令: 创建和克隆仓库: git init:在当前目录初始化一个新的 Git 仓库。git clone <仓库URL>:克隆一个远程仓…...
《动手学深度学习(PyTorch版)》笔记8.5
注:书中对代码的讲解并不详细,本文对很多细节做了详细注释。另外,书上的源代码是在Jupyter Notebook上运行的,较为分散,本文将代码集中起来,并加以完善,全部用vscode在python 3.9.18下测试通过&…...
【蓝桥杯单片机入门记录】LED灯(附多个例程)
目录 一、LED灯概述 1.1 LED发光原理 1.2电路原理图 1.3电路实物图 1.4 开发板LED灯原理图 1.4.1共阳极LED灯操控原理(本开发板) (非实际原理图,便于理解版本)由图可以看出,每个LED灯的左边…...
c语言简单json库
文章目录 写在前面头文件源代码使用示例 写在前面 用c语言实现的一个简单json库,极其轻量 仅1个四百多行源码的源文件,和1个头文件 支持对象、数组、数值、字符串类型 github仓库 头文件 对主要的json API的声明 #ifndef ARCOJSON_ARCOJSON_H #defin…...
Linux操作系统基础(七):Linux常见命令(二)
文章目录 Linux常见命令(二) 一、kill命令 二、ifconfig命令 三、clear命令 四、重启与关机命令 五、which命令 六、hostname命令 七、grep命令 八、|管道 九、useradd命令 十、userdel命令 十一、tar命令 十二、su命令 十三、ps命令 Linu…...
进程状态
广义概念: 从广义上来讲,进程分为新建、运行、阻塞、挂起、退出五个状态,其中新建和退出两个状态可以直接理解字面意思。 运行状态: 这里涉及到运行队列的概念,CPU在读取数据的时候,需要把内存中的进程放入…...
STM32固件库简介与使用指南
1. STM32官方标准固件库简介 STM32官方标准固件库是由STMicroelectronics(ST)提供的一套软件开发工具,旨在简化STM32微控制器的软件开发过程。该固件库提供了丰富的功能和模块,涵盖了STM32微控制器的各种外设,包括但不…...
【开源】SpringBoot框架开发智能教学资源库系统
目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 数据中心模块2.2 课程档案模块2.3 课程资源模块2.4 课程作业模块2.5 课程评价模块 三、系统设计3.1 用例设计3.2 数据库设计3.2.1 课程档案表3.2.2 课程资源表3.2.3 课程作业表3.2.4 课程评价表 四、系统展示五、核心代…...
融资项目——获取树形结构的数据
如下图所示,下列数据是一个树形结构数据,行业中包含若干子节点。表的设计如下图,设置了一个id为1的虚拟根节点。(本树形结构带虚拟根节点共三层) 实现逻辑: 延时展示方法,先展现第二层的信息&a…...
Crypto-RSA2
题目:(BUUCTF在线评测 (buuoj.cn)) 已知e,n,dp/(dq),c求明文: 首先有如下公式: dp ≡ d mod (p-1) ,ed ≡ 1 mod φ(n) ,npq ,φ(n)(p-1)(q-1) python代码实现如下: import libnu…...
IEEE Internet of Things Journal投稿经验
期刊名:IEEE Internet of Things Journal 期刊分区:中科院一区 Top 影响因子:10.6 投稿状态 (1)2023.11.3,投稿成功,状态为:under review(大u大r)࿱…...
实例分割论文阅读之:FCN:《Fully Convolutional Networks for Semantica Segmentation》
论文地址:https://openaccess.thecvf.com/content_cvpr_2015/papers/Long_Fully_Convolutional_Networks_2015_CVPR_paper.pdf 代码链接:https://github.com/pytorch/vision 摘要 卷积网络是强大的视觉模型,可以产生特征层次结构。我们证明,…...
apk反编译修改教程系列---简单去除apk登陆 修改vip与一些反编译基础常识【十二】
往期教程: 安卓玩机-----反编译apk 修改apk 去广告 去弹窗等操作中的一些常识apk反编译修改教程系列-----修改apk应用名称 任意修改名称 签名【一】 apk反编译修改教程系列-----任意修改apk版本号 版本名 防止自动更新【二】 apk反编译修改教程系列-----修改apk中…...
网络安全习题集
第一章 绪论 4 ISO / OSI 安全体系结构中的对象认证安全服务使用( C ) 机制来完成。 A .访问控制 B .加密 C .数字签名 D .数据完整性 5 身份鉴别是安全服务中的重要一环,以下关于身份鉴别的叙述不正确的是…...
C++中的volatile:穿越编译器的屏障
C中的volatile:穿越编译器的屏障 在C编程中,我们经常会遇到需要与硬件交互或多线程环境下访问共享数据的情况。为了确保程序的正确性和可预测性,C提供了关键字volatile来修饰变量。本文将深入解析C中的volatile关键字,介绍其作用、…...
(07)Hive——窗口函数详解
一、 窗口函数知识点 1.1 窗户函数的定义 窗口函数可以拆分为【窗口函数】。窗口函数官网指路: LanguageManual WindowingAndAnalytics - Apache Hive - Apache Software Foundationhttps://cwiki.apache.org/confluence/display/Hive/LanguageManual%20Windowing…...
观成科技:隐蔽隧道工具Ligolo-ng加密流量分析
1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具,该工具基于TUN接口实现其功能,利用反向TCP/TLS连接建立一条隐蔽的通信信道,支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式,适应复杂网…...
Java 语言特性(面试系列2)
一、SQL 基础 1. 复杂查询 (1)连接查询(JOIN) 内连接(INNER JOIN):返回两表匹配的记录。 SELECT e.name, d.dept_name FROM employees e INNER JOIN departments d ON e.dept_id d.dept_id; 左…...
【Oracle APEX开发小技巧12】
有如下需求: 有一个问题反馈页面,要实现在apex页面展示能直观看到反馈时间超过7天未处理的数据,方便管理员及时处理反馈。 我的方法:直接将逻辑写在SQL中,这样可以直接在页面展示 完整代码: SELECTSF.FE…...
【python异步多线程】异步多线程爬虫代码示例
claude生成的python多线程、异步代码示例,模拟20个网页的爬取,每个网页假设要0.5-2秒完成。 代码 Python多线程爬虫教程 核心概念 多线程:允许程序同时执行多个任务,提高IO密集型任务(如网络请求)的效率…...
MySQL用户和授权
开放MySQL白名单 可以通过iptables-save命令确认对应客户端ip是否可以访问MySQL服务: test: # iptables-save | grep 3306 -A mp_srv_whitelist -s 172.16.14.102/32 -p tcp -m tcp --dport 3306 -j ACCEPT -A mp_srv_whitelist -s 172.16.4.16/32 -p tcp -m tcp -…...
站群服务器的应用场景都有哪些?
站群服务器主要是为了多个网站的托管和管理所设计的,可以通过集中管理和高效资源的分配,来支持多个独立的网站同时运行,让每一个网站都可以分配到独立的IP地址,避免出现IP关联的风险,用户还可以通过控制面板进行管理功…...
【MATLAB代码】基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),附源代码|订阅专栏后可直接查看
文章所述的代码实现了基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),针对传感器观测数据中存在的脉冲型异常噪声问题,通过非线性加权机制提升滤波器的抗干扰能力。代码通过对比传统KF与MCC-KF在含异常值场景下的表现,验证了后者在状态估计鲁棒性方面的显著优…...
HybridVLA——让单一LLM同时具备扩散和自回归动作预测能力:训练时既扩散也回归,但推理时则扩散
前言 如上一篇文章《dexcap升级版之DexWild》中的前言部分所说,在叠衣服的过程中,我会带着团队对比各种模型、方法、策略,毕竟针对各个场景始终寻找更优的解决方案,是我个人和我司「七月在线」的职责之一 且个人认为,…...
深度剖析 DeepSeek 开源模型部署与应用:策略、权衡与未来走向
在人工智能技术呈指数级发展的当下,大模型已然成为推动各行业变革的核心驱动力。DeepSeek 开源模型以其卓越的性能和灵活的开源特性,吸引了众多企业与开发者的目光。如何高效且合理地部署与运用 DeepSeek 模型,成为释放其巨大潜力的关键所在&…...
微服务通信安全:深入解析mTLS的原理与实践
🔥「炎码工坊」技术弹药已装填! 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 一、引言:微服务时代的通信安全挑战 随着云原生和微服务架构的普及,服务间的通信安全成为系统设计的核心议题。传统的单体架构中&…...
