首页 > 其他分享 >30天拿下Rust之超级好用的“语法糖”

30天拿下Rust之超级好用的“语法糖”

时间:2024-04-06 09:32:26浏览次数:22  
标签:name 30 Person let fn println main 好用 Rust

概述

        Rust语言的设计非常注重开发者的体验,因此它包含了许多实用的“语法糖”。这些“语法糖”让代码更简洁、易读,同时保持了语言的强大和灵活性。

1、字符串插值

        字符串插值允许我们在字符串中嵌入变量或表达式的值,使用{}作为占位符。

fn main() {
    let name = "CSDN";
    let text = format!("Hello {}", name);
    println!("{}", text);
}

2、if-let表达式

        if let是if和let的结合,用于在if条件中同时进行模式匹配和解构赋值。

fn main() {
    let x = Some(66);
    if let Some(value) = x {
            println!("value is: {}", value);
    } else {
            println!("value is None");
    }
}

3、Range表达式

        ..是不包含结束值的范围操作符,..=是包含结束值的范围操作符。

fn main() {
    for i in 1..=5 {
        println!("{}", i);
    }

    for i in 1..5 {
        println!("{}", i);
    }
}

4、闭包

        闭包是匿名的函数,它们可以捕获其创建环境的变量。在下面的示例代码中,闭包被用来过滤偶数。

fn main() {
    let numbers = vec![1, 2, 3, 4, 5];
    let even_numbers = numbers.iter().filter(|&x| x % 2 == 0).collect::<Vec<_>>();
    println!("{:?}", even_numbers);
}

5、推导式

        通过#[derive(...)]属性,Rust可以自动为结构体或枚举生成标准的方法实现,比如:Debug、Eq、PartialEq等。在下面的示例代码中,我们为Person结构体类型使用了#[derive(Debug)]属性,因此编译器为我们自动生成了Debug Trait的实现。在main函数中,我们使用{:?}格式化占位符来打印person变量的调试信息。

#[derive(Debug)]
struct Person {
    name: String,
    age: u8,
}

fn main() {
    let person = Person {
        name: "Mike".to_string(),
        age: 24,
    };

    // 输出:Person { name: "Mike", age: 24 }
    println!("{:?}", person);
}

6、使用结构体更新语法

        结构体更新语法允许我们在创建或修改结构体实例时,仅指定需要更改的字段,而其余字段则保持其默认值或现有值不变。这种语法特别适用于当结构体有很多字段,而只想改变其中一个或几个字段时。

#[derive(Debug)]
struct Person {
    name: String,
    age: u8,
    city: String,
}

fn main() {
    let mut person = Person {
        name: "Mike".to_string(),
        age: 24,
        city: "London".to_string(),
    };

    // 除name字段外,其余字段保持不变
    person = Person {
        name: "Bob".to_string(),
        ..person
    };

    // 输出: Person { name: "Bob", age: 24, city: "London" }
    println!("{:?}", person);
}

7、使用_忽略不关心的值

        在解构赋值时,可以使用_来忽略我们不关心的值。

fn main() {
    let (_, b) = (66, 99);
    // 输出:99
    println!("{}", b);
}

8、使用?在表达式中处理Result

        ?操作符可以在Result类型的表达式中用来自动处理错误。如果Result是Err,则整个表达式立即返回错误。如果是Ok,则继续执行。

use std::fs::File;
use std::io::Read;

fn read_file(path: &str) -> Result<String, std::io::Error> {
    let mut file = File::open(path)?;
    let mut contents = String::new();
    file.read_to_string(&mut contents)?;
    Ok(contents)
}

fn main() {
    match read_file("example.txt") {
        Ok(contents) => println!("file contents: {}", contents),
        Err(error) => eprintln!("read file failed: {}", error),
    }
}

9、使用break 'label在嵌套循环中提前退出

        通过在循环前使用标签(比如:下面的'outer:和'inner:),可以在嵌套循环中使用break 'label来提前退出指定标签的循环。

fn main() {
    'outer: for i in 0..5 {
        'inner: for j in 0..5 {
            if i == 2 && j == 2 {
                // 跳出外层循环
                break 'outer;
            }
            
            println!("({}, {})", i, j);
        }
    }
}

10、Deref转换

        Deref转换允许通过*操作符自动解引用实现了Deref Trait的类型,从而访问其内部的值。这是一种隐式的类型转换,使得代码更加简洁。

use std::ops::Deref;

struct CustomBox<T> {
    value: T,
}

impl<T> Deref for CustomBox<T> {
    type Target = T;
    fn deref(&self) -> &Self::Target {
        &self.value
    }
}

