0%

#第一阶段

开营前提前刷了下 rustlings,虽然这次练习代码不是最新的 rustlings,但考察内容是一样的。除此之外本次练习还扩充了一些编译和数据结构的题目,数据结构的题目写起来还是有点卡。不说了… LeetCode 走起。

作为想了解操作系统小白一枚,希望能跟完全程吧(🤔)。

第一阶段总结报告

事件0:报名!

因为学长的推荐,正准备自己开始做rcore lab的时候突然在rcore的官方repo里面看到news:

开源操作系统训练营报名!

wow,看到里面的正是自己想要了解学习的内容,一下子打起了12分精神,感觉很切合自己所在的嵌入式方向,并且完美的满足自己想要在更深平台上学习的想法(之前是在stm32的机器上跑过简单的ucOSII 实时操作系统)。

事件1:rust,启动!

感觉自己花在学rust的时间挺长的,主要是想更深入的学习这个语言(正巧大二上学了编译原理),在rustlings上花了不少时间,不想一个个说语法了,只是记得smart_pointers的特性很有意思狠狠的理解了,当然还有所有权(第一次见到在编译阶段去强调这个概念的语言,之前写malloc实验的时候有想过能不能在写语言的时候把内存的管理考虑好),option之类的东西和c++真的很像,前面的智能指针也是c++那一套的东西(有种写cs144的感觉)。范型的使用我就类比之前学java的时候的用法了,让我记忆深刻的还有rust对于错误处理包装成一个enum,居然是个枚举,还有它的宏,也太多了吧(学c的时候确实体会过宏的强大)。

最后10个algorithms花了小半天写完,确实算是对之前的学习合起来应用了一下。

记录的笔记我就留在个人博客上了,因为用的notion写博客,试试推送很方便,所以习惯了:

Rust基础积累—常更

第二阶段总结报告

做实验

从做实验说起,做rcore实验的周期是起伏的,在5.1假期冲刺完成了4个lab,之前4.28号完成第一个lab。但是当时做的策略从最开始先把全部文档(tutorial3.6的)看完再做实验,变为先扫一遍guide文档,然后做lab,遇到不懂的去翻guide文档,这样才把速度提上来,因为当时也只有5.1假期是完整的时间能做rcore,后面被期末考试割得支离破碎,然后接下来每天有时间细读一下文档,再把前面跳过的问答题补上,就完成了这个整个第二阶段的要求。

对于看的每个chapter以及lab,我做了记录,集中在这里:2024春夏OS训练营

谈感受

第一反应是真的觉得学到很多,学校里面讲的内容只有概念,对于代码,涉及到的也只有linux提供的部分系统调用的使用而不是实现。而从rcore这部分的实验,特别是tutorial里面讲的内容,最开始看链接脚本的配置,bss段的清空,我之前是从来没有在完整的计算机平台写这些的,只有在stm32开发板,移植ucOSII实时操作系统的时候,写过一部分,但是单片机需要考虑的内容和完善的一个计算机应该考虑的内容还是相差较远,而且这次是拿rust进行OS开发,对我之前学的rust基础部分是很大提升。想想makefile也用的更熟练了,以前很是不喜欢命令行gdb调试,总要配配vscode的gdb(launch.json)来用用图形化,现在配合dashboard加上写.gdbinit脚本,流畅的使用gdb能够在调试中得到更多信息(不过当然还是想以后找个方法对于这种情况也能上vscode的gdb插件)。

从批处理操作系统开始,到初步的一个分时操作系统,来到ch3的部分,对系统调用的流程更加清晰,cpu硬件和os的配合(特权级别的切换,内核栈、用户栈的切换,sscratch寄存器的使用),系统调用的认识就变成了一个软中断。

感觉难起来的就是地址空间的出现,而且这个就是我特别特别想自己写代码的部分,因为自己嵌入式这边用的单片机(stm32f401re)是没有mmu的,对于课上提到的地址虚拟化这部分的认识,完全是概念上的,只会做题,算页表占用、分配。但是在rcore的实验里面,尤其是读tutorial,riscv的satp CSR用于管理MMU的配置,设置根页表所在物理号,还有我们这里采取的”双页表”地址空间设计,内核和用户程序都各自拥有地址空间,trap的时候也会伴随地址空间切换到内核的,随之而来的在切换部分实现地址平滑过渡的trampoline(跳板页)以及因为只有一个sscratch所以把内核栈设置到用户空间,这些设计细节,远远超过了平时课上的那点概念认识,也让我对原来很陌生,很迷惑的分页机制有了具体的认识。而且这里面rust的代码的书写风格也给了我很深刻的印象,利用rust的生命周期管理,不用时时刻刻都自己去写drop(因为真的很容易就搞忘或者弄混乱了),而是在设计数据结构的抽象的时候,就反映这些依赖关系,交给编译器去放置drop(大部分)。

后面就是引入了进程的完整概念(体现到数据结构),这里对于僵尸进程有一个初步的认识(不过后面加上线程过后这个僵尸进程的完整效果才展现出来),有了进程,我们顺其自然的再加上文件系统,这样从文件中加载进程执行代码,开始有了一个现代分时操作系统的雏形。文件系统虽然是给的一个easy实现(只有根目录一个目录),但是我觉得扩展出其它目录好像也并不困难(因为两者的核心内容是一样的):超级块、索引节点位图、索引节点区域、数据块位图、数据块区域。不过之前UCOSII实时操作系统里面也接触过位图(它的任务调度很依赖于位图),把自己想成计算机,想想给一个文件名,怎么找到它所在的数据块或者如何为它创建一个数据块保存在文件系统里面,就能理解每个部分的作用。

最后就是到线程的引入(前面还有进程通信,不过那部分没看的很仔细),实验实现了一个进程里面线程的死锁检测,这里让我影响最深的还不是算法实现,实现其实很简单,但是测试用例的设计反而让我记忆很深刻,我在想我有时候就需要设计一个死锁的例子,但是总是有时候马上不能想到一个比较好的例子(想出来的都是很简单的而且不一定能真死锁的(有时候执行不产生死锁)),所以分析测试用例的时候还给了我很多思路。

结语

老实说,参加这次的训练营还算是运气,正好在一阶段报名最后一天看到这个训练营,自己本来是听学长介绍,打算做rcore实验来弥补课内动手的不足,结果正好发现这里有个训练营而且会做rcore,于是在看到的第一时间就报名了,直到现在,我依然觉得这是我幸运的一点,没有这种有些紧张的氛围,以及日程安排的指导,我肯定没有这么强的动力在这么短的时间完成。也很感谢和我一起做题的室友,和他们讨论的时候让我学到了更多内容。

第一阶段

我其实在去年的时候就对训练营略有耳闻,但是由于在准备算法竞赛,并且对未来研究生的方向还没有一个明确的规划,所以没有参加。而现在,我基本上已经确定了未来要往体系结构和操作系统方向发展,所以第一时间便报名参加了训练营。

第一阶段学的内容是 Rust。Rust 也是我很喜欢的一门语言,因为他性能高、内存安全、又有良好的包管理器支持,相比于 C、C++,它的语言表达能力更加强大,也有很多好用的语法糖;而相比于 Java、Python 等,它又更为严谨、更贴近硬件。

经过这一段时间的学习,我越学越觉得 Rust 的很多特性其实是为了给他严格的所有权机制打补丁。最明显的就是生命周期了,还有诸如 unsafe 等。刚好最近同时也在一家公司实习做操作系统内核开发,正在使用 C 语言,因此对这两门语言的风格深有所感:

  • C 的原则就是「完全相信使用者」,因此你可以用 C 实现几乎所有操作,非常自由。但是为了安全,必须人为设计一些规范来进行约束。
  • Rust 的原则则是「完全不相信使用者」,所以你会发现 Rust 的很多语法都是为了约束程序员,强迫程序员写出安全的代码。但是,有些时候编译器还是不够聪明,或者说是无法进行判断,因此必须开点绿灯——unsafe

在 Rust 身上可以找到很多为了弥补所有权机制而设计的语法,因此在学习的时候才会觉得 Rust 的语法很复杂。不过这种「语言规定好的规范」对于多人之间的项目合作,特别是开源来说,就是一种优势了。相比于 C 语言项目之间可能存在代码风格相差巨大的情况,Rust 写出来的代码基本上不会有太大的风格差异,这样在参考别人的代码,以及贡献代码的时候就会更为轻松。

第二阶段

第二阶段开始进入操作系统领域的学习了。训练营采用的是清华大学的 rcore 教程,不过相比原版的 rCore-Tutorial-v3 有了很大的变化,删去了很多不必要的任务,任务的指引也更加清晰,教材方面也根据代码做了很多简化。整体上是更容易上手了。

在这几章的学习内容中,我认为最难理解的是虚拟地址页表那一章。一开始我一直困惑于“操作系统到底如何分配内存,如何知道地址是否合法的”,后来经过了解才知道,虚拟地址其实是软件和硬件共同实现的,操作系统需要做的是维护页表,而 cpu 会利用操作系统维护的页表,通过硬件来判断地址是否合法,以及将虚拟地址转换为真实的物理地址。所以整个计算机的发展软硬件是不可分开的。

在完成了第二阶段的学习后,我和队友便着手开始了计算机系统能力大赛的开发。一开始我们是打算基于 arceos 进行开发的,但是了解后才发现 arceos 似乎没有提供一个基础的内核框架,而只是提供了很多的“组件”,而我们此时仅剩下不到两周的时候,根本无法在这么短的时间从头自己写一个内核出来,因此最后还是选择基于训练营第二阶段结束后的 rcore 版本进行开发。

