Rust从入门到精通03-变量

1、变量声明语法

Rust 变量必须先声明,后使用。

对于局部变量,常见是声明语法为:

let variable : i32 = 100;

由于 Rust 是有自动推导类型功能的,所以后面的 :i32 是可以省略的。

1.1 语法解析更容易

局部变量声明一定是以 let 开头,类型一定是跟在冒号 : 的后面。语法歧义更少,语法分析器更容易编写。

1.2 方便引入类型推导功能

Rust 声明变量的特点:要声明的变量前置,类型描述后置。
这是因为在变量声明语句中,最重要的是变量本身,而类型其实是个附属的额外描述,并非必不可少的部分。如果我们可以通过上下文环境由编译器自动分析出这个变量的类型,那么这个类型描述完全可以省略不写。

PS:Rust 支持类型推导,在编译器能够推导类型的情况下,变量类型一般可以省略,但常量(const)和静态变量(static)必须声明类型。

Rust 从一开始就考虑了类型自动推导功能,因此类型后置的语法更加合适。

1.3 模式解构

let 表达式不仅仅用于变量的绑定,还能进行复杂变量的解构:从一个相对复杂的变量中,匹配出该变量的一部分内容:

/* by 01022.hk - online tools website : 01022.hk/zh/reproduce.html */
fn main() {
    let (a, mut b): (bool,bool) = (true, false);
    // a = true,不可变; b = false,可变
    println!("a = {:?}, b = {:?}", a, b);

    b = true;
    assert_eq!(a, b);
}

2、变量命名规则

Rust 里的合法标识符(包括变量名、函数名、trait名等)必须由:
①、数字
②、字母
③、下划线
注意:不能以数字开头!!!

另外:要注意下划线 _ 的特殊用法。 如果你创建了一个变量却不在任何地方使用它,Rust 通常会给你一个警告,为了去掉这个警告,可以用下划线用作变量的开头。

/* by 01022.hk - online tools website : 01022.hk/zh/reproduce.html */
fn main() {
    let _x = 5;
    let y = 10;
}

运行发现只有变量 y 提示警告了。

3、变量遮蔽

Rust 允许在同一个代码块中声明同样名字的变量,后面声明的变量会将前面声明的变量“遮蔽”起来。

//变量遮蔽
fn variable_masking(){
    let x = "123";
    println!("{}",x);

    let x = 1;
    println!("{}",x);
}

注意:这样做并不会产生内存安全问题,因为我们对这块内存拥有完整的所有权,且此时并没有任何其它引用指向这个变量,对这个变量的修改是完全合法的。

4、变量类型推导

Rust的类型推导有两种:
①、从变量声明的当前语句中获取信息进行推导
②、通过上下文信息进行推导

//类型推导
fn type_derivation(){
    //1.1 没有明确标出变量类型,但是通过字面量的后缀,编译器知道x的类型是 u8
    let x = 5u8;
    //1.2 通过字面量值 1,编译器知道y 的类型是 i32
    let y = 1;

    //1.3 创建一个动态数组,但是没有声明数组里面是什么类型
    let mut vec = Vec::new();
    //编译器通过添加的元素 1 ,推导出 vec 的实际类型是 Vec<i32>
    vec.push(1);

    println!("{}",x);
    println!("{}",y);
    println!("{:?}",vec);

}

5、类型别名

通过 type 关键字给同一个类型起个别名。

//类型别名
fn type_alias(){
    //将 i32 这种数据类型起别名为 int
    type int = i32;
    let x : int = 1;
    println!("{}",x);
}

类型别名还可以用于泛型场景:

type Double = (T,Vec);

那么,以后使用 Double 的时候,就等同于(i32,Vec)

6、不可变 mut

Rust 声明的变量默认是不可变的。

默认不可变,这是一个很重要的特性,它符合最小权限原则,有助于我们写出正确且健壮的代码。

// 变量不可变
fn variable_mut(){
    let i = 123;
    i = 2;
}

编译报错:

如果要使得变量可变,必须要用关键字 mut 声明:

fn variable_mut(){
    let mut i = 123;
    i = 2;
}

当你使用 mut 却没有修改变量,Rust 编译期会友好地报警,提示你移除不必要的 mut。

fn main() {
    variable_mut();
}
// 变量不可变
fn variable_mut(){
    let mut i = 123;
    println!("{}",i);
}

编译器警告:

7、静态变量

在 Rust 中,静态变量(Static Variables)是一种全局变量,其生命周期从程序开始直到程序结束,具有静态的存储期。

静态变量与普通的局部变量和堆上的动态分配变量有所不同。它们在编译时被初始化,并在程序的整个执行期间保持不变。

