rust for循环里的所有权 - into_iter / iter / iter_mut

文章目录

  • [1 遍历对象实质为 .into_iter() 生成的迭代器](#1 遍历对象实质为 .into_iter() 生成的迭代器)
  • [2 避免转移 .iter() / .iter_mut()](#2 避免转移 .iter() / .iter_mut())
  • [3 for循环里自变量为什么不用加mut](#3 for循环里自变量为什么不用加mut)
rust 复制代码
// for循环语法糖
for loop_variable in iterator {
    code()
}
// 解糖
{
    let result = match IntoIterator::into_iter(iterator) {
        mut iter => loop {
            match iter.next() {
                None => break,
                Some(loop_variable) => { code(); },
            };
        },
    };
    result
}

1 遍历对象实质为 .into_iter() 生成的迭代器

fn into_iter(self) -> Self::IntoIter

(注意:这里的IntoIter是关联类型,自定义的类型别名)

遍历未实现Copy trait的对象将发生所有权转移

rust 复制代码
let a = vec![0, 1]; // Vec未实现Copy trait
for i in a {i;}
// a; // 去掉注释;报错,a所有权转移

2 避免转移 .iter() / .iter_mut()

fn iter(&self) -> Iter
fn iter_mut(&mut self) -> IterMut

1)

rust 复制代码
let a = vec![0, 1]; 
for i in a.iter() {i;}
a; 

实际上,for循环可以遍历所有实现IntoIterator trait的对象,而IntoIterator trait必须实现的方法就是into_iter();

Iter和IterMut也都实现了IntoIterator trait,所以可以理解为上述代码解糖后操作的对象是a.iter().into_iter(),而基本库里Iter对象into_iter()方法实现是这样的:

rust 复制代码
impl<I: Iterator> IntoIterator for I {
    type Item = I::Item;
    type IntoIter = I; // 类型IntoIter就是待实现对象的类型,这里也就是Iter

    #[inline]
    fn into_iter(self) -> I {
        self // 将iter()产生的Iter对象原封不动地返回,但之前对象的所有权已经转移
    }
}

.into_iter()a.iter()原封不动地返回,此时for循环遍历的对象实质就是a.iter(),而.iter()传递的是调用者的引用

2)

另一种写法就是直接遍历对象的引用

rust 复制代码
    let a = vec![0, 1]; 
    for i in &a {i;}
    a; 

不可变引用都实现了 Copy trait这里自然不存在所有权转移;

同时,【摘自文档 https://rustwiki.org/zh-CN/std/iter/index.html】如果集合类型 C 提供 iter(),则它通常还为 &C 实现 IntoIterator,而该实现只是调用 iter()。 同样,提供 iter_mut() 的集合 C 通常通过委派给 iter_mut() 来为 &mut C 实现 IntoIterator。

所以,这里实际上操作的对象还是a.iter()

p.s 方法的调用会自动解引用,故&a.iter()和a.iter()等价

3 for循环里自变量为什么不用加mut

从文章开头贴出的代码可以看到,解糖后的代码自变量loop_variable实际只是存在于单次循环中的一个临时变量,可以理解成每次循环都会完成一次let loop_variable = iter.next();的赋值操作, 所以,自变量无需mut也能发生变化;当然,添加了mut也不会报错,只是会多个警告提示。

相关推荐
淡定__0094 分钟前
深入理解 .NET 中的 Task 并行库(TPL)
后端
青韵6 分钟前
AgentScope计划模块剖析与模仿demo
后端
zmzb01038 分钟前
C++课后习题训练记录Day47
开发语言·c++
gregmankiw13 分钟前
Rust错误处理
rust
勇敢牛牛_15 分钟前
【aiway】一个Rust实现的API网关
rust·api网关
文心快码BaiduComate16 分钟前
CCF程序员大会码力全开:AI加速营决赛圆满落幕,大奖花落谁家?
前端·后端·程序员
北极糊的狐20 分钟前
报错java: 找不到符号符号: 类 XxxController位置: 程序包 com.ruoyi.xxx.xxx.service
开发语言·windows·python
独自破碎E37 分钟前
如何用最短替换让字符串变平衡?
java·开发语言·算法·leetcode
老华带你飞42 分钟前
宠物商城销售|基于Java+ vue宠物商城销售管理系统(源码+数据库+文档)
java·开发语言·数据库·vue.js·spring boot·宠物
电饭叔1 小时前
一个构建指定坐标轴在默认点(0,0)的构造方法《python语言程序设计》2018版--第8章17题第2部分
开发语言·笔记·python