【Rust自学】13.3. 闭包 Pt.3:使用泛型参数和fn trait来存储闭包

ops/2025/1/20 18:50:26/

13.3.0. 写在正文之前

Rust语言在设计过程中收到了很多语言的启发,而函数式编程对Rust产生了非常显著的影响。函数式编程通常包括通过将函数作为值传递给参数、从其他函数返回它们、将它们分配给变量以供以后执行等等。

在本章中,我们会讨论 Rust 的一些特性,这些特性与许多语言中通常称为函数式的特性相似:

  • 闭包(本文)
  • 迭代器
  • 使用闭包和迭代器改进I/O项目
  • 闭包和迭代器的性能

喜欢的话别忘了点赞、收藏加关注哦(加关注即可阅读全文),对接下来的教程有兴趣的可以关注专栏。谢谢喵!(=・ω・=)
请添加图片描述

13.3.1. 回顾

还记得在 13.1 中的例子吗:

做一个程序,根据人的身体指数等因素生成自定义的运动计划。这个程序的算法逻辑并不是重点,重点是算法在计算过程中会花费几秒的时间。我们的目标是不让用户发生不必要的等待。具体来说就是仅在必要的时候才调用该算法,而且只调用一次。

当时我们修改代码为:

rust">use std::thread;  
use std::time::Duration;  fn main() {  let simulated_user_specified_value = 10;  let simulated_random_number = 7;  generate_workout(  simulated_user_specified_value,  simulated_random_number,  );  
}  fn generate_workout(intensity: u32, random_number: u32) {  let expensive_closure = |num| {  println!("calculating slowly...");  thread::sleep(Duration::from_secs(2));  num  };  if intensity < 25 {  println!("Today, do {} pushups!", expensive_closure(intensity));  println!("Next, do {} situps!", expensive_closure(intensity));} else {  if random_number == 3 {  println!("Take a break today! Remember to stay hydrated!");  } else {  println!("Today, run for {} minutes!", expensive_closure(intensity));  }  }  
}

但是还存在一个问题:这么写没有解决闭包重复调用的问题。intensity小于25的情况下调用了2次闭包。

对于这个问题,一个解决方案是把闭包的值赋给某个本地变量,让这个本地变量被输出语句重复调用。这么写问题的是会造成一些代码的重复。

所以这里更适合使用另一种解决方法:创建一个结构体,它持有闭包及其调用结果。也就是说,在第一次调用闭包后把结果存到闭包里,如果以后还要调用闭包就直接使用存在里面的结果。它的效果是只会在需要结果时才执行该包,而且可缓存结果。

这种模式通常叫 记忆化(memorization)延迟计算(lazy evaluation)

13.3.2. 让结构体持有闭包

根据刚才的解决方法,目前的问题在于如何让结构体持有闭包。

结构体的定义需要知道所有字段的类型,所以如果想在结构体内存储闭包,就必须指明闭包的类型。

每个闭包实例都有自己唯一的匿名类型,即使两个闭包签名完全一样,这两个实例仍然是两个类型。所以存储闭包需要使用泛型以及trait bound(泛型和trait bound的内容在 10.4. trait Pt.2 中有讲,推荐看看这篇)

13.3.3. Fn trait

Fn trait由标准库提供。所有的闭包都至少实现了以下Fn trait之一:

  • Fn
  • FnMut
  • FnOnce

这三个Fn trait间的区别会在下一篇文章讲到。在本例中使用Fn就可以了

知道这些之后就可以修改例子了。首先创建一个结构体:

rust">struct Cache<T: Fn(u32) -> u32>  
{  calculation: T,  value: Option<u32>,  
}
  • 这个结构体有一个泛型参数T,由于它代表的是闭包的类型,它的约束是Fn trait(在本例中使用Fn就可以了),然后参数和返回值是u32,所以写Fn(u32) -> u32
  • 闭包所在的字段是calculation,它的类型就是T
  • 要缓存的值在value字段上,其类型是u32,但是要注意的是不清楚这个值是否已经计算出来并缓存在里面了,所以要用Option类型来包裹,也就是Option<u32>

先在结构体上写一个构造函数用于创建实例:

rust">impl<T: Fn(u32) -> u32> Cache<T> {  fn new(calculation: T) -> Cache<T> {  Cache {  calculation,  value: None,  }  }  
}

这么写看着有点乱,可以用where字句重写一下:

rust">impl<T> Cache<T>   
where   
T: Fn(u32) -> u32  
{  fn new(calculation: T) -> Cache<T> {  Cache {  calculation,  value: None,  }  }  
}

然后,为了实现value有值就取value下的值,valueNone就计算的功能,再写一个函数:

rust">fn value(&mut self, arg: u32) -> u32 {  match self.value {  Some(v) => v,  None => {  let v = (self.calculation)(arg);  self.value = Some(v);  v  }  }  
}

如果实例的value字段有值就返回这个值,没有值就计算出这个值,存储在value字段里再返回。

这部分写好之后,就该把generate_workout的写法改一下,转为使用cache结构体:

rust">fn generate_workout(intensity: u32, random_number: u32) {  let mut expensive_closure = Cache::new(|num|{  println!("calculating slowly...");  thread::sleep(Duration::from_secs(2));  num  });  if intensity < 25 {  println!("Today, do {} pushups!", expensive_closure.value(intensity));  println!("Next, do {} situps!", expensive_closure.value(intensity));  } else {  if random_number == 3 {  println!("Take a break today! Remember to stay hydrated!");  } else {  println!("Today, run for {} minutes!", expensive_closure.value(intensity));  }  }  
}
  • expensive_closure作为Cache结构体的实例,使用new函数把闭包传进去。这里把expensive_closure加上mut设为可变函数是因为后文调用时可能会改变value这个字段的值。
  • 下文所有要使用值的操作都使用value方法来获取。