静态变量存储在只读的静态数据段中,可以在整个程序的作用域内共享和访问。

Rust 中通过 static 关键字声明静态变量,如下:

static NAME: Type = value;

比如声明一个静态变量 GLOBAL:

static GLOBAL : i32 = 0;

这也是 Rust 中唯一声明全局变量的方法。

由于 Rust 非常注重内存安全,因此全局变量的使用有很多限制:
①、全局变量必须在声明的时候马上初始化(对应局部变量可以先声明不初始化,只需要保证使用的时候初始化就行了,我们可以这样理解,全局变量是写在函数外面,而局部变量是写在函数内部,所以需要保证全局变量声明的时候就要初始化);
②、全局变量的初始化必须是编译期可确定的常量,不能包括执行期才能确定的表达式、语句和函数调用;
③、带有 mut 修饰的全局变量,在使用的时候必须使用 unsafe 关键字。

8、常量

Rust 中通过 const 关键字声明常量。如下:

const GLOBAL : i32 = 0;

①、使用 const 声明的是常量,而不是变量。因此不允许使用 mut 关键字修饰这个变量绑定,也不允许使用 let 关键字。

②、常量的初始化表达式也一定要是一个编译期确定的常量,不能是运行期的值。

③、常量名通常约定全部字母都使用大写,并使用下划线分隔单词。

④、常量声明一定要显式声明类型,不能省略。否则会编译报错。

const PI = 3.1415;  // ❌ 错误:missing type for `const` item

与静态变量相比,常量有以下几个区别:

  1. 不可变性:常量是不可变的,一旦初始化后就不能修改。而静态变量可以是可变的,但在 Rust 中通常也是不可变的。
  2. 编译时确定:常量的值在编译时就被确定,并且在程序运行时不能改变。静态变量的值在程序运行时也是不可变的,但它们的初始化可以包含运行时计算的值。
  3. 作用域和可见性:常量的作用域可以是全局的,也可以在特定的作用域内定义。它们可以通过使用 pub 关键字来进行公开,以便在其他模块中使用。静态变量具有全局可见性,可以在程序的任何位置访问和使用。
  4. 存储位置:常量通常直接嵌入到使用它们的代码中,并不占用额外的存储空间。而静态变量存储在只读的静态数据段中,可能占用额外的存储空间。

总结来说,常量是在编译时确定的不可变值,其值在运行时不可修改。它们在使用前就被初始化,并始终保持不变。常量具有诸如不可变性、编译时确定、作用域和存储位置等特性。静态变量与常量的区别在于可变性和初始化时机等方面。

9、变量声明常见错误

9.1 变量必须初始化才能使用

类型没有默认构造函数,变量值没有“默认值”

fn main() {
    let x : i32;
    println!("{}",x);
}

9.2 不能通过 mut 关键字修饰 const 声明的变量

这个很容易理解,常量是不可变的,mut 表示可变,放一起语义冲突了。

//mut和 const 不能一起使用
fn const_mut_error(){
    const mut i : i32 = 1;
    println!("{}",i);
}

9.3 使用下划线开头忽略未使用的变量

如果你创建了一个变量却不在任何地方使用它,Rust 通常会给你一个警告,因为这可能会是个 BUG。

但是有时创建一个不会被使用的变量是有用的,比如你刚刚开始创建一个项目。

如果你希望Rust 不要警告未使用的变量,可以用下划线作为变量名的开头:

fn main() {
    let a = 5;
    let _b = 6;
}

上面只有变量 a 警告了,变量 _b 没有警告。

10、为啥要手动设置变量可变性?

对比其它语言,比如 Java,默认变量是可变的,如果添加 final 关键字,表示变量在初始化后不可再被修改。

而 Rust 中的变量默认是不可变的。如果你想修改一个变量的值,需要在声明时使用 mut 关键字显式指定它为可变。

大概对比一下,从内存管理来看,Java是面向对象语言,内存管理依靠垃圾收集器,程序员不需要直接管理内存分配和释放。而Rust 则依赖所有权和生命周期系统提供了无垃圾收集的内存安全保证,编译器通过所有权规则来保证在任何给定时间,数据要么只有一个可变引用,要么有多个不可变引用,所以默认不可变是很有必要的。

另外,默认声明不可变,也会使得代码意图更明显,想要修改变量,必须增加额外的操作。

默认不可变,也可以防止数据被意外修改,rust 编译器也能够对不可变数据进行更多优化,提高程序的执行效率。

从并发上来讲,如果数据不会改变,就不需要加锁来同步访问,因此可以避免锁带来的开销和复杂性。这使得 Rust 在编写高效且安全的并发代码方面具有天然的优势。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值