当前位置: 首页 > news >正文

学习笔记十三—— 理解 Rust 闭包:从语法到 impl Fn vs Box<dyn Fn>

🧠 理解 Rust 闭包:从语法到 impl Fn vs Box


📚 目录

  1. 闭包是什么?和普通函数有什么不同?
  2. 闭包的语法长什么样?
  3. 闭包“捕获变量”是什么意思?
  4. 闭包和所有权的关系
  5. Fn、FnMut、FnOnce 三种闭包类型的区别和例子
  6. 什么是 Trait?闭包为什么要用 Trait?
  7. 为什么闭包“没有具体类型”?
  8. impl Fn 和 Box 的区别:底层逻辑 + 使用示例
  9. 常见闭包错误示例解析
  10. 总结与记忆建议

1️⃣ 闭包是什么?和函数有什么不同?

Rust 中的闭包是“一种匿名函数”,它和普通函数的不同在于:

它可以使用外部作用域的变量

📌 举个例子:

fn main() {let name = "Tom";let greet = || println!("Hello, {}", name); // 使用了外部变量 namegreet(); // 输出:Hello, Tom
}

这个 || println!(...) 就是闭包,虽然没有参数,但它能自动“记住”外部的 name


2️⃣ 闭包的语法长什么样?

let closure = |参数| 表达式;

🧪 示例:

let square = |x: i32| x * x;
println!("{}", square(4)); // 输出 16

参数可以省略类型,Rust 会自动推断:

let double = |x| x * 2;

3️⃣ 闭包“捕获变量”是什么意思?

闭包之所以强大,是因为它可以用外部的变量,比如:

let count = 5;
let show = || println!("{}", count);

这个闭包自动“借用了”变量 count。它不像普通函数必须把变量作为参数。

这种“带着环境走”的能力叫捕获变量


4️⃣ 闭包和所有权的关系

Rust 中使用变量要遵守所有权规则,闭包捕获变量时有三种方式:

捕获方式说明所属 Trait
借用(&T)只读Fn
可变借用(&mut T)修改变量FnMut
移动(T)拿走变量所有权FnOnce

Rust 会根据闭包内部的行为自动判断。


5️⃣ Fn、FnMut、FnOnce 的区别

Fn:只读、可多次调用

let name = "Alice";
let say_hi = || println!("Hi, {}", name); // 只读借用
say_hi();
say_hi(); // OK,多次调用

FnMut:可修改外部变量

let mut count = 0;
let mut inc = || count += 1; // 可变借用
inc();
inc();
println!("{}", count); // 输出 2

FnOnce:拿走变量所有权,只能用一次

let s = String::from("hi");
let consume = move || println!("{}", s); // s 被 move 进闭包
consume();
// consume(); ❌ 错:值已被使用

6️⃣ 什么是 Trait?闭包为什么要用 Trait?

Trait(特质)就是 Rust 中的“能力接口”。

谁实现了某个 Trait,就可以被当成“具有某种能力”的对象使用。

闭包没有固定类型,只能通过它实现的 Trait 来使用,比如:

  • Fn() 表示能多次调用
  • FnMut() 表示可变调用
  • FnOnce() 表示调用一次

7️⃣ 为什么闭包“没有具体类型”?

❓ 你写过这样的代码吗?

let c = |x| x + 1;
// let f: ??? = c; // 编译器报错!闭包没有类型名

这是因为:

Rust 中的闭包是编译器自动生成的匿名结构体,你无法直接用名字去写它的类型。

🔍 其实编译器背后大致生成了一个这样的结构体:

struct Closure {x: i32
}
impl Fn(i32) for Closure {fn call(&self, y: i32) -> i32 {y + self.x}
}

所以你只能通过它实现的 Fn 系列 Trait 来“访问”它。


8️⃣ impl Fn 和 Box 的区别:底层逻辑 + 使用示例

🧩 这两种写法都能接收闭包:

