Rust 泛型类型约束

RustRustBeginner
立即练习

This tutorial is from open-source community. Access the source code

💡 本教程由 AI 辅助翻译自英文原版。如需查看原文,您可以 切换至英文原版

简介

在本实验中,在Rust中使用泛型时,类型参数必须由trait界定,以指定类型必须实现的功能。

注意:如果实验未指定文件名,你可以使用任何你想要的文件名。例如,你可以使用main.rs,并通过rustc main.rs &&./main进行编译和运行。


Skills Graph

%%%%{init: {'theme':'neutral'}}%%%% flowchart RL rust(("Rust")) -.-> rust/BasicConceptsGroup(["Basic Concepts"]) rust(("Rust")) -.-> rust/DataTypesGroup(["Data Types"]) rust(("Rust")) -.-> rust/FunctionsandClosuresGroup(["Functions and Closures"]) rust(("Rust")) -.-> rust/DataStructuresandEnumsGroup(["Data Structures and Enums"]) rust(("Rust")) -.-> rust/AdvancedTopicsGroup(["Advanced Topics"]) rust/BasicConceptsGroup -.-> rust/variable_declarations("Variable Declarations") rust/DataTypesGroup -.-> rust/floating_types("Floating-point Types") rust/FunctionsandClosuresGroup -.-> rust/function_syntax("Function Syntax") rust/FunctionsandClosuresGroup -.-> rust/expressions_statements("Expressions and Statements") rust/DataStructuresandEnumsGroup -.-> rust/method_syntax("Method Syntax") rust/AdvancedTopicsGroup -.-> rust/traits("Traits") rust/AdvancedTopicsGroup -.-> rust/operator_overloading("Traits for Operator Overloading") subgraph Lab Skills rust/variable_declarations -.-> lab-99348{{"Rust 泛型类型约束"}} rust/floating_types -.-> lab-99348{{"Rust 泛型类型约束"}} rust/function_syntax -.-> lab-99348{{"Rust 泛型类型约束"}} rust/expressions_statements -.-> lab-99348{{"Rust 泛型类型约束"}} rust/method_syntax -.-> lab-99348{{"Rust 泛型类型约束"}} rust/traits -.-> lab-99348{{"Rust 泛型类型约束"}} rust/operator_overloading -.-> lab-99348{{"Rust 泛型类型约束"}} end

界限

在使用泛型时,类型参数通常必须使用trait作为界限,以规定类型实现了哪些功能。例如,以下示例使用trait Display 进行打印,因此它要求 TDisplay 约束;也就是说,T 必须 实现 Display

// 定义一个函数 `printer`,它接受一个泛型类型 `T`,该类型
// 必须实现trait `Display`。
fn printer<T: Display>(t: T) {
    println!("{}", t);
}

设定界限会将泛型限制为符合该界限的类型。也就是说:

struct S<T: Display>(T);

// 错误!`Vec<T>` 没有实现 `Display`。此
// 特化将会失败。
let s = S(vec![1]);

设定界限的另一个作用是,泛型实例可以访问界限中指定的trait的[方法]。例如:

// 一个实现打印标记 `{:?}` 的trait。
use std::fmt::Debug;

trait HasArea {
    fn area(&self) -> f64;
}

impl HasArea for Rectangle {
    fn area(&self) -> f64 { self.length * self.height }
}

#[derive(Debug)]
struct Rectangle { length: f64, height: f64 }
#[allow(dead_code)]
struct Triangle  { length: f64, height: f64 }

// 泛型 `T` 必须实现 `Debug`。无论
// 是什么类型,这都能正常工作。
fn print_debug<T: Debug>(t: &T) {
    println!("{:?}", t);
}

// `T` 必须实现 `HasArea`。任何满足
// 该界限的类型都可以访问 `HasArea` 的函数 `area`。
fn area<T: HasArea>(t: &T) -> f64 { t.area() }

fn main() {
    let rectangle = Rectangle { length: 3.0, height: 4.0 };
    let _triangle = Triangle  { length: 3.0, height: 4.0 };

    print_debug(&rectangle);
    println!("Area: {}", area(&rectangle));

    //print_debug(&_triangle);
    //println!("Area: {}", area(&_triangle));
    // ^ TODO: 尝试取消注释这些行。
    // | 错误:未实现 `Debug` 或 `HasArea`。
}

另外需要注意的是,在某些情况下,where 子句也可用于应用界限,以使其更具表现力。

总结

恭喜你!你已完成“界限”实验。你可以在LabEx中练习更多实验来提升你的技能。