Rust语言入门第四篇-变量与可变性以及隐藏(Shadowing)

2024-04-13 10:36

本文主要是介绍Rust语言入门第四篇-变量与可变性以及隐藏(Shadowing),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • Rust语言入门第四篇-变量与可变性以及隐藏(Shadowing)
    • 概要
    • let 关键字
    • 自动判断变量类型
    • 隐藏(Shadowing)
    • let关键字支持的数据类型
    • let 关键字声明的变量类型转换

Rust语言入门第四篇-变量与可变性以及隐藏(Shadowing)

概要

在 Rust 中,尽管是强类型语言,但编译器拥有类型推断的能力。这意味着你可以在不显式指定变量类型的情况下声明变量,编译器会根据上下文推断出变量的类型。这样的设计在一定程度上使代码更加简洁,但不会牺牲类型安全性。

let 关键字

在 Rust 中,尽管是强类型语言,但编译器拥有类型推断的能力。这意味着你可以在不显式指定变量类型的情况下声明变量,编译器会根据上下文推断出变量的类型。这样的设计在一定程度上使代码更加简洁,但不会牺牲类型安全性。

在 Rust 中,使用 let 关键字声明的变量默认是不可变的(immutable)。如果需要可变的变量,则需要使用 let mut 来声明。例如:

// 不可变变量 x,类型为 i32,编译器会根据赋值推断出类型为 i32
let x = 5;// 可变变量 y,类型为 i32
let mut y = 10;

在这里插入图片描述

强行执行

error[E0384]: cannot assign twice to immutable variable `x`--> src\main.rs:4:5|
3 |     let x = 5;|         -|         ||         first assignment to `x`|         help: consider making this binding mutable: `mut x`
4 |     x = 9;|     ^^^^^ cannot assign twice to immutable variable

原因是 不能为不可变变量x赋值两次

因此,虽然 Rust 具有类型推断的能力,但仍然需要使用 letlet mut 显式声明变量,并且编译器会根据上下文进行类型推断,确保代码的类型安全性。

自动判断变量类型

Rust 代码展示了如何使用 let 关键字声明变量并进行赋值,而不显式指定变量类型,让编译器根据赋值的内容自动推断变量的类型。下面是对每个变量的解释:

fn main() {let a = "123";   // 变量 a 的类型是 &str(字符串切片)let b = 123;     // 变量 b 的类型是 i32(32 位有符号整数)let c = 123.1;   // 变量 c 的类型是 f64(64 位浮点数)let d = true;    // 变量 d 的类型是 bool(布尔值)let e = '1';     // 变量 e 的类型是 char(Unicode 字符)println!("{}", { a });  // 打印变量 a 的值println!("{}", { b });  // 打印变量 b 的值println!("{}", { c });  // 打印变量 c 的值println!("{}", { d });  // 打印变量 d 的值println!("{}", { e });  // 打印变量 e 的值
}

在这个示例中,变量的类型是根据赋值的内容自动推断出来的。Rust 的类型推断机制使得代码更加简洁,同时保持了类型安全性,但不会牺牲类型安全性。
在这里插入图片描述

隐藏(Shadowing)

在 Rust 中,隐藏(Shadowing)是指在同一作用域内,使用一个新的变量名来覆盖已经存在的同名变量。这与变量的重新赋值不同,因为隐藏创建了一个新的变量,而不是修改原有变量的值。隐藏允许你在同一作用域内使用相同的变量名来引入新值,而无需创建新的作用域。

下面是一个示例说明 Rust 中隐藏的概念:

fn main() {let x = 5; // 声明并初始化变量 x// 使用隐藏创建一个新的变量 ylet y = 10;println!("Before shadowing: x = {}, y = {}", x, y);{let y = x+y;println!("Before shadowing: x = {}, y = {}", x, y);}let x = "hello"; // 隐藏变量 x,并赋予一个新值let y = false;   // 隐藏变量 y,并赋予一个新值println!("After shadowing: x = {}, y = {}", x, y);
}

在 Rust 中,使用花括号 {} 创建一个新的作用域(scope)。在你的代码中,{} 中的代码块就是一个新的作用域。在这个作用域内声明的变量 y 与外部作用域中的变量 y 是不同的变量,因为它们位于不同的作用域内。

在这个示例中,我们先声明并初始化了两个变量 xy,然后使用隐藏分别创建了新的变量 xy,并赋予了新的值。在打印变量值时,会分别输出隐藏后的 xy 的值。

隐藏的好处在于可以使用相同的变量名来引入新的值,而不必担心与之前的变量产生冲突。此外,隐藏也能够增强代码的可读性和灵活性。

let关键字支持的数据类型

在 Rust 中,let 变量声明可以支持多种数据类型。下面是一些示例代码,展示了 let 变量声明不同类型的用法:

  1. 整数类型示例
// Signed integers
let x: i8 = 10;
let y: i16 = 100;
let z: i32 = 1000;
let w: i64 = 10000;// Unsigned integers
let a: u8 = 10;
let b: u16 = 100;
let c: u32 = 1000;
let d: u64 = 10000;
  1. 浮点数类型示例
let x: f32 = 3.14;
let y: f64 = 3.14159265359;
  1. 布尔类型示例