也是得益于训练营阶段对 rcore 代码深入学习的经历,让我们在两周不到的时间里,完全原创地完成了操作系统比赛初赛的赛题。接下来是复赛,我们打算继续基于初赛的代码,加入虚拟文件系统、无栈协程等特性,为复赛做好准备!

第三阶段

第三阶段我选择了项目二《ArceOS 宏内核》,原因是我正在参加今年的计算机系统能力大赛(操作系统内核实现赛道),正好项目二和我们的比赛内容较为贴近,可以相互之间参考与借鉴一下。

ArceOS 是 unikernel。其特点是模块组件化,这样在构建操作系统镜像的时候,就可以选择性地编译模块,从而达到尽可能减少系统镜像体积的目的,对嵌入式等低性能的场景有很大的帮助。

但是 unikernal 的缺点是没有用户态和内核态的区分,也就意味着应用程序的权限管理会对内核的稳定性有很大的隐患。其次,ArceOS 目前只支持一个应用程序,不能作为一个通用的内核来使用。

因此,项目二的目的就是为了让 ArceOS 支持用户态和内核态、支持多应用的并发执行,向宏内核靠拢。

rust第一阶段总结:
第一阶段主要是对rust语言进行一个系统的学习,并进行rustlings习题的完成,其实前一百道题还好,对于我来说更像完形填空一样,对我来说后十道算法题比较有难度
本来算法就比较薄弱,再用不熟悉的语言进行数学,挑战还是相当大的(,不过在第一阶段中,我对于rust的理解有了深一层的概念。相信可以在第二阶段更进一步

#前言

这并不是我第一次参加了,正如我去年结营仪式上说的,我今年又来了。时光匆匆,下半年就要考研了,目前我依然在猛刷各种竞赛,希望今年能更上一步,有所收获!

#第一阶段的总结:
由于之前已经参加过了,因此前100题非常轻松的就刷掉了,今年多出来了10题算法题,我一直很喜欢用rust写算法,写起来是真的爽,llvm的优化也很爽,甚至能硬生生的将$n^2$的算法给优化成$n*log(n)$的级别,非常逆天。这10道算法题我只在第八题上卡了一会,疑似是workflow评测的bug?反正我本地跑的单元测试都能通过,最终还是整个重写了一遍才过,非常奇怪。于是顺利的在开始的第二天早上写完了110道题。
战斗,爽!

一阶段

Begin

rust 有趣捏

Tips

所有权

关于所有权的规则

  1. Rust 中每一个值都被一个变量所拥有,该变量被称为值的所有者
  2. 一个值同时只能被一个变量所拥有,或者说一个值只能拥有一个所有者
  3. 当所有者(变量)离开作用域范围时,这个值将被丢弃(drop)
  4. 当值不是可copy的,简单的= 做的是move
  5. 如果想要多个变量指向一个值, 通过引用
  6. mut引用只能有一个, 不可变引用可以很多,两者不能同时存在

tokio

貌似没涉及,但是觉得好玩捏

  • 底层是线程池+调度器
  • tokio::spawn 出来的task 类似一种协程, 可以被调度器调度
  • 每个task 由诸多future组成, future实际上是一种trait, 实现了它的对象可以被poll, poll它的时候是不阻塞的,如果出现io,返回not ready
  • 每个线程维护task队列, 用完时可以偷取其他线程的
  • 非阻塞io的底层实现是epoll, 当epoll返回时,调度器会选择调度

End

谢谢THU捏

太喜欢rust 了

二阶段

二阶段的收获确实很大,以前也有写过mit的xv6 lab,看过很多os的书籍,但是从rcore中还是学到了很多。我认为rcore的长处是

  • 从bare开始构建,这比xv6强的多,写完xv6,也不知道到底怎么从头写一个os,但是rcore教程详细,循序渐进,彻底让我明白如何从裸机构建一个os。收获很大。
  • 使用rust,比c语言厉害多了,内存安全方面的暂且不提,光是alloc方面我就看的眼花缭乱,先进的语言毕竟还是牛逼啊。

但是我认为,rcore的练习对我不是那么有吸引力,毕竟在os上修修补补已经是常见的lab手段,我觉得文档,构建过程才是真的精髓。如过可以在文档中空一些代码让学生补全,直接参与构建os,相比会更有吸引力。但是我理解这样做难度肯定上升,不太好让基础不劳的学生学习(这样做的话,我这个ddl战士应该也写不完了)。

总而言之,我认为rcore真的很棒,我认为是比肩mit了,感谢写rcore的人士,您们真牛。

日程上,很羞愧,五个lab在三天中赶完,所以感觉也不必要详细记录了,但是可不是我只学了三天,在训练营开始之前我就偶然发现了rcore的文档,基本看了一边,之前也有尝试顺着rcore的教程从头写os(最后只写到第5章。。)感觉还是有点抱歉的,很想参与训练营更多的课程,但是最近学校的事情多,只好做ddl战士了。

一、引言

首先非常有幸参加了本次开源操作系统训练营,感谢诸位讲师的无私奉献和合作平台的支持,提供了一次深入探索操作系统内部机制与Rust编程语言应用的宝贵机会.在近期的学习中,我深入了解了Rust程序设计语言,主要参考了《Rust程序设计语言》和rust圣经以及训练营课程。

第一阶段学习目标

对rust语言初步了解,能使用rust语言完成rustlings练习

学习内容概述

Rust基础语法与特性:详细学习了Rust的基本语法结构,包括变量、数据类型、控制流、函数等。重点掌握了Rust的所有权系统、借用检查器和生命周期等核心特性,这些特性确保了Rust的内存安全性。
并发与异步编程:学习了如何使用线程、协程和异步I/O进行高效并发处理。深入了解了互斥锁、通道等如何用rust进行实现。
模块与包管理:了解了如何创建模块和包,对Cargo管理项目的依赖和构建过程有了初步的了解。

学习体会与收获

1.rust的所有权系统和借用检查器有效地避免了空指针解引用、数据竞争等常见的内存错误,和以往学过的c语言相比减少了因程序员的疏忽而导致的bug
2.rust采用多线程与 async/await 相结合,使用复杂度换取可控性和性能,通过对锁和atomic的了解,为接下来在操作系统中的lab实现打下了坚实基础
3.理解了rust的生命周期机制,有助于避免悬垂引用等内存安全问题,写出安全的代码

问题与不足

1.对理论知识只是粗浅的了解,写练习时仍需翻阅资料
2.知识遗忘速度过快,只是有了全体框架,具体内容往往停留纸上谈兵,无法学以致用
3.做练习只是听从编译器的指挥,无法自主解决问题

接下来的打算

1.在进行第二阶段的学习过程中不断复习学过的知识避免遗忘
2.根据oj题目,使用rust语言完成练习,加深对语言的印象
3.重写rustlings,不再是简单的通过,再完成的同时注释实现题目的思路。

第一阶段:RUST语法练习

variables6.rs

关于const:

Rust的常量必须加类型。那为什么常量不推导呢? 这是因为故意这样设计的,指定类型就不会因为自动推导类型出问题。

常量 - 通过例子学 Rust 中文版

static, const, let 声明变量有什么区别? - Rust语言中文社区

1
2
3
4
5
6
7
8
9
10
11
// variables6.rs
//
// Execute `rustlings hint variables6` or use the `hint` watch subcommand for a
// hint.

// const类型需要声明变量类型

const NUMBER : i32 = 3;
fn main() {
println!("Number {}", NUMBER);
}

primitive_types3.rs

数组(array)的三要素:

  • 长度固定
  • 元素必须有相同的类型
  • 依次线性排列

数组必须要初始化,以下这种写法会报错:

1
2
3
4
5
6
7
8
9
fn main() {
let a:[i32; 100];

if a.len() >= 100 {
println!("Wow, that's a big array!");
} else {
println!("Meh, I eat arrays like that for breakfast.");
}
}

改为这样就编译通过:

1
2
3
4
5
6
7
8
9
fn main() {
let a:[i32; 100] = [0; 100];

if a.len() >= 100 {
println!("Wow, that's a big array!");
} else {
println!("Meh, I eat arrays like that for breakfast.");
}
}

vecs2.rs

iter_mut():

iter_mut() 创建一个可变引用迭代器。当你想要修改集合中的元素时,应使用 iter_mut()。iter_mut() 返回的迭代器将生成集合中每个元素的可变引用。

1
2
3
4
let mut v = vec![1, 2, 3];
for i in v.iter_mut() {
*i += 1;
}

move_semantics2.rs

1
2
3
4
5
6
7
8
// move_semantics2.rs
//
// Expected output:
// vec0 has length 3, with contents `[22, 44, 66]`
// vec1 has length 4, with contents `[22, 44, 66, 88]`
//
// Execute `rustlings hint move_semantics2` or use the `hint` watch subcommand
// for a hint.

方法1:

将vec0的内容clone一份传进函数,然后返回值的所有权交给vec1,此时vec1=[22, 44, 66],vec0=[]

然后再把vec0的内容clone一份传进函数,然后返回值的所有权交给vec0,此时vec1=[22, 44, 66],vec0=[22, 44, 66]

这个时候不管是vec0还是vec1都拥有一片自己的堆上的空间,二者互不相关,因此vec1.push(88)只会改变vec1的值,且vec0的值也还存在

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
fn main() {
let mut vec0 = Vec::new();

let mut vec1 = fill_vec(vec0.clone());
vec0 = fill_vec(vec0.clone());
println!("{} has length {}, with contents: `{:?}`", "vec0", vec0.len(), vec0);

vec1.push(88);

println!("{} has length {}, with contents `{:?}`", "vec1", vec1.len(), vec1);
}

fn fill_vec(vec: Vec<i32>) -> Vec<i32> {
let mut vec = vec;

vec.push(22);
vec.push(44);
vec.push(66);

vec
}

方法2:

首先,创建了vec0的一个可变引用&mut vec0,将这个可变引用传入函数,函数接受一个可变引用类型,然后对其进行操作,也就是操作了vec0指向的那片堆,因此在函数内部,vec0就已经变成了[22, 44, 66]

然后最后返回vec.to_vec(),相当于又创建了一个新的vec,作为vec1绑定的值,因此vec1和vec0又变成了互不相关的

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
fn main() {
let mut vec0 = Vec::new();

let mut vec1 = fill_vec(&mut vec0);

println!("{} has length {}, with contents: `{:?}`", "vec0", vec0.len(), vec0);

vec1.push(88);

println!("{} has length {}, with contents `{:?}`", "vec1", vec1.len(), vec1);
}

fn fill_vec(vec: &mut Vec<i32>) -> Vec<i32> {
let mut vec = vec;

vec.push(22);
vec.push(44);
vec.push(66);

vec.to_vec()
}

move_semantics4.rs

这个的意思就是函数不再接收参数,而是直接在里面新创建一个包含[22, 44, 66]的vector返回

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
// move_semantics4.rs
//
// Refactor this code so that instead of passing `vec0` into the `fill_vec`
// function, the Vector gets created in the function itself and passed back to
// the main function.
//
// Execute `rustlings hint move_semantics4` or use the `hint` watch subcommand
// for a hint.

fn main() {
// let vec0 = Vec::new();

let mut vec1 = fill_vec();

println!("{} has length {} content `{:?}`", "vec1", vec1.len(), vec1);

vec1.push(88);

println!("{} has length {} content `{:?}`", "vec1", vec1.len(), vec1);
}

// `fill_vec()` no longer takes `vec: Vec<i32>` as argument
fn fill_vec() -> Vec<i32> {
let mut vec = Vec::new();

vec.push(22);
vec.push(44);
vec.push(66);

vec
}

structs3.rs

结构体,有几点值得注意:

  1. 初始化实例时,每个字段都需要进行初始化
  2. 初始化时的字段顺序不需要和结构体定义时的顺序一致

需要注意的是,必须要将结构体实例声明为可变的,才能修改其中的字段,Rust 不支持将某个结构体某个字段标记为可变。

结构体方法:

  • Unlike functions, methods are defined within the context of a struct,and their first parameter is always self, which represents the instance of the struct the method is being called on.

  • we still need to use the & in front of the self shorthand to indicate that this method borrows the Self instance, just as we did in rectangle: &Rectangle. Methods can take ownership of self, borrow self immutably, as we’ve done here, or borrow self mutably, just as they can any other parameter.

  • 方法参数里面不止&self:package.get_fees(cents_per_gram)

    比如这个,get_fees是结构体的方法,它在结构体里面是这样定义的:

    1
    2
    3
    4
    fn get_fees(&self, cents_per_gram: i32) -> i32 {
    // Something goes here...
    self.weight_in_grams*cents_per_gram
    }

    也就是说,第二个参数跟在&self后面就好了,在外部调用结构体时只需要传入那个另外的参数

例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
// structs3.rs
//
// Structs contain data, but can also have logic. In this exercise we have
// defined the Package struct and we want to test some logic attached to it.
// Make the code compile and the tests pass!
//
// Execute `rustlings hint structs3` or use the `hint` watch subcommand for a
// hint.

// I AM NOT DONE

#[derive(Debug)]
struct Package {
sender_country: String,
recipient_country: String,
weight_in_grams: i32,
}

impl Package {
fn new(sender_country: String, recipient_country: String, weight_in_grams: i32) -> Package {
if weight_in_grams <= 0 {
panic!("Can not ship a weightless package.")
} else {
Package {
sender_country,
recipient_country,
weight_in_grams,
}
}
}

fn is_international(&self) -> bool {
// Something goes here...
self.sender_country != self.recipient_country
}

fn get_fees(&self, cents_per_gram: i32) -> i32 {
// Something goes here...
self.weight_in_grams*cents_per_gram
}
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
#[should_panic]
fn fail_creating_weightless_package() {
let sender_country = String::from("Spain");
let recipient_country = String::from("Austria");

Package::new(sender_country, recipient_country, -2210);
}

#[test]
fn create_international_package() {
let sender_country = String::from("Spain");
let recipient_country = String::from("Russia");

let package = Package::new(sender_country, recipient_country, 1200);

assert!(package.is_international());
}

#[test]
fn create_local_package() {
let sender_country = String::from("Canada");
let recipient_country = sender_country.clone();

let package = Package::new(sender_country, recipient_country, 1200);

assert!(!package.is_international());
}

#[test]
fn calculate_transport_fees() {
let sender_country = String::from("Spain");
let recipient_country = String::from("Spain");

let cents_per_gram = 3;

let package = Package::new(sender_country, recipient_country, 1500);

assert_eq!(package.get_fees(cents_per_gram), 4500);
assert_eq!(package.get_fees(cents_per_gram * 2), 9000);
}
}

enums2.rs

更为复杂的枚举:

Move:包含了一个匿名结构体

Echo:包含了一个String

ChangeColor:包含了三个整数

Quit:没有关联任何数据

1
2
3
4
5
6
7
enum Message {
// TODO: define the different variants used below
Move { x: i32, y: i32 },
Echo(String),
ChangeColor(i32, i32, i32),
Quit
}

enums3.rs

模式匹配和模式绑定

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
// enums3.rs
//
// Address all the TODOs to make the tests pass!
//
// Execute `rustlings hint enums3` or use the `hint` watch subcommand for a
// hint.

// I AM NOT DONE

enum Message {
// TODO: implement the message variant types based on their usage below
ChangeColor(u8, u8, u8),
Echo(String),
Move(Point),
Quit
}

struct Point {
x: u8,
y: u8,
}

struct State {
color: (u8, u8, u8),
position: Point,
quit: bool,
message: String
}

impl State {
fn change_color(&mut self, color: (u8, u8, u8)) {
self.color = color;
}

fn quit(&mut self) {
self.quit = true;
}

fn echo(&mut self, s: String) { self.message = s }

fn move_position(&mut self, p: Point) {
self.position = p;
}

fn process(&mut self, message: Message) {
// TODO: create a match expression to process the different message
// variants
// Remember: When passing a tuple as a function argument, you'll need
// extra parentheses: fn function((t, u, p, l, e))

// 模式匹配message
match message {
// 结构枚举类型绑定的值,相当于还完成了语句:Message::ChangeColor(r,g,b)=message
Message::ChangeColor(r, g, b) => {
self.change_color((r, g, b));
}
Message::Echo(text) => {
self.echo(text);
},
Message::Move(point) => {
self.move_position(point);
}
Message::Quit => {
self.quit();
}
}
}
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
fn test_match_message_call() {
// 创建了一个结构体变量state,并为其赋值
let mut state = State {
quit: false,
position: Point { x: 0, y: 0 },
color: (0, 0, 0),
message: "hello world".to_string(),
};
// 调用结构体内的方法process,参数为枚举类型,每个类型绑定了一个值
state.process(Message::ChangeColor(255, 0, 255));
state.process(Message::Echo(String::from("hello world")));
state.process(Message::Move(Point { x: 10, y: 15 }));
state.process(Message::Quit);

assert_eq!(state.color, (255, 0, 255));
assert_eq!(state.position.x, 10);
assert_eq!(state.position.y, 15);
assert_eq!(state.quit, true);
assert_eq!(state.message, "hello world");
}
}

strings3.rs

  • 字符串的字面量是切片

    一般写字符串可以这样写:let s = "Hello, world!";

    实际上,s 的类型是 &str,因此你也可以这样声明:let s: &str = "Hello, world!";

  • String转&str:取引用即可

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    fn main() {
    let s = String::from("hello,world!");
    say_hello(&s);
    say_hello(&s[..]);
    say_hello(s.as_str());
    }

    fn say_hello(s: &str) {
    println!("{}",s);
    }
  • &str转String:"hello,world".to_string()或者String::from("hello,world")

  • String的操作(必须把String声明为mut)

    由于 String 是可变字符串,因此只有String 这种字符串可以被操作更改

    • push(),在末尾追加字符;push_str() ,在末尾追加字符串

      这两个方法都是在原有的字符串上追加,并不会返回新的字符串,即返回值是()

    • insert() 方法插入单个字符,insert_str() 方法插入字符串字面量

      也是在原有的字符串上面操作,没有返回值

    • replace该方法可适用于 String&str 类型

      该方法是返回一个新的字符串,而不是操作原来的字符串

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
// strings3.rs
//
// Execute `rustlings hint strings3` or use the `hint` watch subcommand for a
// hint.

// I AM NOT DONE

fn trim_me(input: &str) -> String {
// TODO: Remove whitespace from both ends of a string!
input.trim().to_string()
}

fn compose_me(input: &str) -> String {
// TODO: Add " world!" to the string! There's multiple ways to do this!
let mut ret: String = input.to_string();
ret.push_str(" world!");
// push_str没有返回值,因此单独返回ret
ret
}

fn replace_me(input: &str) -> String {
// TODO: Replace "cars" in the string with "balloons"!
let mut ret: String = input.to_string();
// replace()直接返回新的字符串
ret.replace("cars","balloons")
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
fn trim_a_string() {
assert_eq!(trim_me("Hello! "), "Hello!");
assert_eq!(trim_me(" What's up!"), "What's up!");
assert_eq!(trim_me(" Hola! "), "Hola!");
}

#[test]
fn compose_a_string() {
assert_eq!(compose_me("Hello"), "Hello world!");
assert_eq!(compose_me("Goodbye"), "Goodbye world!");
}

#[test]
fn replace_a_string() {
assert_eq!(replace_me("I think cars are cool"), "I think balloons are cool");
assert_eq!(replace_me("I love to look at cars"), "I love to look at balloons");
}
}

hashmaps2.rs

问题1:

关于hashmap的更新实例中:为什么修改count的值就可以修改hashmap的键值呢?

1
2
3
4
5
6
7
8
9
10
11
12
use std::collections::HashMap;

let text = "hello world wonderful world";

let mut map = HashMap::new();
// 根据空格来切分字符串(英文单词都是通过空格切分)
for word in text.split_whitespace() {
let count = map.entry(word).or_insert(0);
*count += 1;
}

println!("{:?}", map);

解答:

map.entry(word) 返回了一个 Entry 枚举类型的值,该枚举有两个变体:OccupiedVacantEntry 枚举表示 HashMap 中某个键对应的条目。

当调用 or_insert 方法时,如果 word 对应的条目已经存在,则 or_insert 方法会返回该条目的值的可变引用(Occupied 变体),如果该条目不存在,则会在 map 中插入一个新的键值对,然后返回新插入的值的可变引用(Vacant 变体)

问题2:

这道题里面有这样一个语句:

1
2
**let mut basket = get_fruit_basket();
assert_eq!(*basket.get(&Fruit::Apple).unwrap(), 4);**

其中,basket是这样来的:

1
2
3
4
5
6
7
8
fn get_fruit_basket() -> HashMap<Fruit, u32> {
let mut basket = HashMap::<Fruit, u32>::new();
basket.insert(Fruit::Apple, 4);
basket.insert(Fruit::Mango, 2);
basket.insert(Fruit::Lychee, 5);

basket
}

所以,为什么basket已经是一个hashmap了,还要用*解引用呢?

解答:

*解引用解的不是basket,而是basket.get(&Fruit::Apple),因为get会返回一个指向该条目键值的引用

跟这段代码一个道理:

1
2
3
4
5
let mut scores = HashMap::new();

// 查询Yellow对应的值,若不存在则插入新值
let v = scores.entry("Yellow").or_insert(5);
assert_eq!(*v, 5); // 不存在,插入5

or_insert()会返回一个指向该条目键值的引用,因此也需要把它解引用来跟5比较

quiz2.rs

这道题主要注意Append这个命令:迭代器是向量中每个元素的引用(想想这是肯定的,不然在for循环里面所有权都丢失了的话有点太危险),而这种引用默认就是不可变引用,那么使用String类型的push_str()自然就是要报错的,因为这个方法是改变原字符串而不是返回一个新的字符串。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
// quiz2.rs
//
// This is a quiz for the following sections:
// - Strings
// - Vecs
// - Move semantics
// - Modules
// - Enums
//
// Let's build a little machine in the form of a function. As input, we're going
// to give a list of strings and commands. These commands determine what action
// is going to be applied to the string. It can either be:
// - Uppercase the string
// - Trim the string
// - Append "bar" to the string a specified amount of times
// The exact form of this will be:
// - The input is going to be a Vector of a 2-length tuple,
// the first element is the string, the second one is the command.
// - The output element is going to be a Vector of strings.
//
// No hints this time!

pub enum Command {
Uppercase,
Trim,
Append(usize),
}

pub mod my_module {
use super::Command;

// TODO: Complete the function signature!
pub fn transformer(input: Vec<(String, Command)>) -> Vec<String> {
// TODO: Complete the output declaration!
let mut output: Vec<String> = vec![];
for (string, command) in input.iter() {
// TODO: Complete the function body. You can do it!
match command {
Command::Uppercase => {
output.push(string.to_uppercase());
},
Command::Trim => {
output.push(string.trim().to_string());
},
Command::Append(size) => {
let mut appened_str = string.clone();
appened_str.push_str(&"bar".repeat(*size));
output.push(appened_str);
}
}
}
output
}
}

#[cfg(test)]
mod tests {
// TODO: What do we need to import to have `transformer` in scope?
use super::my_module::transformer;
use super::Command;

#[test]
fn it_works() {
let output = transformer(vec![
("hello".into(), Command::Uppercase),
(" all roads lead to rome! ".into(), Command::Trim),
("foo".into(), Command::Append(1)),
("bar".into(), Command::Append(5)),
]);
assert_eq!(output[0], "HELLO");
assert_eq!(output[1], "all roads lead to rome!");
assert_eq!(output[2], "foobar");
assert_eq!(output[3], "barbarbarbarbarbar");
}
}

options2.rs

主要是if let语句和while let语句的使用

首先,Option实际上是一种枚举类型,包含两个枚举成员:

1
2
3
4
enum Option<T> {
Some(T),
None,
}

所以下面代码中的if let Some(word) = optional_target {}实际上就是将optional_target这个枚举类型的值结构到word上,然后进行一个操作

while let语句也是一个道理,由于optional_integers的元素都是Some类型的变量,因此首先要把optional_integers的值解构到Some(integer)上,然后再进行操作,因此出现一个Some包裹一个Some

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
// options2.rs
//
// Execute `rustlings hint options2` or use the `hint` watch subcommand for a
// hint.

#[cfg(test)]
mod tests {
#[test]
fn simple_option() {
let target = "rustlings";
let optional_target = Some(target);

// TODO: Make this an if let statement whose value is "Some" type
if let Some(word) = optional_target {
assert_eq!(word, target);
}
}

#[test]
fn layered_option() {
let range = 10;
let mut optional_integers: Vec<Option<i8>> = vec![None];

for i in 1..(range + 1) {
optional_integers.push(Some(i));
}

let mut cursor = range;

// TODO: make this a while let statement - remember that vector.pop also
// adds another layer of Option<T>. You can stack `Option<T>`s into
// while let and if let.
while let Some(Some(integer)) = optional_integers.pop() {
assert_eq!(integer, cursor);
cursor -= 1;
}

assert_eq!(cursor, 0);
}
}

关于if let的详细讲解

if 和 if let 表达式 - Rust 参考手册 中文版

options3.rs

ref和&:

1
2
3
4
5
let c = 'Q';

// 赋值语句中左边的 `ref` 关键字等价于右边的 `&` 符号。
let ref ref_c1 = c;
let ref_c2 = &c;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
// options3.rs
//
// Execute `rustlings hint options3` or use the `hint` watch subcommand for a
// hint.

// I AM NOT DONE

struct Point {
x: i32,
y: i32,
}

fn main() {
let y: Option<Point> = Some(Point { x: 100, y: 200 });

match y {
Some(ref p) => println!("Co-ordinates are {},{} ", p.x, p.y),
_ => panic!("no match!"),
}
y; // Fix without deleting this line.
}

errors2.rs

关于errors:

  • 关于parse:

    parse返回的是一个Result枚举类型,成功的话将会是Result::Ok(),失败是话是Result::Err,因此想获取parse成功时返回的正确值,需要将返回值unwrap()

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    fn main() {
    let astr = "7";

    let n = astr.parse::<i32>().unwrap();
    println!("{:?}", n);

    let n = astr.parse::<i64>().unwrap();
    println!("{:?}", n);

    let n = astr.parse::<u32>().unwrap();
    println!("{:?}", n);

    let n = astr.parse::<u64>().unwrap();
    println!("{:?}", n);

    let astr = "7.42";
    let n: f32 = astr.parse().unwrap();
    println!("{:?}", n);

    let n: f64 = astr.parse().unwrap();
    println!("{:?}", n);
    }
  • 认识到Result这个枚举类型:[Result](https://rustwiki.org/zh-CN/std/result/enum.Result.html)[Option](https://rustwiki.org/zh-CN/std/option/enum.Option.html) 类型的更丰富的版本,描述的是可能的错误而不是可能的不存在

    结果 Result - 通过例子学 Rust 中文版

而做题的时候这个例子,返回的错误类型是std::num::ParseIntError

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
// errors2.rs
//
// Say we're writing a game where you can buy items with tokens. All items cost
// 5 tokens, and whenever you purchase items there is a processing fee of 1
// token. A player of the game will type in how many items they want to buy, and
// the `total_cost` function will calculate the total cost of the tokens. Since
// the player typed in the quantity, though, we get it as a string-- and they
// might have typed anything, not just numbers!
//
// Right now, this function isn't handling the error case at all (and isn't
// handling the success case properly either). What we want to do is: if we call
// the `parse` function on a string that is not a number, that function will
// return a `ParseIntError`, and in that case, we want to immediately return
// that error from our function and not try to multiply and add.
//
// There are at least two ways to implement this that are both correct-- but one
// is a lot shorter!
//
// Execute `rustlings hint errors2` or use the `hint` watch subcommand for a
// hint.

// I AM NOT DONE

use std::num::ParseIntError;

pub fn total_cost(item_quantity: &str) -> Result<i32, ParseIntError> {
let processing_fee = 1;
let cost_per_item = 5;
let qty = item_quantity.parse::<i32>();
// 如果parse()转换成功,qty将会是一个Result::Ok(qty)
// 如果parse()转换失败,qty将会是一个Result::Err(ParseIntError)
match qty {
Ok(qty) => Ok(qty * cost_per_item + processing_fee),
Err(ParseIntError) => Err(ParseIntError)
}
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
fn item_quantity_is_a_valid_number() {
assert_eq!(total_cost("34"), Ok(171));
}

#[test]
fn item_quantity_is_an_invalid_number() {
assert_eq!(
total_cost("beep boop").unwrap_err().to_string(),
"invalid digit found in string"
);
}
}

还可以用?:

1
2
3
4
5
6
7
8
9
10
11
12
13
pub fn total_cost(item_quantity: &str) -> Result<i32, ParseIntError> {
let processing_fee = 1;
let cost_per_item = 5;
// 如果成功就把Ok里面的值给qty,错误就直接返回错误
let qty = item_quantity.parse::<i32>()?;
// match qty {
// Ok(qty) => Ok(qty * cost_per_item + processing_fee),
// Err(ParseIntError) => Err(ParseIntError)
// }

// 成功就返回费用
Ok(qty * cost_per_item + processing_fee)
}

errors3.rs

在函数里面使用?:

因为?的逻辑是这样的:

如果返回的不是Err,就正常运行并且将Result枚举类型的值结构给cost

但如果返回的是Err,函数就会立刻终止,然后返回一个Result::Err类型的值,因此必须给函数设置返回值

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
// errors3.rs
//
// This is a program that is trying to use a completed version of the
// `total_cost` function from the previous exercise. It's not working though!
// Why not? What should we do to fix it?
//
// Execute `rustlings hint errors3` or use the `hint` watch subcommand for a
// hint.

use std::num::ParseIntError;

fn main() -> Result<(), ParseIntError> {
let mut tokens = 100;
let pretend_user_input = "8";

let cost = total_cost(pretend_user_input)?;

if cost > tokens {
println!("You can't afford that many!");
} else {
tokens -= cost;
println!("You now have {} tokens.", tokens);
}
Result::Ok(())
}

pub fn total_cost(item_quantity: &str) -> Result<i32, ParseIntError> {
let processing_fee = 1;
let cost_per_item = 5;
let qty = item_quantity.parse::<i32>()?;

Ok(qty * cost_per_item + processing_fee)
}

但是在main函数里面返回一个Result::Ok(String::from("implement success!")) 是不合法的,因为main函数通常有两种有效的返回类型:()(表示成功)和Result<(), E>(表示成功或出现错误)。main函数的返回类型必须实现Termination trait,该trait规定了程序的终止行为

errors4.rs

可以像这样用匹配:

1
2
3
4
5
6
7
fn new(value: i64) -> Result<PositiveNonzeroInteger, CreationError> {
match value {
x if x < 0 => Err(CreationError::Negative),
x if x == 0 => Err(CreationError::Zero),
x => Ok(PositiveNonzeroInteger(x as u64)),
}
}

但感觉这里也可以直接if判断:

1
2
3
4
5
6
7
if value > 0 {
Ok(PositiveNonzeroInteger(value as u64))
} else if value < 0 {
Err(CreationError::Negative)
} else {
Err(CreationError::Zero)
}

errors6.rs

转换错误类型:将标准库定义的错误类型转换为自定义的ParsePosNonzeroError

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
// errors6.rs
//
// Using catch-all error types like `Box<dyn error::Error>` isn't recommended
// for library code, where callers might want to make decisions based on the
// error content, instead of printing it out or propagating it further. Here, we
// define a custom error type to make it possible for callers to decide what to
// do next when our function returns an error.
//
// Execute `rustlings hint errors6` or use the `hint` watch subcommand for a
// hint.

use std::num::ParseIntError;

// This is a custom error type that we will be using in `parse_pos_nonzero()`.
#[derive(PartialEq, Debug)]
enum ParsePosNonzeroError {
Creation(CreationError),
ParseInt(ParseIntError),
}

impl ParsePosNonzeroError {
// 将错误类型转换为ParsePosNonzeroError的Creation类型
fn from_creation(err: CreationError) -> ParsePosNonzeroError {
ParsePosNonzeroError::Creation(err)
}
// TODO: add another error conversion function here.
// 将错误类型转换为ParsePosNonzeroError的ParseInt类型
fn from_parseint(err: ParseIntError) -> ParsePosNonzeroError {
ParsePosNonzeroError::ParseInt(err)
}
}

fn parse_pos_nonzero(s: &str) -> Result<PositiveNonzeroInteger, ParsePosNonzeroError> {
// TODO: change this to return an appropriate error instead of panicking
// when `parse()` returns an error.

// 处理parse()出错的情况
let x: i64 = s.parse().map_err(ParsePosNonzeroError::from_parseint)?;

// 处理创建时出错的情况
PositiveNonzeroInteger::new(x).map_err(ParsePosNonzeroError::from_creation)
}

// Don't change anything below this line.

#[derive(PartialEq, Debug)]
struct PositiveNonzeroInteger(u64);

#[derive(PartialEq, Debug)]
enum CreationError {
Negative,
Zero,
}

impl PositiveNonzeroInteger {
fn new(value: i64) -> Result<PositiveNonzeroInteger, CreationError> {
match value {
x if x < 0 => Err(CreationError::Negative),
x if x == 0 => Err(CreationError::Zero),
x => Ok(PositiveNonzeroInteger(x as u64)),
}
}
}

#[cfg(test)]
mod test {
use super::*;

#[test]
fn test_parse_error() {
// We can't construct a ParseIntError, so we have to pattern match.
assert!(matches!(
parse_pos_nonzero("not a number"),
Err(ParsePosNonzeroError::ParseInt(_))
));
}

#[test]
fn test_negative() {
assert_eq!(
parse_pos_nonzero("-555"),
Err(ParsePosNonzeroError::Creation(CreationError::Negative))
);
}

#[test]
fn test_zero() {
assert_eq!(
parse_pos_nonzero("0"),
Err(ParsePosNonzeroError::Creation(CreationError::Zero))
);
}

#[test]
fn test_positive() {
let x = PositiveNonzeroInteger::new(42);
assert!(x.is_ok());
assert_eq!(parse_pos_nonzero("42"), Ok(x.unwrap()));
}
}

关于map_err:

  • Maps a Result<T, E> to Result<T, F> by applying a function to a contained Err value, leaving an Ok value untouched.
  • 也就是不触发错误返回,而是等待下一步对这个错误的处理

代码:

1
2
3
4
5
// 处理parse()出错的情况
let x: i64 = s.parse().map_err(ParsePosNonzeroError::from_parseint)?;

// 处理创建时出错的情况
PositiveNonzeroInteger::new(x).map_err(ParsePosNonzeroError::from_creation)
  • 首先s.parse()尝试将s进行转换:
    • 如果出错的话,map_err()就会返回ParsePosNonzeroError::from_parseint 这个自定义的错误类型然后直接返回,而不是ParseIntError
    • 如果没出错,map_err() 就会返回一个Result::Ok(value)作为s转换后的值——因此后面一定要加?,不然出错的情况确实可以正常返回,但是没出错时Result类型的值是无法与i64类型的x绑定的,而?可以将正常返回的值直接unwrap
  • 然后x被成功绑定一个i64类型的整数,就需要判断x是否满足非负数,因此创建一个PositiveNonzeroInteger 类型的结构体,x的值作为结构体的值,然后调用结构体方法new()
    • 如果new成功返回Ok(PositiveNonzeroInteger(x as u64)),那么map_err()就返回这个Result::Ok类型的值
    • 如果new出错了返回了一个Result::Err类型的值,那么map_err()就返回ParsePosNonzeroError::from_creation 这个自定义的错误类型然后直接返回,而不是CreationError

generics2.rs

使用泛型

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
// generics2.rs
//
// This powerful wrapper provides the ability to store a positive integer value.
// Rewrite it using generics so that it supports wrapping ANY type.
//
// Execute `rustlings hint generics2` or use the `hint` watch subcommand for a
// hint.

pub struct Wrapper<T> {
value: T,
}

impl<T> Wrapper<T> {
pub fn new(value: T) -> Self {
Wrapper { value }
}
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
fn store_u32_in_wrapper() {
assert_eq!(Wrapper::<i32>::new(42).value, 42);
}

#[test]
fn store_str_in_wrapper() {
assert_eq!(Wrapper::<&str>::new("Foo").value, "Foo");
}
}

traits5.rs

特征的多重约束

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
// traits5.rs
//
// Your task is to replace the '??' sections so the code compiles.
//
// Don't change any line other than the marked one.
//
// Execute `rustlings hint traits5` or use the `hint` watch subcommand for a
// hint.

pub trait SomeTrait {
fn some_function(&self) -> bool {
true
}
}

pub trait OtherTrait {
fn other_function(&self) -> bool {
true
}
}

struct SomeStruct {}
struct OtherStruct {}

impl SomeTrait for SomeStruct {}
impl OtherTrait for SomeStruct {}
impl SomeTrait for OtherStruct {}
impl OtherTrait for OtherStruct {}

// YOU MAY ONLY CHANGE THE NEXT LINE
fn some_func(item: (impl SomeTrait + OtherTrait)) -> bool {
item.some_function() && item.other_function()
}

fn main() {
some_func(SomeStruct {});
some_func(OtherStruct {});
}

quiz3.rs

关于格式化字符串format!:并不是所有的泛型都可以用这个函数,必须要有Display特征的泛型才可以,因此使用T: Display这个特征约束

pub fn notify<T: Summary>(item1: &T, item2: &T) {}

像这段代码的意思就是:约束T必须具有特征Summary,且item1和item2都是T类型的引用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
// quiz3.rs
//
// This quiz tests:
// - Generics
// - Traits
//
// An imaginary magical school has a new report card generation system written
// in Rust! Currently the system only supports creating report cards where the
// student's grade is represented numerically (e.g. 1.0 -> 5.5). However, the
// school also issues alphabetical grades (A+ -> F-) and needs to be able to
// print both types of report card!
//
// Make the necessary code changes in the struct ReportCard and the impl block
// to support alphabetical report cards. Change the Grade in the second test to
// "A+" to show that your changes allow alphabetical grades.
//
// Execute `rustlings hint quiz3` or use the `hint` watch subcommand for a hint.

// I AM NOT DONE

use std::fmt::Display;

// 因为成绩既有字符串又有浮点数,因此定义grade为一个泛型
pub struct ReportCard<T> {
pub grade: T,
pub student_name: String,
pub student_age: u8,
}
// 这里规定T必须是具有Display特征的类型
impl<T: Display> ReportCard<T> {
pub fn print(&self) -> String {
format!("{} ({}) - achieved a grade of {}",
&(*self).student_name, &self.student_age, &self.grade)
// 这里访问self字段时隐式解引用,但你要手动解引用也可以
}
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
fn generate_numeric_report_card() {
let report_card = ReportCard {
grade: 2.1,
student_name: "Tom Wriggle".to_string(),
student_age: 12,
};
assert_eq!(
report_card.print(),
"Tom Wriggle (12) - achieved a grade of 2.1"
);
}

#[test]
fn generate_alphabetic_report_card() {
// TODO: Make sure to change the grade here after you finish the exercise.
let report_card = ReportCard {
grade: String::from("A+"),
student_name: "Gary Plotter".to_string(),
student_age: 11,
};
assert_eq!(
report_card.print(),
"Gary Plotter (11) - achieved a grade of A+"
);
}
}

lifetimes3.rs

关于结构体的生命周期:

不仅仅函数具有生命周期,结构体其实也有这个概念,只不过我们之前对结构体的使用都停留在非引用类型字段上。细心的同学应该能回想起来,之前为什么不在结构体中使用字符串字面量或者字符串切片,而是统一使用 String 类型?原因很简单,后者在结构体初始化时,只要转移所有权即可,而前者,抱歉,它们是引用,它们不能为所欲为。

既然之前已经理解了生命周期,那么意味着在结构体中使用引用也变得可能:只要为结构体中的每一个引用标注上生命周期即可:

1
2
3
4
5
6
7
8
9
10
11
struct ImportantExcerpt<'a> {
part: &'a str,
}

fn main() {
let novel = String::from("Call me Ishmael. Some years ago...");
let first_sentence = novel.split('.').next().expect("Could not find a '.'");
let i = ImportantExcerpt {
part: first_sentence,
};
}

结构体 ImportantExcerpt 所引用的字符串 str 生命周期需要大于等于该结构体的生命周期

iterators2.rs

迭代器:

  • 迭代器是会被消耗的:

    当使用了next(),迭代器就会被消耗,相当于取出了那一个元素,那个元素其实已经不在迭代器里面了,因此最开始我直接将first转换为大写后直接返回了迭代器剩余的元素,这会导致返回的字符串没有预期的第一个字符

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    // Step 1.
    // Complete the `capitalize_first` function.
    // "hello" -> "Hello"
    pub fn capitalize_first(input: &str) -> String {
    let mut c = input.chars();
    match c.next() {
    None => String::new(),
    Some(first) => {
    let mut first_upp: String = first.to_uppercase().collect();
    let chars_str: String = c.collect();
    first_upp.push_str(&chars_str);
    first_upp
    }
    }
    }

iterators4.rs

使用迭代器处理元素序列 - Rust 程序设计语言 中文版

用迭代器实现递归:使用消费者适配器的方法

1
2
3
4
5
6
7
8
9
10
11
12
pub fn factorial(num: u64) -> u64 {
// Complete this function to return the factorial of num
// Do not use:
// - return
// Try not to use:
// - imperative style loops (for, while)
// - additional variables
// For an extra challenge, don't use:
// - recursion
// Execute `rustlings hint iterators4` for hints.
(1..=num).product()
}

在 Rust 中,(1..=num) 是一个范围(Range)表达式,表示从1到num(包括num本身)的一个范围。这个范围实际上是一个迭代器,它可以生成从1到num的所有数字序列。

在这种情况下,我们可以通过 (1..=num) 创建一个包含从1到num的数字序列的迭代器。然后,我们调用迭代器的 product() 方法,这个方法将迭代器中的所有元素相乘,得到它们的乘积作为结果。

因此,(1..=num).product() 这个语句的意思是:先生成从1到num的数字序列迭代器,然后计算这个序列中所有数字的乘积,最终得到阶乘的结果。

iterators5.rs

关于闭包和迭代器方法的使用

不用循环,找出map里面有多少个键值为value的元素

1
2
3
4
5
fn count_iterator(map: &HashMap<String, Progress>, value: Progress) -> usize {
// map is a hashmap with String keys and Progress values.
// map = { "variables1": Complete, "from_str": None, ... }
map.values().filter(|progress| **progress == value).count()
}
  • .values():使用了HashMap的value方法,获取了一个包含所有键值的迭代器。

    pub fn [values](https://rustwiki.org/zh-CN/std/collections/struct.HashMap.html#method.values)(&self) -> [Values](https://rustwiki.org/zh-CN/std/collections/hash_map/struct.Values.html)<'_, K, V>一个以任意顺序访问所有值的迭代器。 迭代器元素类型为 &'a V ,即对键值的引用

  • .filter()创建一个迭代器,该迭代器使用闭包确定是否应产生元素。给定一个元素,闭包必须返回 truefalse。返回的迭代器将仅生成闭包为其返回 true 的元素。

  • .count() :消耗迭代器,计算迭代次数并返回它。此方法将反复调用 [next](https://rustwiki.org/zh-CN/std/iter/trait.Iterator.html#tymethod.next),直到遇到 [None](https://rustwiki.org/zh-CN/std/option/enum.Option.html#variant.None),并返回它看到 [Some](https://rustwiki.org/zh-CN/std/option/enum.Option.html#variant.Some) 的次数。 请注意,即使迭代器没有任何元素,也必须至少调用一次 [next](https://rustwiki.org/zh-CN/std/iter/trait.Iterator.html#tymethod.next)

返回一个所有哈希表里面键值为指定状态的数量和:

1
2
3
4
5
6
fn count_collection_iterator(collection: &[HashMap<String, Progress>], value: Progress) -> usize {
// collection is a slice of hashmaps.
// collection = [{ "variables1": Complete, "from_str": None, ... },
// { "variables2": Complete, ... }, ... ]
collection.iter().map(|map| count_iterator(map, value)).sum()
}
  • .map() :Iterator特征的函数,获取一个闭包并创建一个迭代器,该迭代器在每个元素上调用该闭包。
  • 首先创建collection的迭代器,再使用map使得迭代器上的每个元素都调用count_iterator(),对于每个哈希表都计算键值为progress的个数
  • .sum() :将迭代器里面的每个元素相加,所以这么写也是对的:
1
2
let iter = collection.iter().map(|map| count_iterator(map, value));
iter.sum()

box1.rs——智能指针

Box堆对象分配 - Rust语言圣经(Rust Course)

1
2
3
4
5
6
7
8
9
10
11
12
13
// 使用Box来定义Cons,将List存到堆上,那么List的大小就固定了
pub enum List {
Cons(i32, Box<List>),
Nil,
}

pub fn create_empty_list() -> List {
List::Nil
}

pub fn create_non_empty_list() -> List {
List::Cons(1, Box::new(List::Cons(2, Box::new(List::Nil))))
}

threads1.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
fn main() {
// 创建handles,用以存储后面10个线程的句柄
let mut handles = vec![];

// 通过循环创建了10个线程,每个线程会执行一个闭包
for i in 0..10 {
// 使用move拿走当前i的所有权
handles.push(thread::spawn(move || {
let start = Instant::now();
thread::sleep(Duration::from_millis(250));

// 250ms后输出,因为i的所有权被拿到了,因此可以访问i
println!("thread {} is complete", i);
start.elapsed().as_millis()
}));
}

// 创建results,用以存储10个线程的执行时间
let mut results: Vec<u128> = vec![];
for handle in handles {
// TODO: a struct is returned from thread::spawn, can you use it?
// 等待每个线程执行完成再结束主线程
results.push(handle.join().unwrap());
}

// 输出时间
for (i, result) in results.into_iter().enumerate() {
println!("thread {} took {}ms", i, result);
}
}
  • 线程返回值是如何得到的?

    handle.join().unwrap() 中,join() 方法会等待线程执行完成并获取线程的返回值,即每个线程的执行时间(以毫秒为单位),然后通过 unwrap() 方法将其取出并存储在 results 向量中。

    Untitled

  • 线程编号和时间是如何输出的?

    results.into_iter().enumerate(): into_iter() 方法将 results 向量转换为一个拥有所有权的迭代器,enumerate() 方法对迭代器进行索引迭代,返回一个元组 (index, value),其中 index 表示元素在迭代器中的索引,value 表示元素的值。

threads3.rs——多个发送者

为什么这里需要克隆发送方?

  • 因为thread使用了闭包关键字move,这会使得在创建第一个线程时,tx的所有权已经被移到了第一个线程里面,第二个线程还想用tx发送信息自然是做不到的
  • 因此克隆一个tx,第二个线程就可以使用了
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
fn send_tx(q: Queue, tx: mpsc::Sender<u32>) -> () {
let qc = Arc::new(q);
let qc1 = Arc::clone(&qc);
let qc2 = Arc::clone(&qc);
// 克隆发送方
let tx_clone = tx.clone();

thread::spawn(move || {
for val in &qc1.first_half {
println!("sending {:?}", val);
tx.send(*val).unwrap();
thread::sleep(Duration::from_secs(1));
}
});

thread::spawn(move || {
for val in &qc2.second_half {
println!("sending {:?}", val);
// 使用克隆的发送方
tx_clone.send(*val).unwrap();
thread::sleep(Duration::from_secs(1));
}
});
}

macros3.rs

  • 使用macro_rules!来定义宏
  • 宏的定义必须在调用之前
  • 通过在 my_macro 宏定义前加上 #[macro_export] 属性,使得宏可以在模块外部使用
1
2
3
4
5
6
7
8
9
10
11
12
pub mod macros {
#[macro_export]
macro_rules! my_macro {
() => {
println!("Check out my macro!");
};
}
}

fn main() {
my_macro!();
}

macros4.rs

模式匹配:

  1. $() 中包含的是模式 $x:expr,该模式中的 expr 表示会匹配任何 Rust 表达式,并给予该模式一个名称 $x
  2. 因此 $x 模式可以跟整数 1 进行匹配,也可以跟字符串 “hello” 进行匹配: vec!["hello", "world"]
  3. $() 之后的逗号,意味着12 之间可以使用逗号进行分割,也意味着 3 既可以没有逗号,也可以有逗号:vec![1, 2, 3,]
  4. *说明之前的模式可以出现零次也可以任意次,这里出现了三次

匹配一次:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
#[rustfmt::skip]
macro_rules! my_macro {
() => {
println!("Check out my macro!");
};
($val:expr) => {
println!("Look at this other macro: {}", $val);
};
}

fn main() {
my_macro!();
my_macro!(7777);
}

匹配多次:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
#[rustfmt::skip]
macro_rules! my_macro {
() => {
println!("Check out my macro!");
};
($($val:expr),*) => {
$(
println!("Look at this other macro: {}", $val);
)*
};
}

fn main() {
my_macro!();
my_macro!(7777, 999, "hello");
}

tests5.rs——裸指针和unsafe

  • 在裸指针 *const T 中,这里的 * 只是类型名称的一部分,并没有解引用的含义

  • 下面的代码基于值的引用同时创建了可变和不可变的裸指针,创建裸指针是安全的行为,而解引用裸指针才是不安全的行为 :

    1
    2
    3
    4
    let mut num = 5;

    let r1 = &num as *const i32; // 不可变裸指针
    let r2 = &mut num as *mut i32; // 可变裸指针

algorithm1——合并链表

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
pub fn merge(list_a:LinkedList<T>,list_b:LinkedList<T>) -> Self
{
//TODO
let mut merged_list = LinkedList::new();
// 获取链表的开始节点
let mut node_a = list_a.start;
let mut node_b = list_b.start;

while node_a.is_some() || node_b.is_some() {
// 指针解引用,其中一个指针有可能为None
// 但是不获得val的所有权
let a_val = node_a.map(|ptr| unsafe{ &(*ptr.as_ptr()).val });
let b_val = node_b.map(|ptr| unsafe{ &(*ptr.as_ptr()).val });

// 比较大小
match (a_val, b_val) {
// 两个都非空
(Some(a), Some(b)) => {
if a < b {
merged_list.add(a.clone());
// 指针解引用且获得指针内容的所有权
node_a = unsafe{ (*node_a.unwrap().as_ptr()).next };
} else {
merged_list.add(b.clone());
node_b = unsafe{ (*node_b.unwrap().as_ptr()).next };
}
},
// a已经空了,直接把b剩下的元素全部加进链表
(None, Some(b)) => {
merged_list.add(b.clone());
node_b = unsafe{ (*node_b.unwrap().as_ptr()).next };
},
(Some(a), None) => {
merged_list.add(a.clone());
node_a = unsafe{ (*node_a.unwrap().as_ptr()).next };
},
(None, None) => {
break
}
}
}

merged_list
}

algorithm2——链表反转

  • 因为用到了clone,因此必须限定T是具有Clone特征的泛型

  • 方法一

    • 使用std::mem::replace(self, reversed_list); 交换新链表和self的所有权
  • 方法二

    • 直接将新链表的所有权交给self
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    impl<T: Clone> LinkedList<T> {
    pub fn reverse(&mut self){
    let mut reversed_list = LinkedList::new();
    // 获取链表的末尾节点
    let mut current_node = self.end;
    while current_node.is_some() {
    // 获取当前节点值的引用
    let value = current_node.map(|ptr| unsafe{ &(*ptr.as_ptr()).val });
    match value {
    Some(x) => {
    // 新链表加入x
    reversed_list.add(x.clone());
    // 指向上一个节点
    current_node = unsafe{ (*current_node.unwrap().as_ptr()).prev };
    }
    None => break
    }
    }
    // 交换新链表和原链表的所有权
    // std::mem::replace(self, reversed_list);
    *self = reversed_list;
    }
    }
  • 其中while循环那段可以这样改:

    1
    2
    3
    4
    5
    while let Some(node_ptr) = current_node {
    let value = unsafe { &(*node_ptr.as_ptr()).val }; // 因为node_ptr肯定是Some类型的,其值一定存在
    reversed_list.add(value.clone());
    current_node = unsafe { (*node_ptr.as_ptr()).prev };
    }

algorithm4——二叉查找树

  • 因为要递归实现插入和查找,所以应该将search和insert实现为TreeNode的方法,所以TreeNode方法的实现应该要写在最前面

  • 二叉树节点的方法:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    // 此时self是二叉树里面的一个节点
    fn insert(&mut self, value: T) {
    match value.cmp(&self.value) {
    // 要插入的节点小于当前节点
    Ordering::Less => {
    // 应该把它往左边递归
    match &mut self.left {
    Some(left) => {
    // 如果左指针有值,就继续递归
    // 左指针是一个Box
    (*left).insert(value);
    }
    _ => {
    // 如果左指针指向None,就让它指向新建节点
    self.left = Some(Box::new(TreeNode::new(value)));
    }
    }
    }
    Ordering::Greater => {
    // 应该把它往右边递归
    match &mut self.right {
    Some(right) => {
    // 如果右指针有值,就继续递归
    // 右指针是一个Box
    (*right).insert(value);
    }
    _ => {
    // 如果右指针指向None,就让它指向新建节点
    self.right = Some(Box::new(TreeNode::new(value)));
    }
    }
    }
    _ => {} // 相等时不插入新节点
    }
    }
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    fn search(&self, value: T) -> bool {
    match value.cmp(&self.value) {
    Ordering::Less => {
    // 说明应该往当前节点的左边找
    match &self.left {
    Some(left) => { return left.search(value); }
    _ => { return false; }
    }
    }
    Ordering::Greater => {
    // 说明应该往当前节点的右边找
    match &self.right {
    Some(right) => { return right.search(value); }
    _ => { return false; }
    }
    }
    _ => { return true; }
    }
    }
  • 二叉树的方法

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    // Insert a value into the BST
    fn insert(&mut self, value: T) {
    // 直接使用节点的插入方法
    match &mut self.root {
    Some(root_node) => {
    root_node.insert(value);
    }
    _ => {
    self.root = Some(Box::new(TreeNode::new(value)));
    }
    }
    }
    1
    2
    3
    4
    5
    6
    7
    // Search for a value in the BST
    fn search(&self, value: T) -> bool {
    match &self.root {
    Some(root_node) => { root_node.search(value) }
    _ => { false }
    }
    }

    algorithm7——用Vec实现栈

    1
    2
    3
    4
    5
    6
    7
    // 坑:size记得-=1
    fn pop(&mut self) -> Option<T> {
    if self.size > 0 {
    self.size -= 1;
    }
    self.data.pop()
    }
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    fn bracket_match(bracket: &str) -> bool
    {
    let mut stack = Stack::new();
    // 遍历字符串里面的字符
    for chr in bracket.chars() {
    // 扫描到左括号就入栈
    if chr == '(' || chr == '{' || chr == '[' {
    stack.push(chr);
    }
    if chr == ')' || chr == '}' || chr == ']' {
    if stack.is_empty() {
    return false;
    }
    if let Some(stack_top) = stack.pop() {
    if chr == ')' {
    if stack_top != '(' { return false; }
    } else if chr == '}' {
    if stack_top != '{' { return false; }
    } else if chr == ']' {
    if stack_top != '[' { return false; }
    }
    }
    }
    }
    stack.is_empty()
    }

    algorithm8——两个队列实现栈

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    // 出队,出第一个元素
    pub fn dequeue(&mut self) -> Result<T, &str> {
    if !self.elements.is_empty() {
    Ok(self.elements.remove(0usize))
    } else {
    Err("Stack is empty") // 坑:记得把原来的“Queue”改成“Stack”
    }
    }

    // 看队头的元素的值
    pub fn peek(&self) -> Result<&T, &str> {
    match self.elements.first() {
    Some(value) => Ok(value),
    None => Err("Stack is empty"),
    }
    }
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    pub fn push(&mut self, elem: T) {
    self.q1.enqueue(elem); // 入队列q1
    }

    pub fn pop(&mut self) -> Result<T, &str> {
    // 将q1里面的除最后一个元素依次出栈存到q2
    for n in 1..self.q1.size() {
    if let Ok(value) = self.q1.dequeue() {
    self.q2.enqueue(value);
    }
    }
    std::mem::swap(&mut self.q1, &mut self.q2); // 交换q1、q2所有权
    self.q2.dequeue() // q2负责出队
    }

    pub fn is_empty(&self) -> bool {
    self.q1.is_empty() && self.q2.is_empty()
    }

    algorithm9——堆排序

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    impl<T> Heap<T>
    where
    T: Default + std::cmp::PartialOrd
    {
    pub fn new(comparator: fn(&T, &T) -> bool) -> Self {
    Self {
    count: 0,
    items: vec![T::default()], // 从下标为1的地方开始存堆
    comparator,
    }
    }

    pub fn len(&self) -> usize {
    self.count
    }

    pub fn is_empty(&self) -> bool {
    self.len() == 0
    }

    pub fn add(&mut self, value: T) {
    self.count += 1; // index从1开始
    self.items.push(value); // 先在堆的最后添加一个节点
    let mut index = self.count;

    while index > 1 {
    let parent_idx = self.parent_idx(index); // 获取最后一个元素的父元素
    if (self.comparator)(&self.items[index], &self.items[parent_idx]) {
    self.items.swap(index, parent_idx); // 将它和父元素交换
    index = parent_idx; // 发生交换后index移动到父元素的位置
    } else {
    break;
    }
    }
    }

    fn parent_idx(&self, idx: usize) -> usize {
    idx / 2
    }

    // 判断idx是否有孩子
    fn children_present(&self, idx: usize) -> bool {
    self.left_child_idx(idx) <= self.count
    }

    fn left_child_idx(&self, idx: usize) -> usize {
    idx * 2
    }

    fn right_child_idx(&self, idx: usize) -> usize {
    self.left_child_idx(idx) + 1
    }

    // 按堆规定的顺序返回两个孩子中应该在前面的一个的索引值
    fn smallest_child_idx(&self, idx: usize) -> usize {
    let left = self.left_child_idx(idx);
    let right = self.right_child_idx(idx);
    if right > self.count {
    left
    } else if (self.comparator)(&self.items[left], &self.items[right]) {
    left
    } else {
    right
    }
    }
    }
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    fn next(&mut self) -> Option<T> {
    if self.count == 0 {
    return None;
    }
    // 相当于堆排序
    let ret = self.items.swap_remove(1); // 将堆顶元素输出,堆底元素升上来
    self.count -= 1;
    // 再次调整堆为大根堆或小根堆
    let mut index = 1;
    // 当当前元素有子元素时
    while self.children_present(index) {
    let child_index = self.smallest_child_idx(index);
    if (self.comparator)(&self.items[child_index], &self.items[index]) {
    self.items.swap(child_index, index);
    index = child_index;
    }
    else {
    break; // 如果不break会进入死循环
    }
    }

    Some(ret)
    }

    algorithm10——图

    用到了get_mut()

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    // 无向图结构的邻接表,用HashMap来存储,键是String,键值是一个向量,其中i32可能表示边长
    pub struct UndirectedGraph {
    adjacency_table: HashMap<String, Vec<(String, i32)>>,
    }
    impl Graph for UndirectedGraph {
    fn new() -> UndirectedGraph {
    UndirectedGraph {
    adjacency_table: HashMap::new(),
    }
    }
    // 获取邻接表的可变引用
    fn adjacency_table_mutable(&mut self) -> &mut HashMap<String, Vec<(String, i32)>> {
    &mut self.adjacency_table
    }
    // 获取邻接表的引用
    fn adjacency_table(&self) -> &HashMap<String, Vec<(String, i32)>> {
    &self.adjacency_table
    }
    // 添加边
    fn add_edge(&mut self, edge: (&str, &str, i32)) {
    // 要在三元组的第一个str和第二个str之间加边
    let (vertice1, vertice2, weight) = edge;

    // 如果第一个点查询得到
    if let Some(adj1) = self.adjacency_table_mutable().get_mut(&String::from(vertice1)) {
    adj1.push((String::from(vertice2), weight));
    } else {
    self.add_node(vertice1);
    if let Some(new_adj1) = self.adjacency_table_mutable().get_mut(vertice1) {
    new_adj1.push((String::from(vertice2), weight));
    }
    }

    // 如果第二个点查询得到
    if let Some(adj2) = self.adjacency_table_mutable().get_mut(vertice2) {
    adj2.push((String::from(vertice1), weight));
    } else {
    self.add_node(vertice2);
    if let Some(new_adj2) = self.adjacency_table_mutable().get_mut(&String::from(vertice2)) {
    new_adj2.push((String::from(vertice1), weight));
    }
    }
    }
    }
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    // 添加成功返回true,添加失败返回false
    fn add_node(&mut self, node: &str) -> bool {
    if self.contains(node) {
    return false;
    }
    else {
    self.adjacency_table_mutable().insert(node.to_string(), Vec::new());
    return true;
    }
    }
    // 在结构体方法中实现了
    fn add_edge(&mut self, edge: (&str, &str, i32));

第一阶段 - Rust学习

前言小记

唠嗑一番:在此次开源操作系统集训营之前,我实际上已经参加过若干次了。最早的时候,是我对象在2022年看到了集训营的开营通知,并在GitHub的issue上回复来报名。然而,由于我之前没有留意到相关信息,错过了过去好几期的机会。这一次也算做是自己重来的机会吧。不过说起来,这个rustlings的确是我从去年一直做到了现在。今年直接新增10道数据结构题, 本fw直接没法也就上了亿点点unsafe, 好在之前有经验,于是三天冲刺完成,就有了本文之第一阶段总结。

Rust何物者也?一个令人安如磐石的C++ Promax (钟离老爷子看后点个赞) ,胜任WebAssembly、普通命令行应用乃至操作系统编写。语法漂亮,match与macro非常赞。具有成体系的包管理器,安装导入新模块轻松如Python,rust-analyzer也很赞,提供比Pylance更好的高亮提示。运行效率也可以,看似充满了函数,其实llvm优化的也挺不错(

算啦,后面是正经时间,将展示本人写的代码示例和正经点的Rust学习感想,再次感谢清华大学开源操作系统集训营!

示例代码

Bellman-Ford

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
struct Edge {
source: usize,
destination: usize,
weight: i32,
}

fn bellman_ford(edges: &Vec<Edge>, num_vertices: usize, source: usize) -> Vec<i32> {
let mut distances = vec![i32::MAX; num_vertices];
distances[source] = 0;

for _ in 0..num_vertices - 1 {
for edge in edges {
if distances[edge.source] + edge.weight < distances[edge.destination] {
distances[edge.destination] = distances[edge.source] + edge.weight;
}
}
}

for edge in edges {
if distances[edge.source] + edge.weight < distances[edge.destination] {
println!("图中包含负权重环");
break;
}
}

distances
}

fn main() {
let num_vertices = 5;
let edges = vec![
Edge { source: 0, destination: 1, weight: -1 },
Edge { source: 0, destination: 2, weight: 4 },
Edge { source: 1, destination: 2, weight: 3 },
Edge { source: 1, destination: 3, weight: 2 },
Edge { source: 1, destination: 4, weight: 2 },
Edge { source: 3, destination: 2, weight: 5 },
Edge { source: 3, destination: 1, weight: 1 },
Edge { source: 4, destination: 3, weight: -3 },
];
let source = 0;

let distances = bellman_ford(&edges, num_vertices, source);

println!("顶点到源顶点的距离");
for (i, &dist) in distances.iter().enumerate() {
println!("{} \t\t {}", i, dist);
}
}

学习感想

有了Rust这样一门强大而复杂的编程语言,我的学习之旅也就充满了挑战和收获。在学习Rust的过程中,我深刻体会到了其独特的设计理念和功能特性,这些特性使得Rust在系统编程领域有着独特的优势和价值。

首先,Rust的内存管理机制让我印象深刻。通过所有权(ownership)、借用(borrowing)和生命周期(lifetime)等概念,Rust实现了内存安全和线程安全,避免了常见的内存错误和数据竞争问题。虽然这些概念在开始时让我感到有些困惑,但通过不断的实践和阅读文档,我逐渐掌握了它们,并开始感受到它们带来的好处。与其他语言相比,Rust的内存管理机制让我感到更加自信,因为我知道我的程序不会因为内存错误而崩溃。

其次,Rust的模式匹配(pattern matching)和错误处理机制让我眼前一亮。模式匹配不仅可以用于解构数据结构,还可以用于控制程序的流程,使得代码更加清晰和易于理解。而错误处理方面,Rust引入了Result和Option等枚举类型,强制程序员处理可能出现的错误情况,避免了传统的异常机制带来的一些问题。学习如何利用这些特性编写健壮的代码是我学习Rust过程中的一大收获。

此外,Rust的并发编程支持也给我留下了深刻的印象。通过标准库提供的基于消息传递的并发模型,以及原子类型和同步原语等工具,Rust让并发编程变得更加安全和容易。我曾经尝试用Rust编写多线程程序,并发现在Rust的帮助下,我可以更加轻松地管理线程之间的通信和同步,避免了常见的并发错误。

2024开源操作系统训练营第一阶段总结-HCHogan

半年前开始就一直在写rust语言,写了很多练手的项目,所以第一阶段的收获有限,比较印象比较深刻的就是结构体初始化的那个语法:

1
2
3
4
User {
email: String::from("another@example.com"),
..user1
};

没有用过,这次正好复习到了。