闭包
闭包:可以捕获其所在环境的匿名函数
- 是匿名函数
- 可以保存为变量,作为参数
- 可以在一个地方创建闭包,然后再另一个上下文中调用闭包来完成运算
- 可从其定义的作用域捕获值
闭包的定义格式
let var = |变量名:变量类型| -> 返回值类型{ 函数体; 返回值 }
- 如果变量类型可以自动推断出,可以省略
- 返回值类型没有或可以自动推断,可以省略
- 函数体只有一行,括号可以省略
使用闭包捕获环境
闭包可以访问定义它的作用域内的变量,而普通函数则不能
rust
fn main(){
let x = 4;
let equal_to_x = |z| z==x;
let y =4
assert!(equal_to_x(y));
}
- 上面的代码可以看出,闭包可以直接访问x变量
- 如果换成函数定义,就会报错,因为函数不能访问外部的变量
- 但是这个操作会产生内存开销
闭包从所在环境捕获值的三种方式
与函数获得参数的三种方式一样:
- 取得所有权:FnOnce
- 可变借用:FnMut
- 不可变借用:Fn 创建闭包时,通过闭包对环境值的使用,Rust推断出具体使用哪个Trait
- 所有的闭包都实现了FnOnce
- 没有移动捕获变量的实现了FnMut
- 无需可变访问捕获变量的闭包实现了Fn 当指定Fn traitBound之一时,首先应该使用Fn,基于闭包体里的情况,rust会提示是否需要FnMut或FnOnce
迭代器
迭代器模式:对一系列项执行某些任务
- 遍历每个项
- 确定序列(遍历)何时完成 Rust的迭代器:
- 懒惰的:除非调用消费迭代器的方法,否则迭代器本身没有任何效果
几个迭代方法
- iter方法:在不可变引用上创建迭代器
- into_iter方法:创建的迭代器会获得所有权
- iter_mut方法:迭代可变的引用
产生迭代器的方法
- 可以通过链式调用使用多个迭代器适配器来执行复杂的操作,这种调用可读性较高 例如map:
- 接收一个闭包,闭包作于每个元素,
- collect方法:消耗性适配器,把结果收集到一个集合类型中
rust
fn iteratior_sum(){
let v1 = vec![1,2,3];
let v2: Vec<i32> = v1.iter().map(|x|x+1).collect();
println!("{:?}",v2);
}
一个迭代器对象经过过滤,返回结果还是迭代器:
rust
fn shoes_in_my_size(shoes: Vec<shoe>, shoe_size: u32) -> Vec<shoe> {
shoes.into_iter().filter(|s| s.size == shoe_size).collect()
}