diff --git a/README.md b/README.md index bf2f7d0a..f9c33a21 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,13 @@ -# Rust语言圣经 (The Course) +

Rust语言圣经

+ +
+ +
+ +
+ +[![Stars Count](https://img.shields.io/github/stars/sunface/rust-course?style=flat)](https://github.com/sunface/rust-by-practice/stargazers) [![Forks Count](https://img.shields.io/github/forks/sunface/rust-course.svg?style=flat)](https://github.com/naaive/orange/network/members) +
- 在线阅读 - 官方: [https://course.rs](https://course.rs) diff --git a/src/SUMMARY.md b/src/SUMMARY.md index 290efc1b..cfc7e5fe 100644 --- a/src/SUMMARY.md +++ b/src/SUMMARY.md @@ -167,7 +167,8 @@ - [持久化单向链表](too-many-lists/persistent-stack/intro.md) - [数据布局和基本操作](too-many-lists/persistent-stack/layout.md) - [Drop、Arc 及完整代码](too-many-lists/persistent-stack/drop-arc.md) - + - [不咋样的双端队列](too-many-lists/deque/intro.md) + - [数据布局和基本操作](too-many-lists/deque/layout.md) - [易混淆概念解析](confonding/intro.md) - [切片和切片引用](confonding/slice.md) - [Eq 和 PartialEq](confonding/eq.md) @@ -241,7 +242,7 @@ - [HashMap todo](std/hashmap.md) - [Iterator 常用方法 todo](std/iterator.md) -- [Ctrl-C/V: 编程常用代码片段 todo](cases/intro.md) +- [CookBook](cases/intro.md) - [命令行解析 todo](cases/cmd.md) - [配置文件解析 todo](cases/config.md) - [编解码 todo](cases/encoding/intro.md) diff --git a/src/about-book.md b/src/about-book.md index b2e9a377..08341cab 100644 --- a/src/about-book.md +++ b/src/about-book.md @@ -1,15 +1,23 @@ -# Rust 语言圣经 (The Course) +

Rust语言圣经

+ +
+ +
+ +
+ +[![Stars Count](https://img.shields.io/github/stars/sunface/rust-course?style=flat)](https://github.com/sunface/rust-by-practice/stargazers) [![Forks Count](https://img.shields.io/github/forks/sunface/rust-course.svg?style=flat)](https://github.com/naaive/orange/network/members) +
- 在线阅读 - 官方: [https://course.rs](https://course.rs) - 知乎: [支持章节内目录跳转,很好用!](https://www.zhihu.com/column/c_1452781034895446017) -> 学习 Rust 光看书不够,精心设计的习题和项目实践可以让你事半功倍。[Rust By Practice](https://github.com/sunface/rust-by-practice) 是本书的配套习题和实践,覆盖了 easy to hard 各个难度,满足大家对 Rust 的所有期待。 -> -> [Rust 语言周刊](https://github.com/sunface/rust-weekly),每周一发布,精选过去一周的技术文章、业界新闻、开源项目和 Rust 语言动态。 -> -> Rust 优秀项目很多,如何在茫茫码海中与它们相遇?相比 Awesome Rust, [Fancy Rust](https://github.com/sunface/fancy-rust) 能带给你全新的体验和选择。 - +- 本书配套项目 + - [Rust 实战练习](https://github.com/sunface/rust-by-practice),它是本书的配套练习册,提供了大量有挑战性的示例、练习和实践项目,帮助大家解决 Rust 语言从学习到实战的问题 — 毕竟这之间还隔着好几个 Go 语言的难度 :D + - [Rust 语言周刊](https://github.com/sunface/rust-weekly),每周一发布,精选过去一周的技术文章、业界新闻、开源项目和 Rust 语言动态 + - [Rust 酷库推荐](https://github.com/sunface/fancy-rust) Rust 优秀项目很多,如何在茫茫码海中与它们相遇?相比 Awesome Rust,它能带给你全新的体验和选择 + ### 教程简介 **`Rust语言圣经`**涵盖从**入门到精通**所需的 Rust 知识,目录及内容都经过深思熟虑的设计,同时语言生动幽默,行文流畅自如,摆脱技术书籍常有的机器味和晦涩感。 diff --git a/src/advance/concurrency-with-threads/message-passing.md b/src/advance/concurrency-with-threads/message-passing.md index d1aad470..dbf7ba23 100644 --- a/src/advance/concurrency-with-threads/message-passing.md +++ b/src/advance/concurrency-with-threads/message-passing.md @@ -226,7 +226,7 @@ fn main() { thread::sleep(Duration::from_secs(3)); println!("睡眠之后"); - println!("收到值 {}", rx.recv().unwrap()); + println!("receive {}", rx.recv().unwrap()); handle.join().unwrap(); } ``` @@ -239,7 +239,7 @@ fn main() { 发送之后 //···睡眠3秒 睡眠之后 -收到值 1 +receive 1 ``` 主线程因为睡眠阻塞了 3 秒,因此并没有进行消息接收,而子线程却在此期间轻松完成了消息的发送。等主线程睡眠结束后,才姗姗来迟的从通道中接收了子线程老早之前发送的消息。 @@ -279,11 +279,11 @@ fn main() { 发送之前 //···睡眠3秒 睡眠之后 -收到值 1 +receive 1 发送之后 ``` -可以看出,主线程由于睡眠被阻塞导致无法接收消息,因此子线程的发送也一直被阻塞,直到主线程结束睡眠并成功接收消息后,发送才成功:**发送之后**的输出是在**收到值 1**之后,说明**只有接收消息彻底成功后,发送消息才算完成**。 +可以看出,主线程由于睡眠被阻塞导致无法接收消息,因此子线程的发送也一直被阻塞,直到主线程结束睡眠并成功接收消息后,发送才成功:**发送之后**的输出是在**receive 1**之后,说明**只有接收消息彻底成功后,发送消息才算完成**。 #### 消息缓存 diff --git a/src/cargo/reference/workspaces.md b/src/cargo/reference/workspaces.md index ce7f6a70..e3d7b751 100644 --- a/src/cargo/reference/workspaces.md +++ b/src/cargo/reference/workspaces.md @@ -39,7 +39,7 @@ members = [ **对于没有主 `package` 的场景或你希望将所有的 `package` 组织在单独的目录中时,这种方式就非常适合。** -例如 [rust-analyzer](https://github.com/rust-analyzer/rust-analyzer) 就是这样的项目,它的根目录中的 `Cargo.toml` 中并没有 `[package]`,说明该根目录不是一个 `package`,但是却有 `[workspacke]` : +例如 [rust-analyzer](https://github.com/rust-analyzer/rust-analyzer) 就是这样的项目,它的根目录中的 `Cargo.toml` 中并没有 `[package]`,说明该根目录不是一个 `package`,但是却有 `[workspace]` : ```toml [workspace] diff --git a/src/cases/intro.md b/src/cases/intro.md index c6b7d124..0e1d78aa 100644 --- a/src/cases/intro.md +++ b/src/cases/intro.md @@ -1 +1,3 @@ # 场景化用例 + +https://rust-lang-nursery.github.io/rust-cookbook/ \ No newline at end of file diff --git a/src/too-many-lists/deque/intro.md b/src/too-many-lists/deque/intro.md new file mode 100644 index 00000000..44e05eef --- /dev/null +++ b/src/too-many-lists/deque/intro.md @@ -0,0 +1,16 @@ +# 不太优秀的双端队列 +在实现了之前的队列后,我们不禁浮想联翩,如果 `Rc` 是可变的,那是不是可以实现一个双向链表? + +心动不如行动,先来创建新的链表文件 `fourth.rs`,并在 `src/lib.rs` 中添加以下内容: +```rust +// in lib.rs + +pub mod first; +pub mod second; +pub mod third; +pub mod fourth; +``` + +依然是熟悉的从零开始,当然,也依然会用到熟悉的 CV 配方。 + +> 声明:大家看到目录名时,心里就应该在嘀咕了吧?其实你的嘀咕是对的,是的,本章的目的是为了证明之前的想法是糟糕的! \ No newline at end of file diff --git a/src/too-many-lists/deque/layout.md b/src/too-many-lists/deque/layout.md new file mode 100644 index 00000000..b58edc99 --- /dev/null +++ b/src/too-many-lists/deque/layout.md @@ -0,0 +1,316 @@ +# 数据布局和构建 +聪明的读者应该已经想到了:让 `Rc` 可变,就需要使用 `RefCell` 的配合。关于 `RefCell` 的一切,在之前的章节都有介绍,还不熟悉的同学请移步[这里](https://course.rs/advance/smart-pointer/cell-refcell.html)。 + +好了,绝世神兵在手,接下来...我们将见识一个绝世啰嗦的数据结构...如果你来自 GC 语言,那很可能就没有见识过这种阵仗。 + +## 数据布局 + +双向链表意味着每一个节点将同时指向前一个和下一个节点,因此我们的数据结构可能会变成这样: +```rust +use std::rc::Rc; +use std::cell::RefCell; + +pub struct List { + head: Link, + tail: Link, +} + +type Link = Option>>>; + +struct Node { + elem: T, + next: Link, + prev: Link, +} +``` + +耳听忐忑,心怀忐忑,尝试编译下,竟然顺利通过了,thanks god! 接下来再来看看该如何使用它。 + +## 构建 +如果按照之前的构建方式来构建新的数据结构,会有点笨拙,因此我们先尝试将其拆分: +```rust +impl Node { + fn new(elem: T) -> Rc> { + Rc::new(RefCell::new(Node { + elem: elem, + prev: None, + next: None, + })) + } +} + +impl List { + pub fn new() -> Self { + List { head: None, tail: None } + } +} +``` + +```rust +> cargo build + +**一大堆 DEAD CODE 警告,但是好歹可以成功编译** +``` + +## Push +很好,再来向链表的头部推入一个元素。由于双向链表的数据结构和操作逻辑明显更加复杂,因此相比单向链表的单行实现,双向链表的 `push` 操作也要复杂的多。 + +除此之外,我们还需要处理一些关于空链表的边界问题:对于绝大部分操作而言,可能只需要使用 `head` 或 `tail` 指针,但是对于空链表,则需要同时使用它们。 + +一个验证方法 `methods` 是否有效的办法就是看它是否能保持不变性, 每个节点都应该有两个指针指向它: 中间的节点被它前后的节点所指向,而头部的端节点除了被它后面的节点所指向外,还会被链表本身所指向,尾部的端节点亦是如此。 + +```rust +pub fn push_front(&mut self, elem: T) { + let new_head = Node::new(elem); + match self.head.take() { + Some(old_head) => { + // 非空链表,将新的节点跟老的头部相链接 + old_head.prev = Some(new_head.clone()); + new_head.next = Some(old_head); + self.head = Some(new_head); + } + None => { + // 空链表,需要设置 tail 和 head + self.tail = Some(new_head.clone()); + self.head = Some(new_head); + } + } +} +``` + +```rust +cargo build + +error[E0609]: no field `prev` on type `std::rc::Rc>>` + --> src/fourth.rs:39:26 + | +39 | old_head.prev = Some(new_head.clone()); // +1 new_head + | ^^^^ unknown field + +error[E0609]: no field `next` on type `std::rc::Rc>>` + --> src/fourth.rs:40:26 + | +40 | new_head.next = Some(old_head); // +1 old_head + | ^^^^ unknown field +``` + +虽然有报错,但是一切尽在掌握,今天真是万事顺利啊! + +从报错来看,我们无法直接去访问 `prev` 和 `next`,回想一下 `RefCell` 的使用方式,修改代码如下: +```rust +pub fn push_front(&mut self, elem: T) { + let new_head = Node::new(elem); + match self.head.take() { + Some(old_head) => { + old_head.borrow_mut().prev = Some(new_head.clone()); + new_head.borrow_mut().next = Some(old_head); + self.head = Some(new_head); + } + None => { + self.tail = Some(new_head.clone()); + self.head = Some(new_head); + } + } +} +``` + +```shell +> cargo build + +warning: field is never used: `elem` + --> src/fourth.rs:12:5 + | +12 | elem: T, + | ^^^^^^^ + | + = note: #[warn(dead_code)] on by default +``` + +嘿,我又可以了!既然状态神勇,那就趁热打铁,再来看看 `pop`。 + +## Pop +如果说 `new` 和 `push` 是在构建链表,那 `pop` 显然就是一个破坏者。 + +何为完美的破坏?按照构建的过程逆着来一遍就是完美的! +```rust +pub fn pop_front(&mut self) -> Option { + self.head.take().map(|old_head| { + match old_head.borrow_mut().next.take() { + Some(new_head) => { + // 非空链表 + new_head.borrow_mut().prev.take(); + self.head = Some(new_head); + } + None => { + // 空链表 + self.tail.take(); + } + } + old_head.elem + }) +} +``` + +```shell +> cargo build + +error[E0609]: no field `elem` on type `std::rc::Rc>>` + --> src/fourth.rs:64:22 + | +64 | old_head.elem + | ^^^^ unknown field +``` + +哎,怎么就不长记性呢,又是 `RefCell` 惹的祸: +```rust +pub fn pop_front(&mut self) -> Option { + self.head.take().map(|old_head| { + match old_head.borrow_mut().next.take() { + Some(new_head) => { + new_head.borrow_mut().prev.take(); + self.head = Some(new_head); + } + None => { + self.tail.take(); + } + } + old_head.borrow_mut().elem + }) +} +``` + +```shell +cargo build + +error[E0507]: cannot move out of borrowed content + --> src/fourth.rs:64:13 + | +64 | old_head.borrow_mut().elem + | ^^^^^^^^^^^^^^^^^^^^^^^^^^ cannot move out of borrowed content +``` + +额... 我凌乱了,看上去 `Box` 是罪魁祸首,`borrow_mut` 只能返回一个 `&mut Node`,因此无法拿走其所有权。 + +我们需要一个方法来拿走 `RefCell` 的所有权,然后返回给我们一个 `T`, 翻一翻[文档](https://doc.rust-lang.org/std/cell/struct.RefCell.html),可以发现下面这段内容: + +> `fn into_inner(self) -> T` + +> 消费掉 RefCell 并返回内部的值 + +喔,看上去好有安全感的方法: +```rust +old_head.into_inner().elem +``` + +```shell +> cargo build + +error[E0507]: cannot move out of an `Rc` + --> src/fourth.rs:64:13 + | +64 | old_head.into_inner().elem + | ^^^^^^^^ cannot move out of an `Rc` +``` + +...看走眼了,没想到你浓眉大眼也会耍花枪。 `into_inner` 想要拿走 `RecCell` 的所有权,但是还有一个 `Rc` 不愿意,因为 `Rc` 只能让我们获取内部值的不可变引用。 + +大家还记得我们之前实现 `Drop` 时用过的方法吗?在这里一样适用: +```rust +Rc::try_unwrap(old_head).unwrap().into_inner().elem +``` + +`Rc::try_unwrap` 返回一个 `Result`,由于我们不关心 `Err` 的情况( 如果代码合理,这里不会是 `Err` ),直接使用 `unwrap` 即可。 + +```shell +> cargo build + +error[E0599]: no method named `unwrap` found for type `std::result::Result>, std::rc::Rc>>>` in the current scope + --> src/fourth.rs:64:38 + | +64 | Rc::try_unwrap(old_head).unwrap().into_inner().elem + | ^^^^^^ + | + = note: the method `unwrap` exists but the following trait bounds were not satisfied: + `std::rc::Rc>> : std::fmt::Debug` +``` + +额,`unwrap` 要求目标类型是实现了 `Debug` 的,这样才能在报错时提供 `debug` 输出,而 `RefCell` 要实现 `Debug` 需要它内部的 `T` 实现 `Debug`,而我们的 `Node` 并没有实现。 + +当然,我们可以选择为 `Node` 实现,也可以这么做: +```rust +Rc::try_unwrap(old_head).ok().unwrap().into_inner().elem +``` + +```shell +cargo build +``` + +终于成功的运行了,下面依然是惯例 - 写几个测试用例 : +```rust +#[cfg(test)] +mod test { + use super::List; + + #[test] + fn basics() { + let mut list = List::new(); + + // Check empty list behaves right + assert_eq!(list.pop_front(), None); + + // Populate list + list.push_front(1); + list.push_front(2); + list.push_front(3); + + // Check normal removal + assert_eq!(list.pop_front(), Some(3)); + assert_eq!(list.pop_front(), Some(2)); + + // Push some more just to make sure nothing's corrupted + list.push_front(4); + list.push_front(5); + + // Check normal removal + assert_eq!(list.pop_front(), Some(5)); + assert_eq!(list.pop_front(), Some(4)); + + // Check exhaustion + assert_eq!(list.pop_front(), Some(1)); + assert_eq!(list.pop_front(), None); + } +} +``` + +```shell +cargo test + + Running target/debug/lists-5c71138492ad4b4a + +running 9 tests +test first::test::basics ... ok +test fourth::test::basics ... ok +test second::test::iter_mut ... ok +test second::test::basics ... ok +test fifth::test::iter_mut ... ok +test third::test::basics ... ok +test second::test::iter ... ok +test third::test::iter ... ok +test second::test::into_iter ... ok + +test result: ok. 9 passed; 0 failed; 0 ignored; 0 measured +``` + +## Drop +在[循环引用章节](),我们介绍过 `Rc` 最怕的就是引用形成循环,而双向链表恰恰如此。因此,当使用默认的实现来 `drop` 我们的链表时,两个端节点会将各自的引用计数减少到 1, 然后就不会继续减少,最终造成内存泄漏。 + +所以,这里最好的实现就是将每个节点 `pop` 出去,直到获得 `None`: +```rust +impl Drop for List { + fn drop(&mut self) { + while self.pop_front().is_some() {} + } +} +``` + +细心的读者可能已经注意到,我们还未实现在链表尾部 `push` 和 `pop` 的操作,但由于所需的实现跟之前差别不大,因此我们会在后面直接给出,下面先来看看更有趣的。 diff --git a/内容变更记录.md b/内容变更记录.md index 5fb4ee35..c76d7d35 100644 --- a/内容变更记录.md +++ b/内容变更记录.md @@ -1,6 +1,10 @@ # ChangeLog 记录一些值得注意的变更。 +## 2022-03-16 + +- 新增章节: [deque-数据布局和基本操作](https://course.rs/too-many-lists/deque/layout.html) + ## 2022-03-14 - 新增章节: [Rust 陷阱 - UTF-8 引发的性能隐患](https://course.rs/pitfalls/utf8-performance.html)