我们在Rust环境配置和入门指南中详细介绍了
下面,我们就之间直入主题了。
通过创建一个名为 main.rs 的文件并将以下代码放入其中来编写我们的第一个 Rust 代码:
fn main() { println!("Hello, Front789!");}
然后通过运行 rustc main.rs 和 ./main.exe 来运行这个程序,就像运行 C 程序一样。
Cargo 是 Rust 的构建系统和包管理器
我们也可以使用 cargo 创建项目。
在 Rust 中,默认情况下「变量是不可变」的,这意味着一旦给变量赋值,其值就不会改变。
所以如果想要一个可变的,即可改变的值,使用 mut。
let a = 5;let mut b = 5; // 可变的
图片
let number: i32 = 42;
图片
let pi: f64 = 3.14159;
let is_rust_cool: bool = true;
let heart_emoji: char = '❤';
图片
let mut s = String::from("front789");
let s1: &str = "front789";
let array: [i32; 3] = [1, 2, 3];let a = [3; 5]; // 用值 3 初始化大小为 5 的数组
图片
为了从元组中获得单个的值,可以使用「模式匹配」来解构元组
还可以通过「索引」并使用点号(.)来访问元组中的值
let tup = (500, 6.4, 1);let (x, y, z) = tup;let aa = tup.0; // 引用元组中的第一个项目
图片
指针是一个变量,它存储了一个值的「内存地址」
Rust 中最常见的指针是引用。引用以 & 符号为标志并「借用了它们所指向的值」。除了引用数据没有任何其他特殊功能。它们也没有任何额外开销,所以应用得最多。
fn main() { // 标量类型 let number: i32 = 42; let pi: f64 = 3.14159; let is_rust_cool: bool = true; let heart_emoji: char = '❤'; // 复合类型 let array: [i32; 3] = [1, 2, 3]; let tuple: (i32, f64, char) = (10, 3.14, 'a'); let slice: &[i32] = &[1, 2, 3]; let string: String = String::from("Hello, Front789!"); let string_slice: &str = "Hello, Front789!"; // 特殊类型 let reference_to_number: &i32 = &number; let optional_value: Option<i32> = Some(42); let result_value: Result<i32, &str> = Ok(42);}
以上内容就是Rust中所涉及到的各种数据类型,我们可以从以下的链接中找到更为详细的解释
不可变数组在 Rust 中用 [T; N] 语法来声明,其中 T 表示数组元素的类型,而 N 表示数组的长度。
对于不可变数组,我们可以使用下标访问其元素,但不能修改元素的值。
let array = [1, 2, 3, 4, 5];let first_element = array[0]; // 访问第一个元素arr[0] = 6; // 这行代码会导致编译错误,因为数组是不可变的// 迭代// 使用 for 循环for &num in &array { println!("{}", num);}// 另一种迭代器array.iter().for_each(|&num| { println!("{}", num);});let slice = &array[1..3]; // 从索引 1 到索引 2(包括)切片
Vec<T> 是 Rust 中可变长数组的实现,它允许您动态地增加或减少数组的大小。
let mut array = [1, 2, 3, 4, 5];array[0] = 10; // 修改第一个元素let mut vec = Vec::new(); // 创建一个空 Vecvec.push(1); // 向 Vec 中添加一个元素vec.push(2);vec.push(3);// 使用 iter() 遍历元素for item in array.iter() { println!("{}", item);}// iter_mut() 方法返回一个可变的迭代器,允许修改 Vec 中的元素for item in array.iter_mut() { *item += 1; // 对每个元素加 1}// maplet doubled_array: Vec<_> = array.iter() .map(|&num| num * 2) .collect();// filterlet even_elements: Vec<_> = array.iter() .filter(|&&num| num % 2 == 0) .collect();// len() 方法返回 Vec 中元素的数量array.len()// remove() 方法移除指定索引位置的元素,并返回该元素。如果索引越界,它将导致 panic。let removed_item = array.remove(2) // removed_item 为3
let s1 = String::from("Hello, ");let s2 = String::from("Front789!");let combined = s1 + &s2; // 注意:s1 在这里被移动,之后不能再使用println!("{}", combined); // 打印 "Hello, Front789!"let mut s = String::from("Hello, ");s.push_str("Front789!");println!("{}", s); // 打印 "Hello, Front789!"// 获取字符let s = String::from("hello");let first_char = s.chars().nth(0); // 访问第一个字符// 子字符串let s = String::from("hello Front789");let substring = &s[0..5]; // 提取 "hello"// len()let s = String::from("hello");let length = s.len(); // 字符串的长度// replacelet s = String::from("hello");let replaced = s.replace("l", "z"); // 替换 "l" 为 "z"// splitlet s = String::from("hello Front789");let words: Vec<&str> = s.split_whitespace().collect(); // 分割成单词// 转换 &str 和 Stringlet s = String::from("hello");let s_ref: &str = &s; // 将 String 转换为 &strlet s_copy: String = s_ref.into(); // 将 &str 转换为 String
let mut v1 = vec![1, 2, 3]; // 使用 vec![] 宏let mut v2: Vec<i32> = Vec::new(); // 使用 Vec::new() 构造函数let mut v = Vec::new();v.push(1);v.push(2);let first_element = v[0]; // 访问第一个元素// 迭代// 使用 for 循环for num in &v { println!("{}", num);}// 使用迭代器v.iter().for_each(|&num| { println!("{}", num);});// slicelet slice = &v[1..3]; // 从索引 1 到索引 2(包括)提取元素// removelet removed_element = v.remove(1); // 移除索引为 1 的元素(返回被移除的元素)// sort()v.sort();// joinlet tt= vec!["hello", "Front789"];let joined_string = tt.join(", "); // 使用逗号和空格连接元素
Rust代码使用「蛇形命名法」来作为规范函数和变量名称的风格。蛇形命名法「只使用小写的字母进行命名,并以下画线分隔单词」。
fn 函数名(参数1: 类型1, 参数2: 类型2) -> 返回类型 { // 函数体 // 可选的表达式}
最后一行返回值时不需要调用 return。
fn add_numbers(x: i32, y: i32) -> i32 { let sum = x + y; sum // 函数中的最后一个表达式会隐式返回}
如果想要一个无返回值的函数,不要定义返回类型。
我们可以在基础概念_函数部分查看更详细的解释
要读取一个值,使用 io stdin 并给出变量的值,在失败时需要提供 expect 消息,否则会出错。
let mut guess = String::new(); io::stdin().read_line(&mut guess).expect("该行读取失败");
println!("输出对应的变量信息 {}", guess); // 这里的 guess 是变量名。
你也可以在末尾有变量
let y = 10;println!("y + 2 = {}", y + 2);
在Rust中,一个「新的声明变量可以覆盖掉旧的同名变量」,我们把这一个现象描述为:「第一个变量被第二个变量遮蔽Shadow了」。这意味着随后使用这个名称时,它指向的将会是第二个变量。
fn main() { let x = 5; // 定义值为 5 的变量 x println!("原始值 x: {}", x); // 打印 "原始值 x: 5" let x = 10; // Shadowing:定义一个新的值为 10 的变量 x println!("Shadowed x: {}", x); // 打印 "Shadowed x: 10"}
图片
图片
if condition1 { // 如果 condition1 为真,则执行的代码} else if condition2 { // 如果 condition2 为真,则执行的代码} else { // 如果 condition1 和 condition2 都为假,则执行的代码}
Rust提供了3种循环
loop { println!("永无止境的执行");}
let mut count = 0;while count < 5 { println!("Count: {}", count); count += 1;}
for i in 0..5 { println!("{}", i);}
当然也少不了对数值的遍历操作。
(1..=5).for_each(|num| { println!("Number: {}", num);});// Number: 1// Number: 2// Number: 3// Number: 4// Number: 5
..:它表示一个扩展运算符,表示从第一个数字到最后一个数字生成。
我们也可以在循环中使用 continue 和 break。
图片
这个概念是需要特别注意和反复观看的部分。
当变量值被重新分配时,值会给新的所有者,并且旧的所有者被丢弃。
这种行为在字符串中经常看到,而不是其他类型,如下所示:
let s1 = String::from("hello");let s2 = s1;println!("{}, world!", s1);
这将导致错误,因为 s1 在 s2=s1 之后不再有效。
如何解决上面的问题呢,我们可以使用 Clone:
let s1 = String::from("hello");let s2 = s1.clone();println!("s1 = {}, s2 = {}", s1, s2);
某些类型隐式实现了 Clone。
let x = 5; // x 拥有整数 5let y = x; // 将 x 的值复制到 y,不传递所有权
例如,整数隐式实现了 Clone,因此这段代码不会报错。
图片
fn main() { let s = String::from("hello"); // s 进入作用域 takes_ownership(s); // s 的值移动进入函数... // ... 所以这里不再有效 let x = 5; // x 进入作用域 makes_copy(x); // x 会移入函数, // 但 i32 是 Copy,所以在之后继续使用 x 是可以的} // 在这里,x 超出作用域,然后是 s。但因为 s 的值被移动了,所以没有什么特别的发生。fn takes_ownership(some_string: String) { // some_string 进入作用域 println!("{}", some_string);} // 在这里,some_string 超出作用域,调用 drop。内存被释放。fn makes_copy(some_integer: i32) { // some_integer 进入作用域 println!("{}", some_integer);} // 在这里,some_integer 超出作用域。没有什么特别的发生。
如果我们像在变量被移动后,继续使用,那么我们就使用 takes_ownership(s.clone()); (或者)在 takes_ownership 函数中返回值,像这样:
fn main() { let s2 = String::from("hello"); // s2 进入作用域 let s3 = takes_and_gives_back(s2); // s2 移入并被返回}fn takes_and_gives_back(a_string: String) -> String { a_string // 返回并移出到调用函数}
传递变量的引用,所有权不会被传递。
我们称「创建引用的操作为借用」。就像现实生活中,如果一个人拥有一样东西,你可以从他们那里借来。借了之后,你必须归还。你不拥有它。
fn main() { let s1 = String::from("hello"); let len = calculate_length(&s1); println!("The length of '{}' is {}.", s1, len);}fn calculate_length(s: &String) -> usize { s.len()}
针对此处更详细的内容,可以翻看我们之前的所有权
struct,或者 structure,是一个「自定义数据类型」,允许我们命名和包装多个相关的值,从而形成一个有意义的组合。
图片
struct User { active: bool, username: String, email: String, sign_in_count: u64,}fn main() { let mut user1 = User { active: true, username: String::from("front789"), email: String::from("front789@example.com"), sign_in_count: 1, }; user1.email = String::from("anotheremail@example.com"); let user2 = User { email: String::from("another@example.com"), ..user1 };}
在 user2 中,你会看到 ..,它是扩展运算符,将 user1 中剩余的值传递给 user2(除了已经定义的 email)。
使用 impl 结构体名,并在其中定义函数。
#[derive(Debug)]struct Rectangle { width: u32, height: u32,}impl Rectangle { fn area(&self) -> u32 { self.width * self.height }}fn main() { let rect1 = Rectangle { width: 30, height: 50, }; println!( "长方形的面积为 {}", rect1.area() );}
图片
针对此处更详细的内容,可以翻看我们之前的结构体
枚举,也被称作 enums。枚举允许你通过「列举可能的成员variants来定义一个类型」
enum IpAddrKind { V4, V6,}let four = IpAddrKind::V4;let six = IpAddrKind::V6;
枚举的成员位于其标识符的「命名空间中」,并「使用两个冒号分开」。
这是类似于 switch 的东西,
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, } let number = 5; match number { 1 => println!("One"), 2 => println!("Two"), 3 | 4 | 5 => println!("Three, Four, or Five"), _ => println!("Other"), // 默认情况 }}
每个分支相关联的代码是一个表达式,而表达式的结果值将作为整个 match 表达式的返回值。
图片
这是一种使用 if 的花式方式,我们在其中定义一个表达式。
fn main() { let optional_number: Option<i32> = Some(5); // 使用 if let 匹配 Some 变体并提取内部值 if let Some(num) = optional_number { println!("Value: {}", num); } else { println!("No value"); }}
图片
为了创建一个新线程,需要调用 thread::spawn 函数并「传递一个闭包」,并在其中包含希望在新线程运行的代码。
可以通过将 thread::spawn 的「返回值储存在变量中来修复新建线程部分没有执行或者完全没有执行的问题」。thread::spawn 的返回值类型是 JoinHandle。JoinHandle 是一个「拥有所有权的值」,当「对其调用 join 方法时,它会等待其线程结束」。
use std::thread;fn main() { // 数据 let numbers = vec![1, 2, 3, 4, 5, 6, 7, 8, 9, 10]; // 将数据分成两部分 let mid = numbers.len() / 2; let (left, right) = numbers.split_at(mid); // 生成两个线程来计算每一半的总和 let handle1 = thread::spawn(move || sum(left)); let handle2 = thread::spawn(move || sum(right)); // 等待线程完成并获取它们的结果 let result1 = handle1.join().unwrap(); let result2 = handle2.join().unwrap(); // 计算最终总和 let total_sum = result1 + result2; println!("Total sum: {}", total_sum);}fn sum(numbers: &[i32]) -> i32 { let mut sum = 0; for &num in numbers { sum += num; } sum}
thread::spawn 要求闭包具有 'static 生命周期,这意味着它不会从周围范围借用任何东西,并且可以在整个程序的持续时间内存在。
因此,我们使用move 闭包,其经常与 thread::spawn 一起使用,因为它允许我们「在一个线程中使用另一个线程的数据」。
本文链接:http://www.28at.com/showinfo-26-85705-0.html一网打尽 Rust 语法
声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com
上一篇: 聊聊业务高可用的保障:异地多活架构