泛型、Trait 与生命周期基础

学习目标

  • 理解泛型如何减少重复代码。
  • 使用 trait 表达共享行为。
  • 建立生命周期的基本直觉。

泛型

泛型允许函数、结构体或枚举处理多种类型:

fn first<T>(items: &[T]) -> Option<&T> {
    items.first()
}

fn main() {
    let numbers = vec![1, 2, 3];
    let names = vec!["Alice", "Bob"];

    println!("{:?}", first(&numbers));
    println!("{:?}", first(&names));
}

T 是类型参数。它表示“某个具体类型”,编译器会在使用时推断出来。

泛型结构体

struct Point<T> {
    x: T,
    y: T,
}

fn main() {
    let integer = Point { x: 1, y: 2 };
    let float = Point { x: 1.0, y: 2.0 };

    println!("{}, {}", integer.x, float.x);
}

如果两个字段允许不同类型,可以使用两个类型参数:

struct Pair<T, U> {
    left: T,
    right: U,
}

Trait

Trait 定义共享行为,类似“能力约束”:

trait Summary {
    fn summarize(&self) -> String;
}

struct Article {
    title: String,
    author: String,
}

impl Summary for Article {
    fn summarize(&self) -> String {
        format!("{} by {}", self.title, self.author)
    }
}

使用 trait 约束函数参数:

fn print_summary(item: &impl Summary) {
    println!("{}", item.summarize());
}

这表示参数可以是任何实现了 Summary 的类型。

Trait Bound

更完整的泛型写法:

fn print_summary<T: Summary>(item: &T) {
    println!("{}", item.summarize());
}

多个约束可以用 +

use std::fmt::Debug;

fn inspect<T: Summary + Debug>(item: &T) {
    println!("{item:?}");
    println!("{}", item.summarize());
}

复杂约束可以用 where

fn compare_and_print<T, U>(left: &T, right: &U)
where
    T: Summary + Clone,
    U: Summary + std::fmt::Debug,
{
    println!("{}", left.summarize());
    println!("{right:?}");
}

默认实现

Trait 可以提供默认方法:

trait Summary {
    fn author(&self) -> &str;

    fn summarize(&self) -> String {
        format!("article by {}", self.author())
    }
}

实现者只需要提供没有默认实现的方法。

生命周期要解决什么

生命周期描述引用之间的有效范围关系。它不会延长任何值的生命,只是让编译器确认引用不会悬垂。

很多时候生命周期可以省略:

fn first_word(text: &str) -> &str {
    text.split_whitespace().next().unwrap_or("")
}

编译器能推断返回值来自参数 text

当有多个输入引用,返回值来源不明显时,需要标注:

fn longest<'a>(left: &'a str, right: &'a str) -> &'a str {
    if left.len() >= right.len() {
        left
    } else {
        right
    }
}

'a 表示返回引用的有效期不能超过 leftright 中较短的那个。

结构体中的引用

如果结构体存储引用,需要生命周期参数:

struct Excerpt<'a> {
    part: &'a str,
}

fn main() {
    let text = String::from("Rust is fast and safe.");
    let first = text.split('.').next().unwrap_or("");
    let excerpt = Excerpt { part: first };

    println!("{}", excerpt.part);
}

这表示 Excerpt 不能比它引用的字符串片段活得更久。

常见误区

  • 泛型不是运行时动态类型;Rust 通常会在编译期为具体类型生成代码。
  • Trait 是行为抽象,不只是接口语法。
  • 生命周期标注不会改变对象实际存活时间。
  • 遇到生命周期错误时,先检查谁拥有数据、谁借用了数据、返回引用来自哪里。

练习

  1. 写一个泛型函数,返回切片的最后一个元素。
  2. 定义一个 DisplayName trait,并为两个结构体实现它。
  3. 写一个函数,在两个 &str 中返回更短的那个,并添加必要生命周期标注。

后记

2026年6月10日17点42分于上海。

Logo

AtomGit 是由开放原子开源基金会联合 CSDN 等生态伙伴共同推出的新一代开源与人工智能协作平台。平台坚持“开放、中立、公益”的理念,把代码托管、模型共享、数据集托管、智能体开发体验和算力服务整合在一起,为开发者提供从开发、训练到部署的一站式体验。

更多推荐