58人参与 • 2025-06-27 • rust
rust 是一门多范式语言,既可以像 c++/java 那样写“命令式代码”,也支持“函数式编程”。但很多刚入门的小伙伴可能会有这些疑问:
函数不就是函数吗?什么是纯函数?
什么又是副作用?函数式和我熟悉的 java/c++ 有啥区别?
我该怎么开始写函数式风格的 rust 代码?
别急,今天我们从零出发,把这些看起来很抽象的概念通通讲清楚!
函数式编程(functional programming)是:用纯粹、可组合的函数来表达程序逻辑,同时避免副作用。
看不懂?我们来一句一句拆开讲 👇
很多人第一反应是:“函数不都是函数吗?为啥还要强调‘纯’?”
来,举个例子你就懂了:
举个例子
fn add(a: i32, b: i32) -> i32 {
a + b
}
这个函数:
✅ 所以它是个“纯函数”。
再举一个反面例子
fn print_and_add(a: i32, b: i32) -> i32 {
println!("正在加法运算!");
a + b
}
这个函数除了计算结果,还打印了一句话,这叫做副作用。
副作用 = 函数除了返回结果,还影响了“外部世界”。
| 行为 | 是副作用吗? | 原因 |
|---|---|---|
| 改了一个全局变量 | ✅ 是 | 改变了外部状态 |
打印输出 println! | ✅ 是 | 改变了控制台 |
| 写入文件 | ✅ 是 | 改变了磁盘状态 |
| 发 http 请求 | ✅ 是 | 影响了外部网络 |
| 单纯返回值 | ❌ 否 | 没动外部任何东西 |
为什么函数式编程追求“无副作用”?
因为副作用:
组合性 = 把小函数像积木一样拼起来,组成更大的逻辑
比如:
let data = vec![1, 2, 3, 4, 5];
let result: i32 = data
.iter() // 遍历
.filter(|x| *x % 2 == 0) // 只保留偶数
.map(|x| x * 2) // 每个数翻倍
.sum(); // 求和
println!("{}", result); // 输出 12(2*2 + 4*2)
这段代码没有循环、没有中间变量,却能一步步地处理数据。
每个函数(如 filter, map)都很简单,但组合起来就完成了复杂的逻辑!
这种“拼积木”的能力,就是组合性。
| 概念 | 通俗解释 | 关键目的 |
|---|---|---|
| 纯函数 | 不依赖外部,不改外部,只靠输入决定输出 | 稳定、可预测 |
| 无副作用 | 不打印、不改文件、不改全局变量 | 可测试、线程安全 |
| 可组合 | 把小函数组合成大逻辑 | 简洁、模块化 |
| 对比点 | 命令式(c++/java) | 函数式(rust风格) |
|---|---|---|
| 编程方式 | 写“怎么做” | 写“要什么” |
| 控制结构 | for、if、变量改来改去 | map/filter/链式处理 |
| 状态管理 | 变量经常变化 | 默认不可变 |
| 副作用 | 难避免 | 尽量消除 |
| 函数角色 | 封装逻辑 | 构建模块 |
| 可读性 | 操作细节多 | 更像自然语言表达 |
| 特征 | rust 中的支持方式 | 示例 |
|---|---|---|
| ✅ 纯函数 | 所有普通函数默认都可以写成纯函数 | fn add(a, b) -> a + b |
| ✅ 不可变性 | 默认 let 是不可变的 | let x = 5; |
| ✅ 闭包(匿名函数) | 使用 |x| x + 1 定义 | let f = |x| x + 1; |
| ✅ 高阶函数 | 函数可以作为参数传入 | map(|x| x * 2) |
| ✅ 惰性计算 | iterator 是惰性执行的 | .iter().map().filter() |
| ✅ 函数组合 | 使用链式调用 | .map().filter().sum() |
🔍 解释闭包:闭包就是一个没有名字的“临时函数”,可以捕获外部变量,语法是 |参数| 表达式。
很多人卡在一开始不知道怎么写函数式代码,我们一步步来:
| 写法 | 示例 | 含义 |
|---|---|---|
| 匿名函数(闭包) | 使用 |x| x + 1 定义 | let f = |x| x + 1; |
| 高阶函数 | map(|x| x * 2) | 传函数给函数 |
| 链式调用 | .filter().map() | 像流水线一样处理数据 |
| collect 收集结果 | .collect::<vec<_>>() | 把处理结果收集成 vec |
传统写法:
let mut result = vec![];
for i in 1..=5 {
if i % 2 == 0 {
result.push(i * 2);
}
}
函数式写法:
let result: vec<_> = (1..=5)
.filter(|x| x % 2 == 0)
.map(|x| x * 2)
.collect();
fn operate(x: i32, f: fn(i32) -> i32) -> i32 {
f(x)
}
fn main() {
let double = |x| x * 2;
println!("{}", operate(3, double)); // 输出 6
}
| 优点 | 对初学者的意义 |
|---|---|
| ✅ 代码更短更清晰 | 不需要手动管理中间变量 |
| ✅ 更容易测试 | 没副作用就是好测试 |
| ✅ 更少 bug | 不容易改错变量 |
| ✅ 更好并发支持 | 不争抢变量,天然线程安全 |
for 循环都试着用 .iter().map().filter() 改写iterator trait 的文档option / result 和函数式结合的优雅用法到此这篇关于rust 函数式编程的具体使用的文章就介绍到这了,更多相关rust 函数式编程内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
您想发表意见!!点此发布评论
版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。
发表评论