13.3.4. 使用缓存器实现的限制

这里的Cache字段就是缓存器,用于缓存某个值,但这么写是有限制的。

我把Cache的声明和其方法的代码贴在这里:

rust">struct Cache<T: Fn(u32) -> u32>  
{  calculation: T,  value: Option<u32>,  
}  impl<T> Cache<T>   
where   
T: Fn(u32) -> u32  
{  fn new(calculation: T) -> Cache<T> {  Cache {  calculation,  value: None,  }  }  fn value(&mut self, arg: u32) -> u32 {  match self.value {  Some(v) => v,  None => {  let v = (self.calculation)(arg);  self.value = Some(v);  v  }  }  }  
}

value这个方法总会得到同样的值:如果value字段没有值,那它就会计算出值然后把值存储在value字段里,之后的其他地方使用value就会得到最开始的计算的这个值,不论传进去的参数是什么。

这么说可能有点模糊,那来看个例子:

rust">fn call_with_different_values(){let mut c = Cache::new(|a| a);let v1 = c.value(1);let v2 = c.value(2);
}
  • cCache的一个实例,传进去了一个闭包。

  • let v1 = c.value(1);这一行时原本cvalue字段没有值,这时候传进去个1,value字段就变成Some(1)了(value字段是Option类型)

  • let v2 = c.value(2);这一行时由于value字段原本有值,所以会直接取value字段的1赋给v2,即使这行的value方法的参数与上一行不一样。

如果不想要这样,就得使用HashMap来代替单个的值,把HashMap的key作为value方法传进去的参数args;而值就作为执行闭包的结果。比如说:

rust">struct ForFun<T: Fn(u32) -> u32>  
{  calculation: T,  value: HashMap<u32, Option<u32>>,  
}  impl<T> ForFun<T>  
where  T: Fn(u32) -> u32  
{  fn new(calculation: T) -> ForFun<T> {  ForFun {  calculation,  value: HashMap::new(),  }  }  fn value(&mut self, arg: u32) -> u32 {  match self.value.get(&arg) {  Some(v) => v.unwrap(),  None => {  let v = (self.calculation)(arg);  self.value.insert(arg, Some(v));  v  }  }  }  
}

这个例子中的缓存器只能接受同样的参数类型和返回值类型。如果想让闭包的参数类型和返回值类型不一样,就可以引入两个及以上的泛型参数。比如说:

rust">struct ForFun<T, R>  
where  T: Fn(u32) -> R,  
{  calculation: T,  value: Option<R>,  
}

http://www.ppmy.cn/ops/151733.html

相关文章

《 C++ 点滴漫谈: 二十二 》操作符炼金术:用C++ operator重塑代码美学

摘要 C 的 operator 关键字和操作符重载是语言的核心特性之一&#xff0c;使开发者能够扩展内置操作符以适应自定义类型&#xff0c;从而实现更高效、直观的代码表达。本文全面解析了 operator 关键字的基本概念、支持重载的操作符范围及其使用场景&#xff0c;详细介绍了操作…

vscode——如何让标点总是成对出现

vscode——如何让标点总是成对出现&#xff1a; 打开vscode&#xff0c;在设置中输入editor.autoClosing 将设置参数全部改成always

Java复习第三天

一、代码题 1.爬楼梯 (1)题目 假设你正在爬楼梯。需要n阶你才能到达楼顶。每次你可以爬1或2个台阶。你有多少种不同的方法可以爬到楼顶呢? 示例 1: 输入:n2 输出:2解释:有两种方法可以爬到楼顶。 1阶1阶 2 阶示例 2: 输入:n3 输出:3解释:有三种方法可以爬到楼顶。 1 阶1阶…

Go 语言 select 的实现原理

介绍 select是Go在语言层面提供的I/O多路复用的机制&#xff0c;其专门用来让Goroutine同时等待多个channel是否准备完毕:可读或可写。在Channel状态改变之前&#xff0c;select会一直阻塞当前线程或者goroutine。 特性&#xff1a; case 必须是一个通信操作&#xff0c;主要是…

K8S的探针说明和使用方式

探针概述 探针分类 K8S中 探针&#xff08;Probes&#xff09; 是用于检查容器的健康状况和可用性的机制。探针可以自动判断应用的运行状态&#xff0c;并根据需要重启容器、替换容器或将流量路由到健康的实例。从而确保应用始终处于健康、可用的状态&#xff0c;并帮助自动化…

Rust 错误处理(下)

目录 1、用 Result 处理可恢复的错误 1.1?传播错误的简写&#xff1a;? 运算符 1.2 哪里可以使用 ? 运算符 2、要不要 panic! 2.1?示例、代码原型和测试都非常适合 panic 2.2?当我们比编译器知道更多的情况 2.3?错误处理指导原则 2.4?创建自定义类型进行有效性验…

MySQL表的创建实验

创建并使用数据库mydb6_product 。 mysql> create database mydb6_product; Query OK, 1 row affected (0.01 sec)mysql> use mydb6_product; Database changed 新建employees表。 对于gender&#xff0c;有默认值意味着不为空&#xff0c;在建表时可以选择不写not nul…

会话_JSP_过滤器_监听器_Ajax

第8章 会话_JSP_过滤器_监听器_Ajax 8.1 会话 8.1.1 会话管理概述 1、为什么需要会话管理 HTTP是无状态协议&#xff1a; 无状态就是不保存状态&#xff0c;即无状态协议(stateless)&#xff0c;HTTP协议自身不对请求和响应之间的通信状态进行保存&#xff0c;也就是说&…