【Rust指南】枚举类与模式匹配精讲

简介: 【Rust指南】枚举类与模式匹配精讲

1、Rust基本枚举类语法


枚举允许我们列举所有可能的值来定义一个类型,枚举中的值也叫变体


1.1、定义枚举


我们举一个例子:


IP地址:IPV4、IPV6
enum IpAddrKind{
  V4,
  V6
}


接收枚举值

let four=IpAddrKind::V4;
let six=IpAddrKind::V6;


枚举的变体都位于标识符的命名空间下,使用::进行分割


1.2、将数据附加到枚举的变体中


形式如下:


enum IpAddr{
  V4(String),
  V6(String)
}


优点:


不需要额外使用struct来确定类型

每个变体可以拥有不同的类型以及关联的数据量

例如:


#[derive(Debug)]
enum IpAddrKind {
    V4(u8,u8,u8,u8),
    V6(String)
}
fn main() {
    let home=IpAddrKind::V4(127, 0, 0, 1);
    let loopback=IpAddrKind::V6(String::from("这是IPV6"));
    println!("{:?}\n{:?}",home,loopback);
}


运行效果:



#[derive(Debug)]作为Rust提供的调试库是可以直接输出结构体和枚举类型的


但是注意占位符只能使用{:?}


标准库中的IpAddr


struct IpV4Addr{
    //--snip--
}
struct IpV6Addr{
    //--snip--
}
enum IpAddr {
    V4(IpV4Addr),
    V6(IpV6Addr)
}


1.3、变体的多种嵌套方式


enum Message {
    Quit,
    Move {x:i32,y:u32},
    Write(String),
    ChangeColor(i32,i32,i32)
}
fn main() {
    let q=Message::Quit;
    let m=Message::Move { x: 6, y: 12 };
    let w=Message::Write(String::from("hello_world"));
    let c=Message::ChangeColor(255, 255, 0);
}


在这段代码中枚举类变体一共有四种数据类型:


不带关联数据Quit

匿名结构体Move

字符串类型Write

匿名元组结构体ChangeColor


1.4、定义枚举方法


和结构体方法类似,使用impl关键字:


impl Message{
  fn call(&self){}
}


这里就不具体实现了,此时枚举的所有变体都可以调用call方法,例如q.call();


2、Option枚举


2.1、引入Option枚举解决控制问题


Option 是 Rust 标准库中的枚举类,这个类用于填补 Rust 不支持 null 引用的空白。


许多语言支持 null 的存在(C/C++、Java),这样很方便,但也制造了极大的问题,null 的发明者也承认这一点,“一个方便的想法造成累计 10 亿美元的损失”。


null 经常在开发者把一切都当作不是 null 的时候给予程序致命一击:毕竟只要出现一个这样的错误,程序的运行就要彻底终止。


为了解决这个问题,很多语言默认不允许 null,但在语言层面支持 null 的出现(常在类型前面用 ? 符号修饰)。


Java 默认支持 null,但可以通过 @NotNull 注解限制出现 null,这是一种应付的办法。


Rust 在语言层面彻底不允许空值 null 的存在,但无奈null 可以高效地解决少量的问题,所以 Rust 引入了 Option 枚举类:


enum Option<T>{
  Some(T),
  None
}

2.2、枚举类的具体使用


枚举类包含在预导入模块中(Prelude),可直接使用:


let some_number=Some(5);
let some_string=Some("a string")
let absent:Option<&str>=None;


注意:


编译器无法推断None是什么类型,所以一定要显示声明

由于absent属于None的变体,因此是无效数据,也就是null


3、match控制流运算符


枚举的目的是对某一类事物的分类,分类的目的是为了对不同的情况进行描述。

基于这个原理,往往枚举类最终都会被分支结构处理(许多语言中的 switch )。

switch 语法很经典,但在 Rust 中并不支持,很多语言摒弃 switch 的原因都是因为 switch 容易存在因忘记添加 break 而产生的串接运行问题,Java 和 C# 这类语言通过安全检查杜绝这种情况出现。

Rust 通过 match 语句来实现分支结构。先认识一下如何用 match 处理枚举类:


fn main() {
    enum Book {
        Papery {index: u32},
        Electronic {url: String},
    }
    let book = Book::Papery{index: 1001};
    let ebook = Book::Electronic{url: String::from("url...")};
    match book {
        Book::Papery { index } => {
            println!("Papery book {}", index);
        },
        Book::Electronic { url } => {
            println!("E-book {}", url);
        }
    }
}
//运行结果:Papery book 1001


这是由于book属于Papery的变体,因此会执行第一个打印语句

match 块也可以当作函数表达式来对待,它也是可以有返回值的:


match 枚举类实例 {
    分类1 => 返回值表达式,
    分类2 => 返回值表达式,
    ...
}


但是要谨记:所有返回值表达式的类型必须一样!

如果把枚举类附加属性定义成元组,在 match 块中需要临时指定一个名字:


enum Book {
    Papery(u32),
    Electronic {url: String},
}
let book = Book::Papery(1001);
match book {
    Book::Papery(i) => {
        println!("{}", i);
    },
    Book::Electronic { url } => {
        println!("{}", url);
    }
}


变体Papery指定了i变量,Electronic指定了url

match 除了能够对枚举类进行分支选择以外,还可以对整数、浮点数、字符和字符串切片引用(&str)类型的数据进行分支选择。其中,浮点数类型被分支选择虽然合法,但不推荐这样使用,因为精度问题可能会导致分支错误。


