Rust 函数式编程的具体使用

 更新时间:2025年06月27日 10:11:32   作者:pumpkin84514  
本文主要介绍了Rust 函数式编程的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

Rust 是一门多范式语言,既可以像 C++/Java 那样写“命令式代码”,也支持“函数式编程”。但很多刚入门的小伙伴可能会有这些疑问:

函数不就是函数吗?什么是纯函数?
什么又是副作用?函数式和我熟悉的 Java/C++ 有啥区别?
我该怎么开始写函数式风格的 Rust 代码?

别急,今天我们从零出发,把这些看起来很抽象的概念通通讲清楚

一、什么是函数式编程?一句话概括

函数式编程(Functional Programming)是:用纯粹、可组合的函数来表达程序逻辑,同时避免副作用。

看不懂?我们来一句一句拆开讲 👇

1. 什么是“纯粹的函数”(纯函数)?

很多人第一反应是:“函数不都是函数吗?为啥还要强调‘纯’?”

来,举个例子你就懂了:

举个例子

fn add(a: i32, b: i32) -> i32 {
    a + b
}

这个函数:

  • 输入什么,输出就是什么(比如 2 + 3 永远等于 5)
  • 不会打印东西、不写文件、不改全局变量

✅ 所以它是个“纯函数”。

再举一个反面例子

fn print_and_add(a: i32, b: i32) -> i32 {
    println!("正在加法运算!");
    a + b
}

这个函数除了计算结果,还打印了一句话,这叫做副作用

2. 什么是“副作用”?

副作用 = 函数除了返回结果,还影响了“外部世界”

行为是副作用吗?原因
改了一个全局变量✅ 是改变了外部状态
打印输出 println!✅ 是改变了控制台
写入文件✅ 是改变了磁盘状态
发 HTTP 请求✅ 是影响了外部网络
单纯返回值❌ 否没动外部任何东西

为什么函数式编程追求“无副作用”?

因为副作用:

  • 会让程序变得难以预测(打印/写文件在哪影响了谁?)
  • 不利于并发(多个线程操作全局变量可能会打架)
  • 不好测试(一个函数打印日志、改配置很难自动验证)

3. 什么是“组合性”?为啥函数式编程要“可组合”?

组合性 = 把小函数像积木一样拼起来,组成更大的逻辑

比如:

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)

这段代码没有循环、没有中间变量,却能一步步地处理数据。
每个函数(如 filtermap)都很简单,但组合起来就完成了复杂的逻辑!

这种“拼积木”的能力,就是组合性

函数式编程的三大思想总结:

概念通俗解释关键目的
纯函数不依赖外部,不改外部,只靠输入决定输出稳定、可预测
无副作用不打印、不改文件、不改全局变量可测试、线程安全
可组合把小函数组合成大逻辑简洁、模块化

函数式 VS 命令式(C++/Java)

对比点命令式(C++/Java)函数式(Rust风格)
编程方式写“怎么做”写“要什么”
控制结构for、if、变量改来改去map/filter/链式处理
状态管理变量经常变化默认不可变
副作用难避免尽量消除
函数角色封装逻辑构建模块
可读性操作细节多更像自然语言表达

Rust 支持函数式编程的方式(显著特征表)

特征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()

🔍 解释闭包:闭包就是一个没有名字的“临时函数”,可以捕获外部变量,语法是 |参数| 表达式

如何从零开始上手 Rust 函数式编程?

很多人卡在一开始不知道怎么写函数式代码,我们一步步来:

第一步:掌握函数式写法格式

写法示例含义
匿名函数(闭包)使用 |x| x + 1 定义let f = |x| x + 1;
高阶函数map(|x| x * 2)传函数给函数
链式调用.filter().map()像流水线一样处理数据
collect 收集结果.collect::<Vec<_>>()把处理结果收集成 Vec

第二步:从 for 循环重构开始

传统写法:

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
}

总结:Rust 函数式编程,到底有什么价值?

优点对初学者的意义
✅ 代码更短更清晰不需要手动管理中间变量
✅ 更容易测试没副作用就是好测试
✅ 更少 bug不容易改错变量
✅ 更好并发支持不争抢变量,天然线程安全