fn main() {
    let my_box = CustomBox { value: 66 };
    // 通过Deref转换,可以直接解引用MyBox实例,从而访问其内部的value字段
    println!("{}", *my_box);
}

标签:name,30,Person,let,fn,println,main,好用,Rust
From: https://blog.csdn.net/hope_wisdom/article/details/137201746

相关文章

  • rust 面向对象编程特性、模式与模式匹配、高级特征
    面向对象编程OOP学习了结构体、枚举,它们可以包含自定义数据字段,也可以定义内部方法,它们提供了与对象相同的功能。面向对象的四大特征:封装、继承、多态通过pub标记为公有的结构体,在其他模块中可以访问使用这个结构体。但是对于结构体内部字段,如果不用pub,则仍是私有的,则可以通过......
  • Rust语言基础:语法、数据类型与操作符
    Rust语言基础:语法、数据类型与操作符Rust是一种系统编程语言,致力于安全、并发和实用性。它是由Mozilla基金会开发的,并得到了广泛的应用。在本篇文章中,我们将带你了解Rust的基础知识,包括语法、数据类型和操作符。1.Rust的语法Rust的语法类似于C++和Java,但同时又更加简洁......
  • LeetCode in Python 300. Longest Increasing Subsequence (最长递增子序列)
    求最长递增子序列是深度优先搜索(DFS)的一种应用,有两种比较好的方法可以解决。第一种是动态规划法,时间复杂度为O(n*n),即设置边界条件和更新迭代公式求解最优解。第二种使用二分查找将时间复杂度降为O(nlogn)。本文给出两种方法的实现代码及说明。示例:图1最长递增子序列输入......
  • 为什么可以使用TrustZone和CCA来构建TEE
    前言    使用ARMTrustZone和CCA来构建可信执行环境基于这两种技术提供的安全特性和目标。它们各自在不同层面提供硬件级别的安全保障和隔离,能够有效地保护代码和数据在执行时不被操作系统或其他应用访问,即使是在有高级权限的软件也无法突破这一保护。这种安全级别对......
  • 文心一言 VS 讯飞星火 VS chatgpt (230)-- 算法导论17.1 1题
    一、如果栈操作包括MULTIPUSH操作,它将k个数据项压入栈中,那么栈操作的摊还代价的界还是O(1)吗?文心一言:在栈的基本操作中,我们通常考虑的是单个元素的压入(push)和弹出(pop)操作,这些操作的摊还代价通常是O(1)。摊还代价考虑的是一系列操作的平均代价,而不是单个操作的最坏......
  • P3052 [USACO12MAR] Cows in a Skyscraper G
    原题链接题解模拟,遍历n个物品,一开始一个箱子不给,遍历到某个物品时,先把所有已经给了的箱子放进去试试,再创一个新箱子放进去试试code#include<bits/stdc++.h>usingnamespacestd;intn,w;intcnt,ans;intchongdie=0;intbox[20],c[20];voidmoni(intnow,intcnt)//now......
  • CF1530G 题解
    考虑对操作进行转换。假设\(a_i\)为第\(i\)个\(1\)前面的\(0\)的个数。则操作可以进行如下转换:转换1:选择一个长度为\(k+1\)的子区间\(a_{l~l+k}\)。我们先把\(a_{l+1~l+k-1}\)翻转,然后更改\(a_l\)和\(a_{l+k}\)使得\(a_l+a_{l+k}\)不......
  • 代码随想录第30天|51. N皇后
    51. N皇后51.N皇后-力扣(LeetCode)代码随想录(programmercarl.com)这就是传说中的N皇后?回溯算法安排!|LeetCode:51.N皇后_哔哩哔哩_bilibili按照国际象棋的规则,皇后可以攻击与之处在同一行或同一列或同一斜线上的棋子。n 皇后问题 研究的是如何将 n 个皇后放置在......
  • 挑战程序设计竞赛 2.6章习题 POJ 1930 Dead Fraction
    https://vjudge.csgrandeur.cn/problem/POJ-1930迈克在最后一刻拼命地赶着完成他的论文。在接下来的3天里,他需要将所有的研究笔记整理成较为连贯的形式。不幸的是,他注意到他在计算方面非常粗心。每当他需要进行算术运算时,他只是将其输入计算器,并将他认为相关的答案写下来。每当......
  • 2024.3.30 模拟赛
    A数列删除至少删除\(m\)个数,意思就是最多保留\(n-m\)个数。删除的总和最小,意思就是保留的总和最大。非降子序列问题可以用经典的动态规划来解决。用\(f[i][j]\)表示,当前选的最后一个数是\(a[i]\),一共选了\(j\)个数,选的数总和最大是多少。转移就是枚举上一个数\(a[......