✅ 写法一:impl Fn()(静态分发)
fn call_twice(f: impl Fn()) {f();f();
}fn main() {let name = "Tom";let say_hi = || println!("Hi {}", name);call_twice(say_hi); // OK
}
  • 编译时就知道闭包的类型(编译器展开 inline)
  • ,无额外开销

✅ 写法二:Box<dyn Fn()>(动态分发)
fn call_twice(f: Box<dyn Fn()>) {f();f();
}fn main() {let name = "Tom".to_string();let say_hi = move || println!("Hi {}", name);call_twice(Box::new(say_hi)); // OK
}
  • Box<dyn Fn()> 是一个Trait 对象
  • 用于运行时决定具体调用哪个函数(通过虚表 vtable 实现)
  • 更加灵活(适合多个不同类型的闭包集合)

🧪 类比理解:

对比维度impl Fn()Box<dyn Fn()>
类型是否确定编译时确定编译时未知,运行时查表
性能快(无虚表)稍慢(要查 vtable)
是否堆分配不需要
使用场景简单、性能敏感场景复杂或多种类型的集合场景

🧠 类比:点菜 vs 做饭

  • impl Fn() 就像自己做饭,提前准备、速度快;
  • Box<dyn Fn()> 就像去餐厅点菜,灵活但慢一点,因为要看菜单(vtable)。

❗ 什么时候必须用 Box<dyn Fn()>?

比如你要存多个不同闭包进 Vec:

let mut funcs: Vec<Box<dyn Fn()>> = vec![];funcs.push(Box::new(|| println!("hello")));
funcs.push(Box::new(|| println!("world")));for f in funcs {f(); // 运行时查表调用
}

不能用 Vec<impl Fn()>,因为每个闭包的底层结构体不同,大小不一样,Rust 不允许放在一个 Vec 里。


9️⃣ 常见闭包错误示例解析

❌ 错误:借用了可变变量,但用 Fn

fn twice<F: Fn()>(f: F) {f();f();
}let mut count = 0;
let mut closure = || count += 1;twice(closure); // ❌ 错误!因为 closure 是 FnMut

✅ 修复方式:

fn twice<F: FnMut()>(mut f: F) {f();f();
}

🔟 总结与记忆建议

概念通俗解释
闭包能记住外部变量的小函数
Trait能力接口,如 Fn 表示可调用
impl Fn()编译期固定,快
Box<dyn Fn()>运行时多态,灵活
Trait 对象抽象能力 + 虚表,运行时查找实际调用方法

🧩 记忆口诀

闭包没名字,Trait 来代管;
impl 是静态,Box 是多态;
多种闭包放 Vec,必须用 Box;
借改拿三种捕获,决定 Fn 哪种管。

相关文章:

  • 工作记录3
  • Spark-Sql编程(三)
  • 计算机视觉——基于人工智能视觉注意力的在线广告中评估检测技术
  • 二进制求和 - 简单
  • 数据加载与保存
  • Ubuntu服务器中了木马且处于局域网内无法直接通过公网正向连接
  • Mac OS系统下kernel_task占用大量CPU资源导致系统卡顿
  • Linux:Makefile
  • 数字电子技术基础(四十七)——使用Mutlisim软件来模拟74LS85芯片
  • STM32基础教程——DMA+ADC多通道
  • 【后端】【python】利用反射器----动态设置装饰器
  • 智能语音处理+1.1下载需要的库(100%实现)
  • 【Lerobot】加载本地数据LeRobotDataset数据、读取并解析parquet
  • 【c语言】深入理解指针1
  • 排序(java)
  • 任务的状态
  • 投资理财_从0到1:如何用1000元开启你的二级市场投资之旅?
  • 实战5:Python使用循环神经网络生成诗歌
  • 解决virtualbox7.1无法启动3d加速的问题
  • 大数据人工智能
  • VM2008 做网站/百度一下你就知道手机版
  • 网站页面排版/seo平台有哪些
  • 真人做爰直播视频网站/网站排名系统
  • 网络工作室怎么赚钱/自动app优化最新版
  • 手机网站宽度多少合适/东莞网络推广公司
  • 网络公司怎么优化网站/开发一个app需要多少钱