let x: bool = true;
let y: bool = false;
  1. 字符类型示例
let x: char = 'a';
let y: char = '😀';

在这里插入图片描述

  1. 元组类型示例
let x: (i32, f64, char) = (42, 3.14, 'a');
  1. 数组类型示例
let x: [i32; 5] = [1, 2, 3, 4, 5];
  1. 切片类型示例
let x: &[i32] = &[1, 2, 3, 4, 5];
  1. 字符串类型示例
let x: &str = "Hello, world!";
let y: String = String::from("Hello, world!");
  1. 结构体类型示例
struct Point {x: i32,y: i32,
}let p: Point = Point { x: 10, y: 20 };
  1. 枚举类型示例
enum Direction {Up,Down,Left,Right,
}let d: Direction = Direction::Up;

这些示例展示了 let 变量声明不同类型的用法,你可以根据具体的需求选择合适的类型来声明变量。

let 关键字声明的变量类型转换

在 Rust 中,let 关键字用于声明变量。通过 let 关键字声明的变量可以转换为多种类型,具体取决于变量的使用和上下文。下面是一些常见的类型转换:

  1. 不可变绑定(Immutable Binding)转换为可变绑定(Mutable Binding):通过使用 mut 关键字,可以将不可变绑定转换为可变绑定。例如:
let x = 5; // 不可变绑定
let mut y = x; // 转换为可变绑定
  1. 从一个类型转换为另一个类型:通过重新赋值或使用类型转换函数,可以将变量从一种类型转换为另一种类型。例如:
let x = 5; // 整数类型
let y = x as f32; // 将整数类型转换为浮点数类型
  1. 引用类型转换为可变引用类型:通过使用 &mut 关键字,可以将不可变引用转换为可变引用。例如:
let mut x = 5; // 可变绑定
let y = &mut x; // 转换为可变引用类型
  1. 通过模式匹配转换:通过模式匹配可以将变量从一种类型转换为另一种类型。例如:
let x = Some(5); // Option<i32> 类型
if let Some(y) = x {// 将 Option<i32> 类型转换为 i32 类型println!("The value of x is: {}", y);
}

总的来说,Rust 中的 let 变量可以在一定程度上灵活转换为多种类型,但转换的方式取决于具体的场景和需求。

这篇关于Rust语言入门第四篇-变量与可变性以及隐藏(Shadowing)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/899898

相关文章

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

变量与命名

引言         在前两个课时中,我们已经了解了 Python 程序的基本结构,学习了如何正确地使用缩进来组织代码,并且知道了注释的重要性。现在我们将进一步深入到 Python 编程的核心——变量与命名。变量是我们存储数据的主要方式,而合理的命名则有助于提高代码的可读性和可维护性。 变量的概念与使用         在 Python 中,变量是一种用来存储数据值的标识符。创建变量很简单,

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验

数论入门整理(updating)

一、gcd lcm 基础中的基础,一般用来处理计算第一步什么的,分数化简之类。 LL gcd(LL a, LL b) { return b ? gcd(b, a % b) : a; } <pre name="code" class="cpp">LL lcm(LL a, LL b){LL c = gcd(a, b);return a / c * b;} 例题:

Java 创建图形用户界面(GUI)入门指南(Swing库 JFrame 类)概述

概述 基本概念 Java Swing 的架构 Java Swing 是一个为 Java 设计的 GUI 工具包,是 JAVA 基础类的一部分,基于 Java AWT 构建,提供了一系列轻量级、可定制的图形用户界面(GUI)组件。 与 AWT 相比,Swing 提供了许多比 AWT 更好的屏幕显示元素,更加灵活和可定制,具有更好的跨平台性能。 组件和容器 Java Swing 提供了许多

【IPV6从入门到起飞】5-1 IPV6+Home Assistant(搭建基本环境)

【IPV6从入门到起飞】5-1 IPV6+Home Assistant #搭建基本环境 1 背景2 docker下载 hass3 创建容器4 浏览器访问 hass5 手机APP远程访问hass6 更多玩法 1 背景 既然电脑可以IPV6入站,手机流量可以访问IPV6网络的服务,为什么不在电脑搭建Home Assistant(hass),来控制你的设备呢?@智能家居 @万物互联

poj 2104 and hdu 2665 划分树模板入门题

题意: 给一个数组n(1e5)个数,给一个范围(fr, to, k),求这个范围中第k大的数。 解析: 划分树入门。 bing神的模板。 坑爹的地方是把-l 看成了-1........ 一直re。 代码: poj 2104: #include <iostream>#include <cstdio>#include <cstdlib>#include <al

MySQL-CRUD入门1

文章目录 认识配置文件client节点mysql节点mysqld节点 数据的添加(Create)添加一行数据添加多行数据两种添加数据的效率对比 数据的查询(Retrieve)全列查询指定列查询查询中带有表达式关于字面量关于as重命名 临时表引入distinct去重order by 排序关于NULL 认识配置文件 在我们的MySQL服务安装好了之后, 会有一个配置文件, 也就

C语言 | Leetcode C语言题解之第393题UTF-8编码验证

题目: 题解: static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num & MASK1) == 0) {return