Rust中的迭代器的使用:map转换、filter过滤、fold聚合、chain链接
什么是迭代器
Rust中的迭代器是一种强大的工具,它提供了一种灵活、通用的方法来遍历序列。迭代器是实现了Iterator trait的类型,并需要至少实现一个next函数,用于让迭代器指向下一个迭代对象,并返回一个Option用于指示对象是否存在。
fn next (&mut self) -> Option<Self::Item>;
迭代器相比于for循环有一些优势。首先,迭代器提供了一种灵活、通用的方法来迭代序列。它允许你使用各种方法来处理序列中的元素,例如map、filter、fold等。这些方法可以让你更简洁、更清晰地表达你的意图。
此外,迭代器和Rust的所有权系统密切相连。这意味着你可以使用迭代器来安全地处理序列中的元素,而不必担心内存安全问题。
迭代器是Rust的零抽象之一,这意味着迭代器抽象不会引入运行时开销,不会有任何性能上的影响
迭代器可以做什么
Rust中的迭代器可以通过实现Iterator trait来创建,也可以通过调用现有类型的iter方法来获取。例如,Vec提供了一个iter方法,可以返回一个迭代器,用于遍历Vec中的元素。
let v = vec![1, 2, 3];
for i in v.iter() {
println!("{}", i);
}
除了for循环外,迭代器还提供了许多其他有用的方法,例如: 迭代器模式允许你对一个项的序列进行某些处理。Rust中的迭代器提供了一种简洁、高效的方式来处理序列,例如通过使用map、filter、fold等方法来转换、过滤和聚合数据。这些方法通常比手写循环更简洁、更易读,也更容易优化。
map:转换数据。接受一个闭包并为迭代器中的每个元素调用该闭包,然后返回一个新的迭代器,其中包含闭包返回的值。
let v = vec![1, 2, 3];
let v_squared: Vec<i32> = v.iter().map(|x| x * x).collect();
filter:过滤数据。接受一个闭包并为迭代器中的每个元素调用该闭包。如果闭包返回true,则元素将包含在新的迭代器中。
let v = vec![1, 2, 3];
let v_even: Vec<&i32> = v.iter().filter(|x| *x % 2 == 0).collect();
fold:聚合数据。接受一个初始值和一个闭包,并将闭包应用于初始值和迭代器中的每个元素,以生成一个单一的最终值。
let v = vec![1, 2, 3];
let sum: i32 = v.iter().fold(0, |acc, x| acc + x);
chain:该方法是Iteratortrait的一个方法,它允许你将两个迭代器链接在一起,形成一个新的迭代器。这个新的迭代器会先遍历第一个迭代器中的所有元素,然后再遍历第二个迭代器中的所有元素。
例如,你可以使用chain方法将两个数组中的元素链接在一起:
let a = [1, 2, 3];
let b = [4, 5];
let c: Vec<i32> = a.iter().chain(b.iter()).copied().collect();
assert_eq!(c, [1, 2, 3, 4, 5]);
在这个例子中,我们创建了两个数组a和b,然后使用chain方法将它们链接在一起,形成一个新的迭代器。最后,我们使用collect方法将迭代器中的元素收集到一个向量中。
希望这对你有所帮助。 ## 如何创建一个迭代器 要创建一个迭代器,你需要给一个类型实现Iterator trait,并实现next方法。下面是一个例子,它展示了如何在一个斐波那契数列类型上创建迭代器:
struct Fib {
a: i32,
b: i32,
}
impl Iterator for Fib {
type Item = i32;
fn next(&mut self) -> Option<Self::Item> {
let res = self.a;
self.a = self.b;
self.b = res + self.b;
Some(res)
}
}
fn main() {
let fib = Fib { a: 1, b: 1 };
for i in fib.take(10) {
println!("{}", i);
}
}
这个例子中,我们定义了一个Fib结构体,它包含两个字段a和b。然后我们为Fib结构体实现了Iterator trait,并实现了next方法。在next方法中,我们计算出下一个斐波那契数,并返回它。最后,在main函数中,我们创建了一个Fib实例,并使用take方法获取前10个斐波那契数1。
使用迭代器要注意什么
在使用Rust中的迭代器时,有几点需要注意:
- 迭代器是惰性的:迭代器不会立即计算它们的值,而是在调用next方法时才会计算。这意味着你需要调用collect或其他消耗迭代器的方法来获取最终结果,这里有一个详细的例子来解释迭代器的惰性:
fn main() {
let v = vec![1, 2, 3];
let v_iter = v.iter().map(|x| {
println!("Mapping value: {}", x);
x * 2
});
println!("Created iterator");
for val in v_iter {
println!("Got value: {}", val);
}
}
在这个例子中,我们创建了一个迭代器v_iter,它使用map方法将序列中的每个元素乘以2。注意,在创建迭代器时,我们并没有看到任何输出。这是因为迭代器是惰性的,它不会立即计算它们的值。
接下来,我们使用for循环来打印出计算后结果。在这个过程中,我们可以看到输出。这是因为for循环会调用迭代器的next方法来获取下一个值。在调用next方法时,迭代器才会计算它的值。
- 注意所有权和借用:当你使用迭代器时,需要注意所有权和借用规则。例如,如果你想要在迭代器中修改元素,你需要使用iter_mut而不是iter方法。
let mut v = vec![1, 2, 3];
for i in v.iter_mut() {
*i *= 2;
}
- 注意迭代器失效:当你修改了迭代器所指向的集合时,迭代器可能会失效。例如,如果你在遍历Vec的同时向其中添加元素,可能会导致迭代器失效。 下面是一个例子,它展示了如何在遍历Vec的同时向其中添加元素,导致迭代器失效:
fn main() {
let mut v = vec![1, 2, 3];
let mut v_iter = v.iter_mut();
while let Some(val) = v_iter.next() {
println!("Got value: {}", val);
if *val == 2 {
v.push(4);
}
}
}
在这个例子中,我们创建了一个可变迭代器v_iter来遍历Vec。然后我们使用while循环来遍历迭代器。在遍历过程中,当我们遇到值为2的元素时,我们向Vec中添加了一个新元素。
然而,在Rust中,这样的操作是不允许的。当你运行这段代码时,你会得到一个运行时错误,提示你迭代器已经失效。
- 注意性能问题:虽然迭代器通常比手写循环更简洁、更易读,但它们并不总是最快的。如果性能至关重要,你应该测试不同的实现方式,并选择最快的一种。 例如,下面是两个计算1到10的和的例子:
fn main() {
let sum: i32 = (1..11).sum();
println!("{}", sum);
let mut sum = 0;
for i in 1..11 {
sum += i;
}
println!("{}", sum);
}
第一个例子使用了迭代器来计算和,而第二个例子使用了手写循环。在大多数情况下,这两个例子的性能差异并不明显。但是,在某些情况下,手写循环可能会比迭代器更快。单数使用迭代器比使用for循环更简洁,并且保证内存安全
总之,Rust中的迭代器是一种强大的工具,它提供了一种简洁、高效、安全的方式来操作数据。在使用迭代器时,应注意惰性、所有权和借用、迭代器失效和性能问题。 from刘金,转载请注明原文链接。感谢!
- 注意性能问题:虽然迭代器通常比手写循环更简洁、更易读,但它们并不总是最快的。如果性能至关重要,你应该测试不同的实现方式,并选择最快的一种。 例如,下面是两个计算1到10的和的例子:
fn main() {
let sum: i32 = (1..11).sum();
println!("{}", sum);
let mut sum = 0;
for i in 1..11 {
sum += i;
}
println!("{}", sum);
}
复制代码
第一个例子使用了迭代器来计算和,而第二个例子使用了手写循环。在大多数情况下,这两个例子的性能差异并不明显。但是,在某些情况下,手写循环可能会比迭代器更快。单数使用迭代器比使用for循环更简洁,并且保证内存安全
总之,Rust中的迭代器是一种强大的工具,它提供了一种简洁、高效、安全的方式来操作数据。在使用迭代器时,应注意惰性、所有权和借用、迭代器失效和性能问题。from刘金,转载请注明原文链接。感谢!
Rust中的迭代器的使用:map转换、filter过滤、fold聚合、chain链接的更多相关文章
- Python 函数式编程 & Python中的高阶函数map reduce filter 和sorted
1. 函数式编程 1)概念 函数式编程是一种编程模型,他将计算机运算看做是数学中函数的计算,并且避免了状态以及变量的概念.wiki 我们知道,对象是面向对象的第一型,那么函数式编程也是一样,函数是函数 ...
- Rust中的迭代器
和闭包一样,练代码 struct Counter { count: u32, } impl Counter { fn new() -> Counter { Counter {count: } } ...
- javascript中some,every,map,filter是只用和ansyc中的each,eachLimit,map,mapLImit,filter的使用
var t = [1,2,3,4,5]; //some找到数组中第一个符合要求的值后就不在继续执行//用来判断数组中是否存符合要求的值,返回结果true|false//function返回类型为boo ...
- python中lambda、yield、map、filter、reduce的使用
1. 匿名函数lambda python中允许使用lambda关键字定义一个匿名函数.所谓的匿名函数就是说使用一次或者几次之后就不再需要的函数,属于"一次性"函数. #例1:求两数 ...
- 辅助函数和高阶函数 map、filter、reduce
辅助函数和高阶函数 map.filter.reduce: 一.辅助函数:(1-1)响应式函数 (数组更新检测): push() pop() shift() unshift() ...
- JAVA中List转换String,String转换List,Map转换String,String转换Map之间的转换类
<pre name="code" class="java"></pre><pre name="code" cl ...
- 牛客网Java刷题知识点之Java 集合框架的构成、集合框架中的迭代器Iterator、集合框架中的集合接口Collection(List和Set)、集合框架中的Map集合
不多说,直接上干货! 集合框架中包含了大量集合接口.这些接口的实现类和操作它们的算法. 集合容器因为内部的数据结构不同,有多种具体容器. 不断的向上抽取,就形成了集合框架. Map是一次添加一对元素. ...
- 刷完欧拉计划中难度系数为5%的所有63道题,我学会了Rust中的哪些知识点?
我为什么学Rust? 2019年6月18日,Facebook发布了数字货币Libra的技术白皮书,我也第一时间体验了一下它的智能合约编程语言MOVE,发现这个MOVE是用Rust编写的,看来想准确理解 ...
- ES6中的迭代器(Iterator)和生成器(Generator)
前面的话 用循环语句迭代数据时,必须要初始化一个变量来记录每一次迭代在数据集合中的位置,而在许多编程语言中,已经开始通过程序化的方式用迭代器对象返回迭代过程中集合的每一个元素 迭代器的使用可以极大地简 ...
- 【Scala篇】--Scala中集合数组,list,set,map,元祖
一.前述 Scala在常用的集合的类别有数组,List,Set,Map,元祖. 二.具体实现 数组 1.创建数组 new Array[Int](10) 赋值:arr(0) = xxx Array[ ...
随机推荐
- Web_Servlet之间请求转发
Servlet2 @WebServlet(urlPatterns = "/aa") public class JspService extends HttpServlet { pr ...
- GDB 调试 - 正确地加载调试符号文件
一.开发流程 1. 编译可执行文件 1 #include <stdio.h> 2 #include <unistd.h> 3 4 void test() 5 { 6 char ...
- 01 关于HTML基础-构建Web,这些你都知道吗?(很全)
以下均是参考MDN web docs,总结出来的比较重要的知识点,与君共勉.不妥之处,还望大家及时提出! 什么是HTML? 是一种告诉浏览器如何组织页面的标记语言.它由一系列元素组成. HTML元素 ...
- Kubernetes(k8s)pod详解
一.简介 在Kubernetes集群中,Pod是所有业务类型的基础,也是K8S管理的最小单位级,它是一个或多个容器的组合.这些容器共享存储.网络和命名空间,以及如何运行的规范.在Pod中,所有容器都被 ...
- 代码片断:GDI绘制带一定角度的椭圆
//先将DXF文件中的Ellipse 解析到elpList 中 foreach (Ellipse ellipse in elpList) { //定义一个矩形 RectangleF rect = ne ...
- java: javacTask: 源发行版 8 需要目标发行版 1.8
idea同一工作空间中不同工程使用不同的jkd版本.在本地idea同时使用jdk1.7和jdk1.8,不同的java工程使用不同的jdk版本,但是在java代码编译时报错,其报错信息为:[java: ...
- 运行npm报错:无法加载文件 D:\nodejs\node_global\webpack.ps1,因为在此系统上禁止运行脚本
npm报错 在 windows终端输入 vue init webpack app, 创建一个名为 app 的 Vue 项目时报错如下: 无法加载文件 D:\nodejs\node_global\web ...
- 2.面向对象基础-01Java类和对象
写在前面: (1)编程语言的发展(人越来越容易理解): 机器语言 汇编语言 高级语言-面向过程 高级语言-面向对象 (2)面向对象的特点: 封装性 继承性 多态性 01Java类和对象 对象:属性(静 ...
- Oracle-登录的用户名和密码大小写敏感
Oracle-登录的用户名和密码大小写敏感
- 几行代码,把zip文件直接破解
几行代码,把zip文件直接破解,不想讲解了,如果要使用就直接复制吧,讲解挺累的 import itertools import zipfile import os filename = "z ...