Rust中的Fn、FnMut 和 FnOnce都有什么区别?

本文涉及的产品
服务治理 MSE Sentinel/OpenSergo,Agent数量 不受限
可观测可视化 Grafana 版,10个用户账号 1个月
简介: 在 Rust 中,`Fn`、`FnMut` 和 `FnOnce` 是三个用于表示闭包类型的 trait。闭包是一种可以捕获其环境变量的函数。在创建闭包是会默认实现这几个 trait 中的一个。

在 Rust 中,FnFnMutFnOnce 是三个用于表示闭包类型的 trait。闭包是一种可以捕获其环境变量的函数。在创建闭包是会默认实现这几个 trait 中的一个。

以下是三个 trait 的区别:

  • FnFn 是最基本的闭包 trait。它表示闭包可以捕获其环境变量的不可变引用。

  • FnMutFnMut 表示闭包可以捕获其环境变量的可变引用。这意味着闭包可以修改其环境变量的值。

  • FnOnceFnOnce 表示闭包只能调用一次。它表示闭包可以捕获其环境变量的所有权。这意味着闭包可以移动其环境变量的值。

以下是三个 trait 的一些示例:

// Fn

let add_one_closure = |x: i32| -> i32 {
   
    let y = 10;
    x + y
};

// FnMut

let add_two_closure = |mut x: i32| -> i32 {
   
    x += 2;
    x
};

// FnOnce

let consume_string_closure = |s: String| -> String {
   
    println!("Consumed string: {}", s);
    s
};

输出结果:

Consumed string: Hello, world!

总结:

  • Fn:闭包可以捕获其环境变量的不可变引用。
  • FnMut:闭包可以捕获其环境变量的可变引用。
  • FnOnce:闭包只能调用一次,并可以捕获其环境变量的所有权。

选择哪个 trait 取决于闭包的用途。

  • 如果闭包只需要读取环境变量的值,则可以使用 Fn trait。
  • 如果闭包需要修改环境变量的值,则可以使用 FnMut trait。
  • 如果闭包只需要使用环境变量的值一次,则可以使用 FnOnce trait。

以下是一些使用建议:

  • 优先使用 Fn trait,因为它是最通用的。
  • 只有在需要修改环境变量的值时才使用 FnMut trait。
  • 只有在需要使用环境变量的值一次时才使用 FnOnce trait。

此外,FnFnMutFnOnce 还可以相互转换。

  • Fn 可以转换为 FnMutFnOnce
  • FnMut 可以转换为 FnOnce

以下是转换示例:

// Fn -> FnMut

fn add_one(x: i32) -> i32 {
   
    let y = 10;
    x + y
}

let mut add_one_fnmut: FnMut(i32) -> i32 = add_one;

// FnMut -> FnOnce

fn add_two(mut x: i32) -> i32 {
   
    x += 2;
    x
}

let add_two_fnonce: FnOnce(i32) -> i32 = add_two;

从源码的角度看它们为什么能够转换:

pub trait FnOnce<Args: Tuple> {
   
    ...
}

pub trait FnMut<Args: Tuple>: FnOnce<Args> {
   
    ...
}

pub trait Fn<Args: Tuple>: FnMut<Args> {
   
    ...
}

从源码可以看出,Fn继承了FnMut,而FnMut又继承了FnOnce。这是类型逐渐泛化的过程,这就好像坤坤属于男人,男人属于灵长类一个道理。
Fn要求只能只能捕获环境变量,而FnMut不但捕获环境变量还需要修改环境变量,所以Fn一定能满足FnMut的条件。就像1是正整数,那么它必然也是整数。同样的,FnOnce要求拿到环境变量的所有权,如此一来,不管是Fn还是FnMut都是在没有所有权的基础上运行的,有了所有权那就是能无所无能了。因此,无论是Fn还是FnMut都可以转换为FnOnce。

总结:

FnFnMutFnOnce 是三个用于表示闭包类型的 trait。选择哪个 trait 取决于闭包的用途。此外,FnFnMutFnOnce 还可以相互转换。

我正在编写《Rust 精华小册》,这是一个免费专栏,有需要的朋友可以自助获取哦,也可以关注我的公众号“程序饲养员”获取最新动态。

相关文章
|
3月前
|
存储 Rust 编译器
Rust中不可变变量与const有何区别?
Rust作者认为变量默认应该是`immutable`,即声明后不能被改变的变量。这一点是让跨语言学习者觉得很别扭,不过这一点小的改变带来了诸多好处,本节我们来学习Rust的变量。
47 9
Rust中不可变变量与const有何区别?
|
4月前
|
Rust 安全 网络协议
Rust 笔记:Rust 语言中的枚举 与 模式匹配
Rust 笔记:Rust 语言中的枚举 与 模式匹配
48 0
|
4月前
|
存储 Rust 安全
Rust 笔记:Rust 语言中的 所有权 与 生命周期
Rust 笔记:Rust 语言中的 所有权 与 生命周期
136 0
|
25天前
|
Rust 安全 程序员
开发语言漫谈-rust
if(){}else{}就是C家族的
|
2月前
|
Rust 监控 数据安全/隐私保护
Rust语言在员工屏幕监控系统中的应用指南
员工屏幕监控系统在现代企业管理中扮演着重要角色。它们能够帮助企业监控员工的活动,确保他们的工作效率和数据安全。在这篇文章中,我们将探讨如何使用Rust语言构建一个简单而高效的员工屏幕监控系统,并提供一些代码示例以帮助你入门。
144 0
|
3月前
|
Rust 监控 JavaScript
抖音技术分享:飞鸽IM桌面端基于Rust语言进行重构的技术选型和实践总结
本文将介绍飞鸽IM前端团队如何结合Rust对飞鸽客户端接待能力进行的技术提升,一步步从概念验证、路径分解到分工开发,再到最后上线收益论证,并分享了其中遇到的技术挑战与经验总结等。
62 1
|
3月前
|
Rust 安全 前端开发
Rust还是其他语言:考量因素与案例分析
【2月更文挑战第1天】本文将探讨在选择编程语言时,为什么Rust可能会成为理想的选择。我们将分析Rust的主要优势,如内存安全、性能、并发编程和所有权系统,并将其与其他流行的编程语言进行比较。此外,我们还将通过具体的案例分析,展示Rust在实际应用中的优势和应用场景。
|
3月前
|
Rust 安全 Java
Rust 语言的类型系统
假如让你设计编程语言的类型,你会怎么做? 要定义哪些类型? 类型之间如何交互? 是否需要类型推断? 类型系统是编程语言中用于定义和控制类型的一组规则。
Rust 语言的类型系统
|
3月前
|
Rust 安全
Rust语言中的控制流:条件语句、循环与模式匹配详解
本文将深入探讨Rust编程语言中的控制流构造,包括条件语句、循环和模式匹配。我们将了解如何使用这些工具来构建高效、可读和安全的代码。此外,我们还将探讨Rust在这些构造中提供的一些独特功能和优化。
|
4月前
|
存储 缓存 Rust
Rust 笔记:Rust 语言中哈希结构(哈希映射,HashMap)、集合(哈希集,HashSet)及其使用
Rust 笔记:Rust 语言中哈希结构(哈希映射,HashMap)、集合(哈希集,HashSet)及其使用
388 0