对非枚举类进行分支选择时必须注意处理例外情况,即使在例外情况下没有任何要做的事。例外情况用下划线 _ 表示:


fn main() {
    let t = "abc";
    match t {
        "abc" => println!("Yes"),
        _ => {},
    }
}


4、if let 语法


通过一个简单的流程控制代码理解此部分知识:


let i = 0;
match i {
    0 => println!("zero"),
    _ => {},
}
//主函数中运行结果:zero


这段程序的目的是判断 i 是否是数字 0,如果是就打印 zero。

那么现在用 if let 语法缩短这段代码:


let i = 0;
if let 0 = i {
    println!("zero");
}


if let 语法格式如下:

if let 匹配值 = 源变量 {
    语句块
}


可以在之后添加一个 else 块来处理例外情况。

if let 语法可以认为是只区分两种情况的 match 语句的"语法糖"


在枚举类中的使用:


fn main() {
    enum Book {
        Papery(u32),
        Electronic(String)
    }
    let book = Book::Electronic(String::from("url"));
    if let Book::Papery(index) = book {
        println!("Papery {}", index);
    } else {
        println!("Not papery book");
    }
}
//运行结果:Not papery book
相关文章
|
5月前
|
Rust 安全 网络协议
Rust 笔记:Rust 语言中的枚举 与 模式匹配
Rust 笔记:Rust 语言中的枚举 与 模式匹配
76 0
|
2月前
|
Rust 安全 C++
30天拿下Rust之枚举
Rust中的枚举是一种用户定义的类型,它允许你为一组相关的值赋予友好的名称。在Rust中,枚举是强大的工具,它们不仅仅用于表示几个固定的值,还可以包含函数和方法,使得枚举成员可以有自己的行为。通过与模式匹配和其他Rust特性结合使用,枚举在构建健壮、无崩溃的应用程序中发挥了重要作用,并可大幅提高代码的可读性、可维护性和类型安全性。
48 10
|
28天前
|
Rust 安全 开发者
30天拿下Rust之模式与模式匹配
30天拿下Rust之模式与模式匹配
44 1
|
2月前
|
Rust 开发者 C#
解锁Rust高手的秘密武器:模式匹配与宏,学会这一招,编程效率翻倍!
【8月更文挑战第31天】Xamarin 是移动应用开发领域的强大跨平台工具,采用 C# 语言,具备高代码复用性、熟悉开发语言及接近原生性能等优势。开发者可通过共享项目实现多平台业务逻辑复用,简化开发流程。然而,Xamarin 也存在学习曲线陡峭、需处理平台差异及第三方库兼容性等问题。总体而言,Xamarin 在提高开发效率的同时,也对开发者提出了新的挑战。
22 0
|
4月前
|
Rust
Rust的if let语法:更简洁的模式匹配
Rust的if let语法:更简洁的模式匹配
|
5月前
|
Rust 安全 算法
【深入探索Rust:结构体、枚举与模式匹配】A Deep Dive into Rust: Structs, Enums, and Pattern Matching
【深入探索Rust:结构体、枚举与模式匹配】A Deep Dive into Rust: Structs, Enums, and Pattern Matching
89 0
【深入探索Rust:结构体、枚举与模式匹配】A Deep Dive into Rust: Structs, Enums, and Pattern Matching
|
5月前
|
Rust 安全
Rust语言中的控制流:条件语句、循环与模式匹配详解
本文将深入探讨Rust编程语言中的控制流构造,包括条件语句、循环和模式匹配。我们将了解如何使用这些工具来构建高效、可读和安全的代码。此外,我们还将探讨Rust在这些构造中提供的一些独特功能和优化。
|
5月前
|
C++ Rust NoSQL
Rust 数据类型 之 类C枚举 c-like enum
Rust 数据类型 之 类C枚举 c-like enum
57 0
Rust 数据类型 之 类C枚举 c-like enum
|
缓存 Rust 网络协议
一行“无用”的枚举反使Rust执行效率提升10%,编程到最后都是极致的艺术!
最近不少读者都留言说博客中的代码越来越反哺归真,但讨论的问题反倒越来越高大上了,从并发到乱序执行再到内存布局各种放飞自我。 其实这倒不是什么放飞,只是Rust对我来说学习门槛太高了,学习过程中的挫败感也很强,在写完了之前的《Rust胖指针胖到底在哪》之后笔者一度决定脱坑Rust了,但截至本周这个目标还是没有实现,因为我所在的Rust学习群,有一个灵魂拷问,Rust的技术本质什么?不回答好这个问题,我简真是没法得到安宁。
一行“无用”的枚举反使Rust执行效率提升10%,编程到最后都是极致的艺术!
|
2月前
|
Rust 安全 Go
揭秘Rust语言:为何它能让你在编程江湖中,既安全驰骋又高效超车,颠覆你的编程世界观!
【8月更文挑战第31天】Rust 是一门新兴的系统级编程语言,以其卓越的安全性、高性能和强大的并发能力著称。它通过独特的所有权和借用检查机制解决了内存安全问题,使开发者既能享受 C/C++ 的性能,又能避免常见的内存错误。Rust 支持零成本抽象,确保高级抽象不牺牲性能,同时提供模块化和并发编程支持,适用于系统应用、嵌入式设备及网络服务等多种场景。从简单的 “Hello World” 程序到复杂的系统开发,Rust 正逐渐成为现代软件开发的热门选择。
56 1