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语言圣经
+
+
+

+
+
+
+
+[](https://github.com/sunface/rust-by-practice/stargazers) [](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语言圣经
+
+
+

+
+
+
+
+[](https://github.com/sunface/rust-by-practice/stargazers) [](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)