后续你可以这样学习函数式思维:

  • 把所有 for 循环都试着用 .iter().map().filter() 改写
  • 学会闭包、理解闭包和变量捕获
  • 阅读标准库 Iterator Trait 的文档
  • 多写链式组合:map、filter、fold、collect
  • 理解 Option / Result 和函数式结合的优雅用法

到此这篇关于Rust 函数式编程的具体使用的文章就介绍到这了,更多相关Rust 函数式编程内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家! 

相关文章

  • Rust 中的 Packages 与 Crates模块化构建的基础及开发流程

    Rust 中的 Packages 与 Crates模块化构建的基础及开发流程

    Rust中的Crate是编译器处理的最小代码单元,可以是二进制或库,每个Crate由一个CrateRoot文件(通常是src/main.rs或src/lib.rs)定义,本文给大家介绍Rust 中的 Packages 与 Crates模块化构建的基础及开发流程,感兴趣的朋友一起看看吧
    2025-02-02
  • 一文弄懂rust生命周期

    一文弄懂rust生命周期

    生命周期是Rust语言中的一个概念,用于决内存安全问题,本文主要介绍了一文弄懂rust生命周期,具有一定的参考价值,感兴趣的可以了解一下
    2023-12-12
  • Rust 中 Deref Coercion讲解

    Rust 中 Deref Coercion讲解

    Rust 的设计理念一向是显式比隐式好,也就是说所有的行为尽量在代码中表现出来,这篇文章主要介绍了Rust 中 Deref Coercion 介绍,需要的朋友可以参考下
    2022-10-10
  • 用rust 写一个jar包 class冲突检测工具

    用rust 写一个jar包 class冲突检测工具

    这篇文章主要介绍了用rust 写一个jar包 class冲突检测工具 的相关资料,需要的朋友可以参考下
    2023-05-05
  • Rust动态数组Vec基本概念及用法

    Rust动态数组Vec基本概念及用法

    Rust中的Vec是一种动态数组,它可以在运行时自动调整大小,本文主要介绍了Rust动态数组Vec基本概念及用法,具有一定的参考价值,感兴趣的可以了解一下
    2023-12-12
  • Rust生命周期之验证引用有效性与防止悬垂引用方式

    Rust生命周期之验证引用有效性与防止悬垂引用方式

    本文介绍了Rust中生命周期注解的应用,包括防止悬垂引用、在函数中使用泛型生命周期、生命周期省略规则、在结构体中使用生命周期、静态生命周期以及如何将生命周期与泛型和特质约束结合,通过这些机制,Rust在编译时就能捕获内存安全问题
    2025-02-02
  • Rust中的Box<T>之堆上的数据与递归类型详解

    Rust中的Box<T>之堆上的数据与递归类型详解

    本文介绍了Rust中的Box<T>类型,包括其在堆与栈之间的内存分配,性能优势,以及如何利用Box<T>来实现递归类型和处理大小未知类型,通过Box<T>,Rust程序员可以更灵活地管理内存,避免编译时大小不确定的问题,并提高代码的效率和灵活性
    2025-02-02
  • rust程序静态编译的两种方法实例小结

    rust程序静态编译的两种方法实例小结

    这篇文章主要介绍了rust程序静态编译的两种方法总结,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧
    2025-05-05
  • 最新Rust错误处理简介

    最新Rust错误处理简介

    Rust并不像C++一样使用try catch的异常机制来进行错误处理,他将错误分为可恢复错误和不可恢复错误两类,主要使用panic!宏和Result<T,E>类型来进行错误处理,这篇文章主要介绍了Rust错误处理简介,需要的朋友可以参考下
    2022-11-11
  • Rust中的Trait与Trait Bounds详解

    Rust中的Trait与Trait Bounds详解

    Rust中的Trait与TraitBounds通过《西游记》的故事背景进行解释,Trait是一种接口定义机制,用于描述角色的能力;TraitBounds用于限制函数或结构体的参数类型必须实现某些trait;BlanketImplementations可以为所有实现了某类trait的类型提供默认的trait实现
    2025-02-02

最新评论