当前位置: 首页 > article >正文

Rust 变量与可变性

文章目录

  • 变量与可变性
    • 常量
    • 遮蔽(Shadowing)

变量与可变性

Rust 变量与可变性

Rust中变量默认是不可变的,这是 Rust 鼓励你编写更安全、易于并发代码的众多方式之一。不过,你仍然可以选择让变量可变。让我们来探讨 Rust 为什么鼓励你优先使用不可变性,以及为什么有时你可能需要选择可变性。

当变量是不可变的时,一旦一个值被绑定到名字上,你就无法更改该值。为说明这一点,请在你的项目目录下用 cargo new variables 创建一个新项目。

然后,在新建的 variables 目录下,打开 src/main.rs,并将其代码替换为以下内容(此代码暂时无法编译):

文件名:src/main.rs

此代码编译不通过!

fn main() {let x = 5;println!("The value of x is: {x}");x = 6;println!("The value of x is: {x}");
}

保存并使用 cargo run 运行程序。你会收到关于不可变性错误的提示,如下所示:

$ cargo runCompiling variables v0.1.0 (file:///projects/variables)
error[E0384]: cannot assign twice to immutable variable `x`--> src/main.rs:4:5|
2 |     let x = 5;|         - first assignment to `x`
3 |     println!("The value of x is: {x}");
4 |     x = 6;|     ^^^^^ cannot assign twice to immutable variable|
help: consider making this binding mutable|
2 |     let mut x = 5;|         +++

有关此错误的更多信息,请尝试 rustc --explain E0384
error: could not compile variables (bin “variables”) due to 1 previous error

这个例子展示了编译器如何帮助你发现程序中的错误。

你收到“不能对不可变变量 x 进行二次赋值”的错误,是因为你试图给不可变变量 x 赋第二个值。

当我们试图更改被指定为不可变的值时,在编译时报错是很有价值的,因为不应改变值的变量发生了改变很容易导致 bug。如果代码的一部分要求某个值永远不变,而另一部分代码却改变了这个值,那么第一部分代码可能就无法按预期工作。尤其是当第二部分代码只在某些情况下才改变值时,这种 bug 很难追踪。Rust 编译器保证你声明值不会改变时,它真的不会改变,因此你无需自己跟踪。这样你的代码更易于理解。

但可变性有时非常有用,也能让代码更方便编写。虽然变量默认是不可变的,但我们可以在变量名前加上 mut 使其可变。加上 mut 也向未来的代码阅读者传达了意图,表明代码的其他部分会更改该变量的值。

例如,让我们将 src/main.rs 改为如下内容:

文件名:src/main.rs

fn main() {let mut x = 5;println!("The value of x is: {x}");x = 6;println!("The value of x is: {x}");
}

现在运行程序,输出如下:

$ cargo runCompiling variables v0.1.0 (file:///projects/variables)Finished `dev` profile [unoptimized + debuginfo] target(s) in 0.30sRunning `target/debug/variables`
The value of x is: 5
The value of x is: 6

使用 mut 后,我们可以将 x 的值从 5 改为 6。最终,是否使用可变性取决于我们实际的需求和应用场景。

常量

与不可变变量类似,常量也是绑定到名字上的值,且不允许更改,但常量和变量之间有一些区别。

首先,常量不能使用 mut。常量不仅默认不可变——它们始终不可变。我们需要用 const 关键字声明常量,而不是 let,并且必须标注值的类型。

常量可以在任何作用域声明,包括全局作用域,从而可供多个模块或代码片使用。

最后一个区别是,常量只能被赋值为常量表达式,而非在运行时计算的值。

下面是一个常量声明的例子:

const THREE_HOURS_IN_SECONDS: u32 = 60 * 60 * 3;

常量名为 THREE_HOURS_IN_SECONDS,其值为 60(每分钟的秒数)乘以 60(每小时的分钟数)再乘以 3(我们想要计算的小时数)。Rust 的常量命名约定是全大写并用下划线分隔单词。编译器能在编译时完成有限的运算,这让我们可以用更易理解的方式写出这个值,而不是直接写 10,800。更多关于常量声明时可用操作的信息,请参阅 Rust Reference 的常量求值部分。

在声明的作用域内,常量于整个程序运行期间都有效。此属性使得常量可以作为多处代码使用的全局范围的值,例如一个游戏中所有玩家可以获取的最高分或者光速。

将程序中用到的硬编码值声明为常量,能帮助后来的代码维护人员了解值的意图。如果将来需要更改硬编码值,也只需改动一处即可。

遮蔽(Shadowing)

正如你在第 2 章的猜数字游戏教程中看到的,你可以用相同的名字声明新变量。Rustacean 称第一个变量被第二个变量“遮蔽”,即编译器在使用变量名时会看到第二个变量。实际上,第二个变量覆盖了第一个变量,直到它自己被遮蔽或作用域结束。我们可以通过重复使用 let 关键字和相同变量名来遮蔽变量,如下所示:

文件名:src/main.rs

fn main() {let x = 5;let x = x + 1;{let x = x * 2;println!("The value of x in the inner scope is: {x}");}println!("The value of x is: {x}");
}

该程序首先将 x 绑定为 5。然后通过 let x = x + 1; 创建了一个新变量 x,此时 x 的值为 6。接着,在用花括号创建的内部作用域中,第三个 let 语句再次遮蔽 x,创建了一个新变量,其值为前一个值乘以 2,即 12。当该作用域结束后,内部遮蔽结束,x 恢复为 6。运行该程序,输出如下:

$ cargo runCompiling variables v0.1.0 (file:///projects/variables)Finished `dev` profile [unoptimized + debuginfo] target(s) in 0.31sRunning `target/debug/variables`
The value of x in the inner scope is: 12
The value of x is: 6

遮蔽与将变量标记为 mut 不同,因为如果你试图在没有使用 let 关键字的情况下重新赋值,会得到编译时错误。通过使用 let,我们可以对值进行多次转换,但在这些转换完成后变量仍然是不可变的。

mut遮蔽的另一个区别是,使用 let 关键字实际上创建了一个新变量,因此我们可以更改值的类型,但仍然复用相同的名字。例如,假设我们的程序让用户输入文本之间要有多少个空格,用户输入空格字符后,我们想把这个输入存储为数字:

let spaces = "   ";
let spaces = spaces.len();

第一个 spaces 变量是字符串类型,第二个 spaces 变量是数字类型。遮蔽让我们无需起不同的名字(如 spaces_str 和 spaces_num),而可以复用更简单的 spaces 名字。但如果我们尝试用 mut,如下所示,会得到编译时错误:

此代码无法编译!

let mut spaces = "   ";
spaces = spaces.len();

错误提示我们不能更改变量的类型:

$ cargo runCompiling variables v0.1.0 (file:///projects/variables)
error[E0308]: mismatched types--> src/main.rs:3:14|
2 |     let mut spaces = "   ";|                      ----- expected due to this value
3 |     spaces = spaces.len();|              ^^^^^^^^^^^^ expected `&str`, found `usize`

有关此错误的更多信息,请尝试 rustc --explain E0308
error: could not compile variables (bin “variables”) due to 1 previous error

整体代码示例如下:

fn main() {// 变量可变性  let mut x = 5;println!("The value of x is: {x}");x = 6;println!("The value of x is: {x}");// 常量const MAX_POINTS: u32 = 100_000;println!("The maximum points is: {MAX_POINTS}");// 变量遮蔽let x = 5;let x = x + 1;{let x = x * 2;println!("The value of x in the inner scope is: {x}");}println!("The value of x in the outer scope is: {x}");
}

运行结果如下

The value of x is: 5
The value of x is: 6
The maximum points is: 100000
The value of x in the inner scope is: 12
The value of x in the outer scope is: 6

现在我们已经了解了变量的工作方式,接下来让我们看看它们可以拥有的数据类型。

相关文章:

Rust 变量与可变性

文章目录 变量与可变性常量遮蔽(Shadowing) 变量与可变性 Rust中变量默认是不可变的,这是 Rust 鼓励你编写更安全、易于并发代码的众多方式之一。不过,你仍然可以选择让变量可变。让我们来探讨 Rust 为什么鼓励你优先使用不可变性…...

深入理解 C++ 中的 list 容器:从基础使用到模拟实现

一、list 的底层数据结构与核心特性 1.1 双向循环链表的物理结构 节点定义&#xff1a;每个节点包含三个部分 template <typename T> struct ListNode {T data; // 存储的数据ListNode* prev; // 指向前驱节点的指针ListNode* next; // 指向后继节点的指针L…...

状态机实现文件单词统计

系统如何查找可执行文件 默认&#xff1a;在PATH路径下寻找文件文件下 执行当前目录下文件&#xff1a; ./&#xff1a;指定文件目录是当前目录 ./count:执行当前目录文件 编译.c文件为运行文件 gcc -o count 0voice.c #将0voice.c编译为名字count 为什么主函数要那么写&a…...

从0开始学习R语言--Day13--混合效应与生存分析

混合效应模型&#xff08;Mixed Effects Model&#xff09; 对于数据来说&#xff0c;我们通常把所有样本共有的影响因素&#xff08;性别&#xff0c;实验处理&#xff0c;实验方法&#xff09;&#xff0c;这种可以推广到总体的叫做固有效应&#xff0c;而仅适用于特定分组的…...

基于mediapipe深度学习的虚拟画板系统python源码

目录 1.前言 2.算法运行效果图预览 3.算法运行软件版本 4.部分核心程序 5.算法仿真参数 6.算法理论概述 7.参考文献 8.算法完整程序工程 1.前言 虚拟画板系统基于计算机视觉与深度学习技术&#xff0c;通过摄像头捕获用户手部动作&#xff0c;利用 MediaPipe框架实现手…...

复变函数 $w = z^2$ 的映射图像演示

复变函数 w z 2 w z^2 wz2 的映射图像演示 复变函数 w z 2 w z^2 wz2 是一个基本的二次函数&#xff0c;在复平面上具有有趣的映射性质。下面我将介绍这个函数的映射特性&#xff0c;并使用MATLAB进行可视化演示。 映射特性 极坐标表示&#xff1a;若 z r e i θ z …...

Python实现P-PSO优化算法优化循环神经网络LSTM回归模型项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档&#xff09;&#xff0c;如需数据代码文档可以直接到文章最后关注获取。 1.项目背景 在当今数据驱动的时代&#xff0c;时间序列预测和回归分析是许多领域中不可或缺的技术手段。循环神经网络&#xff…...

复合机器人:纠偏算法如何重塑工业精度与效率?

在智能制造领域&#xff0c;复合机器人正成为柔性生产与高精度作业的核心载体。面对“复合机器人有纠偏算法吗&#xff1f;”这一行业高频问题&#xff0c;富唯智能以多模态纠偏技术体系给出了答案——通过自研的AI驱动纠偏算法与多传感器融合方案&#xff0c;复合机器人不仅能…...

审计- 1- 审计概述

1.财务报表审计的概念 财务报表审计是指注册会计师对财务报表是否不存在重大错报提供合理保证&#xff0c;以积极方式提出意见&#xff0c;增强除管理层之外的预期使用者对财务报表信赖的程度。 1.1 审计业务三方关系人 注册会计师对财务报表发表审计意见是注册会计师的责任管…...

在MDK中自动部署LVGL,在stm32f407ZGT6移植LVGL-8.4,运行demo,显示label

在MDK中自动部署LVGL&#xff0c;在stm32f407ZGT6移植LVGL-8.4 一、硬件平台二、实现功能三、移植步骤1、下载LVGL-8.42、MDK中安装LVGL-8.43、配置RTE4、配置头文件 lv_conf_cmsis.h5、配置lv_port_disp_template 四、添加心跳相关文件1、在STM32CubeMX中配置TIM7的参数2、使能…...

模块二:C++核心能力进阶(5篇) 篇一:《STL源码剖析:vector扩容策略与迭代器失效》

一、前言&#xff1a;重新认识vector的复杂性 在C开发者中&#xff0c;std::vector常被视为"动态数组"的简单实现&#xff0c;但其底层机制实则蕴含着深刻的工程智慧。本篇将通过&#xff1a; 多维度源码剖析&#xff08;GCC/Clang/MSVC三平台实现对比&#xff09;…...

计算机组成原理核心剖析:CPU、存储、I/O 与总线系统全解

引言 在当今数字化时代&#xff0c;计算机已经渗透到我们生活的方方面面&#xff0c;从智能手机到超级计算机&#xff0c;从智能家居到自动驾驶汽车。然而&#xff0c;你是否曾好奇过&#xff0c;这些功能强大的设备内部究竟是如何工作的&#xff1f;是什么让计算机能够执行各种…...

数据分类分级的实践与反思:源自数据分析、治理与安全交叉视角的洞察

在数据安全体系的构建中&#xff0c;数据分类分级无疑扮演着基石性的角色&#xff0c;其系统性与复杂性对从业者提出了极高要求。尽管个人并非专注于该领域的研究专家&#xff0c;但凭借在数据分析、数据治理及数据安全策略应用交叉地带的长期实践与系统性认知&#xff0c;得以…...

自动化立体仓库WCS的设计与实现

导语 大家好&#xff0c;我是社长&#xff0c;老K。专注分享智能制造和智能仓储物流等内容。欢迎大家使用我们的仓储物流技术AI智能体。 新书《智能物流系统构成与技术实践》 新书《智能仓储项目出海-英语手册&#xff0c;必备&#xff01;》 完整版文件和更多学习资料&#xf…...

百度蜘蛛池的作用是什么?技术@baidutopseo

百度蜘蛛池的作用 百度蜘蛛池是一种用于集中管理和调度百度搜索引擎蜘蛛&#xff08;爬虫&#xff09;的技术工具。其主要目的是优化网站内容的收录效率&#xff0c;帮助搜索引擎更高效地发现和抓取网页内容。技术baidutopseo 提升收录效率 通过集中大量百度蜘蛛的访问请求&…...

8.linux文件与文件夹内处理命令cp,mv,rm

cp 可以用于复制文件/文件夹 cp 【-r】 参数1 参数2 -r&#xff0c;用于复制文件夹使用&#xff0c;表示递归 mv 用于移动文件/文件夹&#xff0c;没有选项 参数2&#xff1a;表示移动去的地方&#xff0c;如果参数2不存在&#xff0c;则进行改名&#xff0c;保证目标存在…...

JavaScript性能优化:实战技巧提升10倍速度

JavaScript 性能优化实战技术文章大纲 基础优化策略 减少 DOM 操作&#xff1a;频繁的 DOM 操作会导致重绘和回流&#xff0c;影响性能。使用文档片段&#xff08;DocumentFragment&#xff09;或虚拟 DOM 技术优化批量操作。 避免全局变量污染&#xff1a;全局变量会增加内…...

核函数:解锁支持向量机的强大能力

在机器学习的世界中&#xff0c;支持向量机&#xff08;SVM&#xff09;是一种强大的分类算法&#xff0c;而核函数则是其背后的“魔法”&#xff0c;让 SVM 能够处理复杂的非线性问题。今天&#xff0c;我们就来深入探讨核函数的奥秘&#xff0c;看看它们是如何帮助 SVM 在高维…...

UE5 2D地图曝光太亮怎么修改

UE5 2D地图曝光怎么修改 在场景添加后期处理体积 修改后期处理体积Exposure曝光参数最大值最小值都改为0 勾选Infinite Extend 全地图范围应用此后期处理体积...

C# 类和继承(基类访问)

基类访问 如果派生类必须访问被隐藏的继承成员&#xff0c;可以使用基类访问&#xff08;base access&#xff09;表达式。基类 访问表达式由关键字base后面跟着一个点和成员的名称组成&#xff0c;如下所示&#xff1a; 例如&#xff0c;在下面的代码中&#xff0c;派生类Oth…...

帕金森带来的生活困境

当这种健康状况出现&#xff0c;行动不再自如成为最明显的改变。日常行走时&#xff0c;步伐会逐渐变小、变慢&#xff0c;甚至会出现 “小碎步” 往前冲&#xff0c;难以停下&#xff0c;简单的起身、转身都可能变得艰难。手部也会不受控制地颤抖&#xff0c;拿水杯、系纽扣这…...

集成测试的流程总结

首先我们的目的是进行自动化测试&#xff0c;也就是通过cl工具来对我们的项目用我们自己写的yaml文件中的命令来测试项目&#xff0c;这是我们的根本性目的&#xff0c;现在用github action cl工具以及maestro cli 云端作为例子通一遍流程。 首先用xcode创建我们的ios app应用程…...

Redis最佳实践——性能优化技巧之Pipeline 批量操作

Redis Pipeline批量操作在电商应用中的性能优化技巧 一、Pipeline核心原理与性能优势 1. 工作机制对比&#xff1a; sequenceDiagramtitle 常规请求 vs Pipeline请求# 常规模式Client->>Redis: 命令1Redis-->>Client: 响应1Client->>Redis: 命令2Redis--&g…...

Node.js 项目调试指南

Node.js 项目调试指南 &#x1f9ed; 一、调试工具和方式总览 方式难度场景说明console.log 调试★简单问题定位最常见&#xff0c;但效率低debug 模块★★模块化输出日志支持命名空间的调试日志VSCode 断点调试★★★跟踪函数调用、变量状态推荐使用node inspect / ndb★★★…...

win32相关(虚拟内存和物理内存)

虚拟内存和物理内存 在win32操作系统下&#xff0c;每个进程都有它自己独立的4GB空间&#xff0c;是window给它分配的一个虚拟空间&#xff0c;并不是真正的物理空间&#xff0c;这4GB空间中&#xff0c;分为高2G和低2G&#xff0c;高2G是应用程序的&#xff0c;低2G空间是给内…...

Linux操作系统安全管理概述与命令操作

前言&#xff1a; 1.本文将详细描述让读者了解Linux操作系统安全管理的概述和SELinux安全上下文以及基础操作命令&#xff1b; 2.本文将让读者掌握Linux操作系统防火墙firewall的结构和命令使用方法&#xff1b; 3.了解Iptables防火墙配置的结构与特点以及…...

《操作系统真相还原》——中断

可以毫不夸张的说&#xff0c;操作系统离不开中断 此时我们将中断处理程序放在了汇编文件中了&#xff0c;很显然我们不能很方便的编写中断处理程序&#xff0c;不如在汇编程序里调用c函数。 在这个感觉过可以在c语言中直接内联汇编完成这些。 定时器 将时钟中断的频率提高后…...

[yolov11改进系列]基于yolov11引入特征融合注意网络FFA-Net的python源码+训练源码

【FFA-Net介绍】 北大和北航联合提出的FFA-net: Feature Fusion Attention Network for Single Image Dehazing图像增强去雾网络&#xff0c;该网络的主要思想是利用特征融合注意力网络&#xff08;Feature Fusion Attention Network&#xff09;直接恢复无雾图像&#xff0c;…...

助力活力生活的饮食营养指南

日常生活中&#xff0c;想要维持良好的身体状态&#xff0c;合理的营养补充至关重要。对于易受身体变化困扰的人群来说&#xff0c;更需要从饮食中摄取充足养分。​ 蛋白质是身体的重要 “建筑材料”&#xff0c;鱼肉、鸡肉、豆类制品富含优质蛋白&#xff0c;易于消化吸收&am…...

【软件测试】测试框架(unittest/pytest)

本文介绍了Python 中最常用的两个测试框架&#xff1a;unittest 和 pytest&#xff0c;帮助你编写更规范、可维护的自动化测试用例。 一、unittest 框架 unittest 是 Python 内置的标准库&#xff0c;无需额外安装&#xff0c;适合初学者入门。它借鉴了 JUnit 的设计理念&…...