diff --git a/.gitignore b/.gitignore
new file mode 100644
index 00000000..314f02b1
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1 @@
+*.txt
\ No newline at end of file
diff --git a/README.md b/README.md
index b69e91cf..0493e185 100644
--- a/README.md
+++ b/README.md
@@ -1,18 +1,13 @@
+
+| Ⅰ | Ⅱ | Ⅲ | Ⅳ | Ⅴ | Ⅵ | Ⅶ | Ⅷ | Ⅸ | Ⅹ |
+| :--------: | :---------: | :---------: | :---------: | :---------: | :---------:| :---------: | :-------: | :-------:| :------:|
+|网络[:cloud:](#网络-cloud) |操作系统[:computer:](#操作系统-computer)| 算法[:pencil2:](#数据结构与算法-pencil2)| 面向对象[:couple:](#面向对象-couple) |数据库[:floppy_disk:](#数据库-floppy_disk)| Java [:coffee:](#java-coffee)| 分布式[:sweat_drops:](#分布式-sweat_drops)| 工具[:hammer:](#工具-hammer)| 编码实践[:speak_no_evil:](#编码实践-speak_no_evil)| 后记[:memo:](#后记-memo) |
+
-
+:loudspeaker: 本仓库不参与商业行为,不向读者收取任何费用。
-
-
-
-
-
-
- 
+:loudspeaker: This repository is not engaging in business activities, and does not charge readers any fee.
+
## 网络 :cloud:
@@ -38,7 +33,7 @@
> [算法](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/算法.md)
-整理自《算法 第四版》,主要整理了面试常问的排序和查找算法。
+整理自《算法 第四版》
> [剑指 Offer 题解](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/剑指%20offer%20题解.md)
@@ -46,7 +41,7 @@
> [Leetcode 题解](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/Leetcode%20题解.md)
-对题目做了一个分类,并对每种题型的解题思路做了总结。已经整理了 300+ 的题目,基本涵盖所有经典题目。
+对题目做了一个分类,并对每种题型的解题思路做了总结。
## 面向对象 :couple:
@@ -56,7 +51,7 @@
> [面向对象思想](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/面向对象思想.md)
-一些面向对象思想和原则。
+一些面向对象思想和设计原则。
## 数据库 :floppy_disk:
@@ -64,19 +59,23 @@
整理自《数据库系统概论 第四版》
-> [SQL 语法](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/SQL%20语法.md)
+> [SQL](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/SQL.md)
整理自《SQL 必知必会》
> [MySQL](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/MySQL.md)
-整理自《高性能 MySQL》,整理了一些重点内容。
+整理自《高性能 MySQL》
+
+> [Redis](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/Redis.md)
+
+整理自《Redis 设计与实现》和《Redis 实战》
## Java :coffee:
-> [JVM](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/JVM.md)
+> [Java 虚拟机](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/Java%20虚拟机.md)
-整理自《深入理解 Java 虚拟机》,主要整理了内存模型、垃圾回收以及类加载机制。
+整理自《深入理解 Java 虚拟机》
> [Java 并发](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/Java%20并发.md)
@@ -86,15 +85,44 @@
容器的一些总结,包含容器源码的分析。
-> [Java IO](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/Java%20IO.md)
+> [Java I/O](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/Java%20IO.md)
-File、InputStream OutputStream、Reader Writer、Serializable、Socket、NIO
+File, InputStream OutputStream, Reader Writer, Serializable, Socket, NIO
> [Java 基础](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/Java%20基础.md)
整理了一些常见考点。
-## 编码实践 :hammer:
+> [JDK 中的设计模式](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/JDK%20中的设计模式.md)
+
+对每种设计模式做了一个总结,并给出在 JDK 中的使用实例。
+
+## 分布式 :sweat_drops:
+
+> [分布式基础](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/分布式基础.md)
+
+整理自《大规模分布式存储系统》
+
+> [一致性协议](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/一致性协议.md)
+
+两阶段提交、Paxos、Raft、拜占庭将军问题。
+
+> [分布式问题分析](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/分布式问题分析.md)
+
+分布式事务、负载均衡算法与实现、分布式锁、分布式 Session、分库分表的分布式困境与应对之策。
+
+
+## 工具 :hammer:
+
+> [Git](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/Git.md)
+
+整理一些 Git 的使用和概念。
+
+> [正则表达式](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/正则表达式.md)
+
+整理自《正则表达式必知必会》
+
+## 编码实践 :speak_no_evil:
> [重构](https://github.com/CyC2018/InnterviewNotes/blob/master/notes/重构.md)
@@ -108,33 +136,40 @@ File、InputStream OutputStream、Reader Writer、Serializable、Socket、NIO
Google 开源项目的代码风格规范。
-
-
## 后记 :memo:
-网上有很多相关的资料,但是这些资料都比较零散。本仓库的笔记是从经典的书籍和材料中整理而来,在整理出重点的同时会尽可能保证知识的系统性,因此比较适合作为应对面试的学习资料。
+**关于仓库**
-笔记内容按照 [中文文案排版指北](http://mazhuang.org/wiki/chinese-copywriting-guidelines/#%E4%B8%8D%E8%A6%81%E4%BD%BF%E7%94%A8%E4%B8%8D%E5%9C%B0%E9%81%93%E7%9A%84%E7%BC%A9%E5%86%99) 进行排版,以保证内容的可读性。这里提供了本人实现的 Markdown 文档排版工具的下载:[Markdown-Typesetting](https://github.com/CyC2018/Markdown-Typesetting)。
+本仓库是笔者在准备 2018 年春招实习过程中的学习总结,内容以计算机书籍的学习笔记为主,在整理重点知识的同时会尽量保证知识的系统性。
-由于 Github 使用的 GFM 不支持 MathJax 公式,也不支持 TOC 标记,为了把本地的 Markdown 文档转换为 GFM 支持的格式,需要替换 MathJax 公式为 CodeCogs 的云服务和重新生成 TOC 目录。并且为了让图片显示效果更好,笔记内容基本使用了 <center> 标记来让图片居中显示,但是 GFM 却不支持 <center> 标记,因此也需要进行一定的转换。这里提供了本人实现的 GFM 文档转换工具的下载:[GFM-Converter](https://github.com/CyC2018/GFM-Converter)。
+**关于贡献**
-因为大部分内容是一个字一个字打上去的,难免会有一些笔误,如果发现,可以直接在相应的文档上编辑修改。
+因为大部分内容是笔者一个字一个字打上去的,所有难免会有一些笔误。如果发现,可以直接在相应的文档上编辑修改。
-如果觉得内容不够完善或者有写的不好的地方,您可以在 Issues 中发表反馈意见。
+笔者能力有限,很多内容还不够完善。如果您希望和笔者一起完善这个仓库,可以在发表一个 Issue,表明您想要添加的内容,笔者会及时查看。
-笔记内容可供个人随意使用,转载或者引用请注明出处,毕竟写了很长时间没那么轻松~
+因为不打算将这个仓库做成一个大而全的面试宝典,只希望添加一些比较通用的基础知识,或者是与 Java 和分布式相关的内容,但是不添加 Java Web 相关的内容。
-## Donate
+您也可以在 Issues 中发表关于改进本仓库的建议。
-[Alipay](https://github.com/CyC2018/InterviewNotes/blob/master/other/alipay.md)
+**关于上传**
-## License
+笔者在本地使用为知笔记软件进行书写,为了方便将本地笔记内容上传到 Github 上,实现了一整套自动化上传方案,包括文本文件的导出、提取图片、Markdown 文档转换、Git 同步。
-
+进行 Markdown 文档转换是因为 Github 使用的 GFM 不支持 MathJax 公式和 TOC 标记,所以需要替换 MathJax 公式为 CodeCogs 的云服务和重新生成 TOC 目录。这里提供了笔者实现的 GFM 文档转换工具的下载:[GFM-Converter](https://github.com/CyC2018/GFM-Converter)。
+
+**关于排版**
+
+笔记内容按照 [中文文案排版指北](http://mazhuang.org/wiki/chinese-copywriting-guidelines/#%E4%B8%8D%E8%A6%81%E4%BD%BF%E7%94%A8%E4%B8%8D%E5%9C%B0%E9%81%93%E7%9A%84%E7%BC%A9%E5%86%99) 进行排版,以保证内容的可读性。这里提供了笔者实现的中英混排文档在线排版工具:[Text-Typesetting](https://github.com/CyC2018/Markdown-Typesetting),目前实现了加空格的功能,之后打算实现对英文专有名词提示首字母大写的功能。
+
+不使用 `![]()` 这种方式来引用图片是为了能够控制图片以合适的大小显示。而且 GFM 不支持 `
![]() ` 让图片居中显示,只能使用 `` ,所以只能使用 img 标签来引用图片。
+
+**关于转载**
+
+本仓库内容使用到的资料都会在最后面的参考资料中给出引用链接,希望您在使用本仓库的内容时也能给出相应的引用链接。
+
+**鸣谢**
+
+[TeeKee](https://github.com/linw7)
-本作品采用 知识共享署名-非商业性使用-相同方式共享 3.0 中国大陆许可协议 进行许可。
diff --git a/notes/Git.md b/notes/Git.md
new file mode 100644
index 00000000..1695faad
--- /dev/null
+++ b/notes/Git.md
@@ -0,0 +1,158 @@
+
+* [学习资料](#学习资料)
+* [集中式与分布式](#集中式与分布式)
+* [Git 的中心服务器](#git-的中心服务器)
+* [Git 工作流](#git-工作流)
+* [分支实现](#分支实现)
+* [冲突](#冲突)
+* [Fast forward](#fast-forward)
+* [分支管理策略](#分支管理策略)
+* [储藏(Stashing)](#储藏stashing)
+* [SSH 传输设置](#ssh-传输设置)
+* [.gitignore 文件](#gitignore-文件)
+* [Git 命令一览](#git-命令一览)
+
+
+
+# 学习资料
+
+- [Git - 简明指南](http://rogerdudler.github.io/git-guide/index.zh.html)
+- [图解 Git](http://marklodato.github.io/visual-git-guide/index-zh-cn.html)
+- [廖雪峰 : Git 教程](https://www.liaoxuefeng.com/wiki/0013739516305929606dd18361248578c67b8067c8c017b000)
+- [Learn Git Branching](https://learngitbranching.js.org/)
+
+# 集中式与分布式
+
+Git 属于分布式版本控制系统,而 SVN 属于集中式。
+
+集中式版本控制只有中心服务器拥有一份代码,而分布式版本控制每个人的电脑上就有一份完整的代码。
+
+集中式版本控制有安全性问题,当中心服务器挂了所有人都没办法工作了。
+
+集中式版本控制需要连网才能工作,如果网速过慢,那么提交一个文件的会慢的无法让人忍受。而分布式版本控制不需要连网就能工作。
+
+分布式版本控制新建分支、合并分支操作速度非常快,而集中式版本控制新建一个分支相当于复制一份完整代码。
+
+# Git 的中心服务器
+
+Git 的中心服务器用来交换每个用户的修改。没有中心服务器也能工作,但是中心服务器能够 24 小时保持开机状态,这样就能更方便的交换修改。Github 就是一种 Git 中心服务器。
+
+# Git 工作流
+
+
+
+新建一个仓库之后,当前目录就成为了工作区,工作区下有一个隐藏目录 .git,它属于 Git 的版本库。
+
+Git 版本库有一个称为 stage 的暂存区,还有自动创建的 master 分支以及指向分支的 HEAD 指针。
+
+
+
+- git add files 把文件的修改添加到暂存区
+- git commit 把暂存区的修改提交到当前分支,提交之后暂存区就被清空了
+- git reset -- files 使用当前分支上的修改覆盖暂缓区,用来撤销最后一次 git add files
+- git checkout -- files 使用暂存区的修改覆盖工作目录,用来撤销本地修改
+
+
+
+可以跳过暂存区域直接从分支中取出修改或者直接提交修改到分支中
+
+- git commit -a 直接把所有文件的修改添加到暂缓区然后执行提交
+- git checkout HEAD -- files 取出最后一次修改,可以用来进行回滚操作
+
+# 分支实现
+
+Git 把每次提交都连成一条时间线。分支使用指针来实现,例如 master 分支指针指向时间线的最后一个节点,也就是最后一次提交。HEAD 指针指向的是当前分支。
+
+
+
+新建分支是新建一个指针指向时间线的最后一个节点,并让 HEAD 指针指向新分支表示新分支成为当前分支。
+
+
+
+每次提交只会让当前分支向前移动,而其它分支不会移动。
+
+
+
+合并分支也只需要改变指针即可。
+
+
+
+# 冲突
+
+当两个分支都对同一个文件的同一行进行了修改,在分支合并时就会产生冲突。
+
+
+
+Git 会使用 <<<<<<< ,======= ,>>>>>>> 标记出不同分支的内容,只需要把不同分支中冲突部分修改成一样就能解决冲突。
+
+```
+<<<<<<< HEAD
+Creating a new branch is quick & simple.
+=======
+Creating a new branch is quick AND simple.
+>>>>>>> feature1
+```
+
+# Fast forward
+
+"快进式合并"(fast-farward merge),会直接将 master 分支指向合并的分支,这种模式下进行分支合并会丢失分支信息,也就不能在分支历史上看出分支信息。
+
+可以在合并时加上 --no-ff 参数来禁用 Fast forward 模式,并且加上 -m 参数让合并时产生一个新的 commit。
+
+```
+$ git merge --no-ff -m "merge with no-ff" dev
+```
+
+
+
+# 分支管理策略
+
+master 分支应该是非常稳定的,只用来发布新版本;
+
+日常开发在开发分支 dev 上进行。
+
+
+
+# 储藏(Stashing)
+
+在一个分支上操作之后,如果还没有将修改提交到分支上,此时进行切换分支,那么另一个分支上也能看到新的修改。这是因为所有分支都共用一个工作区的缘故。
+
+可以使用 git stash 将当前分支的修改储藏起来,此时当前工作区的所有修改都会被存到栈上,也就是说当前工作区是干净的,没有任何未提交的修改。此时就可以安全的切换到其它分支上了。
+
+```
+$ git stash
+Saved working directory and index state \ "WIP on master: 049d078 added the index file"
+HEAD is now at 049d078 added the index file (To restore them type "git stash apply")
+```
+
+该功能可以用于 bug 分支的实现。如果当前正在 dev 分支上进行开发,但是此时 master 上有个 bug 需要修复,但是 dev 分支上的开发还未完成,不想立即提交。在新建 bug 分支并切换到 bug 分支之前就需要使用 git stash 将 dev 分支的未提交修改储藏起来。
+
+# SSH 传输设置
+
+Git 仓库和 Github 中心仓库之间是通过 SSH 加密。
+
+如果工作区下没有 .ssh 目录,或者该目录下没有 id_rsa 和 id_rsa.pub 这两个文件,可以通过以下命令来创建 SSH Key:
+
+```
+$ ssh-keygen -t rsa -C "youremail@example.com"
+```
+
+然后把公钥 id_rsa.pub 的内容复制到 Github "Account settings" 的 SSH Keys 中。
+
+# .gitignore 文件
+
+忽略以下文件:
+
+1. 操作系统自动生成的文件,比如缩略图;
+2. 编译生成的中间文件,比如 Java 编译产生的 .class 文件;
+3. 自己的敏感信息,比如存放口令的配置文件。
+
+不需要全部自己编写,可以到 [https://github.com/github/gitignore](https://github.com/github/gitignore) 中进行查询。
+
+# Git 命令一览
+
+
+
+比较详细的地址:http://www.cheat-sheets.org/saved-copy/git-cheat-sheet.pdf
+
+
diff --git a/notes/HTTP.md b/notes/HTTP.md
index 1aa5f46c..7238542e 100644
--- a/notes/HTTP.md
+++ b/notes/HTTP.md
@@ -1,47 +1,65 @@
-* [基础概念](#基础概念)
+* [一 、基础概念](#一-基础概念)
* [Web 基础](#web-基础)
* [URL](#url)
* [请求和响应报文](#请求和响应报文)
-* [HTTP 方法](#http-方法)
- * [GET:获取资源](#get获取资源)
- * [POST:传输实体主体](#post传输实体主体)
- * [HEAD:获取报文首部](#head获取报文首部)
- * [PUT:上传文件](#put上传文件)
- * [DELETE:删除文件](#delete删除文件)
- * [OPTIONS:查询支持的方法](#options查询支持的方法)
- * [TRACE:追踪路径](#trace追踪路径)
- * [CONNECT:要求用隧道协议连接代理](#connect要求用隧道协议连接代理)
-* [HTTP 状态码](#http-状态码)
+* [二、HTTP 方法](#二http-方法)
+ * [GET](#get)
+ * [HEAD](#head)
+ * [POST](#post)
+ * [PUT](#put)
+ * [PATCH](#patch)
+ * [DELETE](#delete)
+ * [OPTIONS](#options)
+ * [CONNECT](#connect)
+ * [TRACE](#trace)
+* [三、HTTP 状态码](#三http-状态码)
+ * [1XX 信息](#1xx-信息)
* [2XX 成功](#2xx-成功)
* [3XX 重定向](#3xx-重定向)
* [4XX 客户端错误](#4xx-客户端错误)
* [5XX 服务器错误](#5xx-服务器错误)
-* [HTTP 首部](#http-首部)
+* [四、HTTP 首部](#四http-首部)
* [通用首部字段](#通用首部字段)
* [请求首部字段](#请求首部字段)
* [响应首部字段](#响应首部字段)
* [实体首部字段](#实体首部字段)
-* [具体应用](#具体应用)
+* [五、具体应用](#五具体应用)
* [Cookie](#cookie)
* [缓存](#缓存)
* [持久连接](#持久连接)
+ * [管线化处理](#管线化处理)
* [编码](#编码)
- * [分块传输](#分块传输)
+ * [分块传输编码](#分块传输编码)
* [多部分对象集合](#多部分对象集合)
* [范围请求](#范围请求)
* [内容协商](#内容协商)
* [虚拟主机](#虚拟主机)
* [通信数据转发](#通信数据转发)
-* [HTTPs](#https)
+* [六、HTTPs](#六https)
* [加密](#加密)
* [认证](#认证)
* [完整性](#完整性)
-* [HTTP/1.0 与 HTTP/1.1 的区别](#http10-与-http11-的区别)
+* [七、Web 攻击技术](#七web-攻击技术)
+ * [攻击模式](#攻击模式)
+ * [跨站脚本攻击](#跨站脚本攻击)
+ * [跨站点请求伪造](#跨站点请求伪造)
+ * [SQL 注入攻击](#sql-注入攻击)
+ * [拒绝服务攻击](#拒绝服务攻击)
+* [八、GET 和 POST 的区别](#八get-和-post-的区别)
+ * [参数](#参数)
+ * [安全](#安全)
+ * [幂等性](#幂等性)
+ * [可缓存](#可缓存)
+ * [XMLHttpRequest](#xmlhttprequest)
+* [九、各版本比较](#九各版本比较)
+ * [HTTP/1.0 与 HTTP/1.1 的区别](#http10-与-http11-的区别)
+ * [HTTP/1.1 与 HTTP/2.0 的区别](#http11-与-http20-的区别)
+* [参考资料](#参考资料)
-# 基础概念
+# 一 、基础概念
## Web 基础
@@ -57,96 +75,138 @@
URI 包含 URL 和 URN,目前 WEB 只有 URL 比较流行,所以见到的基本都是 URL。
-
+
## 请求和响应报文
-**请求报文**
+### 1. 请求报文
-
+
-**响应报文**
+### 2. 响应报文
-
+
-# HTTP 方法
+# 二、HTTP 方法
-客户端发送的请求报文第一行为请求行,包含了方法字段。
+客户端发送的 **请求报文** 第一行为请求行,包含了方法字段。
-## GET:获取资源
+## GET
-## POST:传输实体主体
+> 获取资源
-POST 主要目的不是获取资源,而是传输实体主体数据。
+当前网络请求中,绝大部分使用的是 GET 方法。
-GET 和 POST 的请求都能使用额外的参数,但是 GET 的参数是以查询字符串出现在 URL 中,而 POST 的参数存储在实体主体部分。
+## HEAD
-```
-GET /test/demo_form.asp?name1=value1&name2=value2 HTTP/1.1
-```
-```
-POST /test/demo_form.asp HTTP/1.1
-Host: w3schools.com
-name1=value1&name2=value2
-```
-
-GET 的传参方式相比于 POST 安全性较差,因为 GET 传的参数在 URL 是可见的,可能会泄露私密信息。并且 GET 只支持 ASCII 字符,如果参数为中文则可能会出现乱码,而 POST 支持标准字符集。
-
-## HEAD:获取报文首部
+> 获取报文首部
和 GET 方法一样,但是不返回报文实体主体部分。
主要用于确认 URL 的有效性以及资源更新的日期时间等。
-## PUT:上传文件
+## POST
-由于自身不带验证机制,任何人都可以上传文件,因此存在安全性问题,一般 WEB 网站不使用该方法。
+> 传输实体主体
-## DELETE:删除文件
+POST 主要用来传输数据,而 GET 主要用来获取资源。
+
+更多 POST 与 GET 的比较请见第八章。
+
+## PUT
+
+> 上传文件
+
+由于自身不带验证机制,任何人都可以上传文件,因此存在安全性问题,一般不使用该方法。
+
+```html
+PUT /new.html HTTP/1.1
+Host: example.com
+Content-type: text/html
+Content-length: 16
+
+New File
+```
+
+## PATCH
+
+> 对资源进行部分修改
+
+PUT 也可以用于修改资源,但是只能完全替代原始资源,PATCH 允许部分修改。
+
+```html
+PATCH /file.txt HTTP/1.1
+Host: www.example.com
+Content-Type: application/example
+If-Match: "e0023aa4e"
+Content-Length: 100
+
+[description of changes]
+```
+
+## DELETE
+
+> 删除文件
与 PUT 功能相反,并且同样不带验证机制。
-## OPTIONS:查询支持的方法
+```html
+DELETE /file.html HTTP/1.1
+```
+
+## OPTIONS
+
+> 查询支持的方法
查询指定的 URL 能够支持的方法。
会返回 Allow: GET, POST, HEAD, OPTIONS 这样的内容。
-## TRACE:追踪路径
+## CONNECT
+
+> 要求用隧道协议连接代理
+
+要求在与代理服务器通信时建立隧道,使用 SSL(Secure Sockets Layer,安全套接层)和 TLS(Transport Layer Security,传输层安全)协议把通信内容加密后经网络隧道传输。
+
+```html
+CONNECT www.example.com:443 HTTP/1.1
+```
+
+
+
+## TRACE
+
+> 追踪路径
服务器会将通信路径返回给客户端。
发送请求时,在 Max-Forwards 首部字段中填入数值,每经过一个服务器就会减 1,当数值为 0 时就停止传输。
-TRACE 一般不会使用,并且它容易受到 XST 攻击(Cross-Site Tracing,跨站追踪),因此更不会去使用它。
+通常不会使用 TRACE,并且它容易受到 XST 攻击(Cross-Site Tracing,跨站追踪),因此更不会去使用它。
-
+# 三、HTTP 状态码
-## CONNECT:要求用隧道协议连接代理
-
-主要使用 SSL(Secure Sokets Layer,安全套接字)和 TLS(Transport Layer Security,传输层安全)协议把通信内容加密后经网络隧道传输。
-
-
-
-# HTTP 状态码
-
-服务器返回的响应报文中第一行为状态行,包含了状态码以及原因短语,来告知客户端请求的结果。
+服务器返回的 **响应报文** 中第一行为状态行,包含了状态码以及原因短语,用来告知客户端请求的结果。
| 状态码 | 类别 | 原因短语 |
-| --- | --- | --- |
+| :---: | :---: | :---: |
| 1XX | Informational(信息性状态码) | 接收的请求正在处理 |
| 2XX | Success(成功状态码) | 请求正常处理完毕 |
| 3XX | Redirection(重定向状态码) | 需要进行附加操作以完成请求 |
| 4XX | Client Error(客户端错误状态码) | 服务器无法处理请求 |
| 5XX | Server Error(服务器错误状态码) | 服务器处理请求出错 |
+## 1XX 信息
+
+- **100 Continue** :表明到目前为止都很正常,客户端可以继续发送请求或者忽略这个响应。
+
## 2XX 成功
- **200 OK**
- **204 No Content** :请求已经成功处理,但是返回的响应报文不包含实体的主体部分。一般在只需要从客户端往服务器发送信息,而不需要返回数据时使用。
-- **206 Partial Content**
+- **206 Partial Content** :表示客户端进行了范围请求。响应报文包含由 Content-Range 指定范围的实体内容。
## 3XX 重定向
@@ -154,21 +214,19 @@ TRACE 一般不会使用,并且它容易受到 XST 攻击(Cross-Site Tracing
- **302 Found** :临时性重定向
-- **303 See Other**
+- **303 See Other** :和 302 有着相同的功能,但是 303 明确要求客户端应该采用 GET 方法获取资源。
-- 注:虽然 HTTP 协议规定 301、302 状态下重定向时不允许把 POST 方法改成 GET 方法,但是大多数浏览器都会 在 301、302 和 303 状态下的重定向把 POST 方法改成 GET 方法。
+- 注:虽然 HTTP 协议规定 301、302 状态下重定向时不允许把 POST 方法改成 GET 方法,但是大多数浏览器都会在 301、302 和 303 状态下的重定向把 POST 方法改成 GET 方法。
-- **304 Not Modified** :如果请求报文首部包含一些条件,例如:If-Match,If-ModifiedSince,If-None-Match,If-Range,If-Unmodified-Since,但是不满足条件,则服务器会返回 304 状态码。
+- **304 Not Modified** :如果请求报文首部包含一些条件,例如:If-Match,If-ModifiedSince,If-None-Match,If-Range,If-Unmodified-Since,如果不满足条件,则服务器会返回 304 状态码。
- **307 Temporary Redirect** :临时重定向,与 302 的含义类似,但是 307 要求浏览器不会把重定向请求的 POST 方法改成 GET 方法。
## 4XX 客户端错误
-- **400 Bad Request** :请求报文中存在语法错误
+- **400 Bad Request** :请求报文中存在语法错误。
-- **401 Unauthorized** :该状态码表示发送的请求需要有通过 HTTP 认证(BASIC 认证、DIGEST 认证)的认证信息。如果之前已进行过一次请求,则表示用户认证失败。
-
-
+- **401 Unauthorized** :该状态码表示发送的请求需要有认证信息(BASIC 认证、DIGEST 认证)。如果之前已进行过一次请求,则表示用户认证失败。
- **403 Forbidden** :请求被拒绝,服务器端没有必要给出拒绝的详细理由。
@@ -176,11 +234,11 @@ TRACE 一般不会使用,并且它容易受到 XST 攻击(Cross-Site Tracing
## 5XX 服务器错误
-- **500 Internal Server Error** :服务器正在执行请求时发生错误
+- **500 Internal Server Error** :服务器正在执行请求时发生错误。
-- **503 Service Unavilable** :该状态码表明服务器暂时处于超负载或正在进行停机维护,现在无法处理请求。
+- **503 Service Unavilable** :服务器暂时处于超负载或正在进行停机维护,现在无法处理请求。
-# HTTP 首部
+# 四、HTTP 首部
有 4 种类型的首部字段:通用首部字段、请求首部字段、响应首部字段和实体首部字段。
@@ -189,9 +247,9 @@ TRACE 一般不会使用,并且它容易受到 XST 攻击(Cross-Site Tracing
## 通用首部字段
| 首部字段名 | 说明 |
-| -- | -- |
+| :--: | :--: |
| Cache-Control | 控制缓存的行为 |
-| Connection | 逐跳首部、 连接的管理 |
+| Connection | 控制不再转发给代理的首部字段、管理持久连接|
| Date | 创建报文的日期时间 |
| Pragma | 报文指令 |
| Trailer | 报文末端的首部一览 |
@@ -203,7 +261,7 @@ TRACE 一般不会使用,并且它容易受到 XST 攻击(Cross-Site Tracing
## 请求首部字段
| 首部字段名 | 说明 |
-| -- | -- |
+| :--: | :--: |
| Accept | 用户代理可处理的媒体类型 |
| Accept-Charset | 优先的字符集 |
| Accept-Encoding | 优先的内容编码 |
@@ -227,7 +285,7 @@ TRACE 一般不会使用,并且它容易受到 XST 攻击(Cross-Site Tracing
## 响应首部字段
| 首部字段名 | 说明 |
-| -- | -- |
+| :--: | :--: |
| Accept-Ranges | 是否接受字节范围请求 |
| Age | 推算资源创建经过时间 |
| ETag | 资源的匹配信息 |
@@ -241,11 +299,11 @@ TRACE 一般不会使用,并且它容易受到 XST 攻击(Cross-Site Tracing
## 实体首部字段
| 首部字段名 | 说明 |
-| -- | -- |
+| :--: | :--: |
| Allow | 资源可支持的 HTTP 方法 |
| Content-Encoding | 实体主体适用的编码方式 |
| Content-Language | 实体主体的自然语言 |
-| Content-Length | 实体主体的大小(单位: 字节) |
+| Content-Length | 实体主体的大小 |
| Content-Location | 替代对应资源的 URI |
| Content-MD5 | 实体主体的报文摘要 |
| Content-Range | 实体主体的位置范围 |
@@ -253,80 +311,169 @@ TRACE 一般不会使用,并且它容易受到 XST 攻击(Cross-Site Tracing
| Expires | 实体主体过期的日期时间 |
| Last-Modified | 资源的最后修改日期时间 |
-# 具体应用
+# 五、具体应用
## Cookie
HTTP 协议是无状态的,主要是为了让 HTTP 协议尽可能简单,使得它能够处理大量事务。HTTP/1.1 引入 Cookie 来保存状态信息。
-服务器发送的响应报文包含 Set-Cookie 字段,客户端得到响应报文后把 Cookie 内容保存到浏览器中。下次再发送请求时,从浏览器中读出 Cookie 值,在请求报文中包含 Cookie 字段,这样服务器就知道客户端的状态信息了。Cookie 状态信息保存在客户端浏览器中,而不是服务器上。
+Cookie 是服务器发送给客户端的数据,该数据会被保存在浏览器中,并且客户端的下一次请求报文会包含该数据。通过 Cookie 可以让服务器知道两个请求是否来自于同一个客户端,从而实现保持登录状态等功能。
-
+### 1. 创建过程
-Set-Cookie 字段有以下属性:
+服务器发送的响应报文包含 Set-Cookie 字段,客户端得到响应报文后把 Cookie 内容保存到浏览器中。
+
+```html
+HTTP/1.0 200 OK
+Content-type: text/html
+Set-Cookie: yummy_cookie=choco
+Set-Cookie: tasty_cookie=strawberry
+
+[page content]
+```
+
+客户端之后发送请求时,会从浏览器中读出 Cookie 值,在请求报文中包含 Cookie 字段。
+
+```html
+GET /sample_page.html HTTP/1.1
+Host: www.example.org
+Cookie: yummy_cookie=choco; tasty_cookie=strawberry
+```
+
+### 2. 分类
+
+- 会话期 Cookie:浏览器关闭之后它会被自动删除,也就是说它仅在会话期内有效。
+- 持久性 Cookie:指定一个特定的过期时间(Expires)或有效期(Max-Age)之后就成为了持久性的 Cookie。
+
+```html
+Set-Cookie: id=a3fWa; Expires=Wed, 21 Oct 2015 07:28:00 GMT;
+```
+
+### 3. Set-Cookie
| 属性 | 说明 |
-| -- | -- |
+| :--: | -- |
| NAME=VALUE | 赋予 Cookie 的名称和其值(必需项) |
| expires=DATE | Cookie 的有效期(若不明确指定则默认为浏览器关闭前为止) |
| path=PATH | 将服务器上的文件目录作为 Cookie 的适用对象(若不指定则默认为文档所在的文件目录) |
-| domain= 域名 | 作为 Cookie 适用对象的域名(若不指定则默认为创建 Cookie 的服务器的域名) |
-| Secure | 仅在 HTTPS 安全通信时才会发送 Cookie |
+| domain=域名 | 作为 Cookie 适用对象的域名(若不指定则默认为创建 Cookie 的服务器的域名) |
+| Secure | 仅在 HTTPs 安全通信时才会发送 Cookie |
| HttpOnly | 加以限制,使 Cookie 不能被 JavaScript 脚本访问 |
-**Session 和 Cookie 区别**
+### 4. Session 和 Cookie 区别
-Session 是服务器用来跟踪用户的一种手段,每个 Session 都有一个唯一标识:Session ID。当服务器创建了一个 Session 时,给客户端发送的响应报文就包含了 Set-Cookie 字段,其中有一个名为 sid 的键值对,这个键值对就是 Session ID。客户端收到后就把 Cookie 保存在浏览器中,并且之后发送的请求报文都包含 Session ID。HTTP 就是通过 Session 和 Cookie 这两种方式一起合作来实现跟踪用户状态的,Session 用于服务器端,Cookie 用于客户端。
+Session 是服务器用来跟踪用户的一种手段,每个 Session 都有一个唯一标识:Session ID。当服务器创建了一个 Session 时,给客户端发送的响应报文包含了 Set-Cookie 字段,其中有一个名为 sid 的键值对,这个键值对就是 Session ID。客户端收到后就把 Cookie 保存在浏览器中,并且之后发送的请求报文都包含 Session ID。HTTP 就是通过 Session 和 Cookie 这两种方式一起合作来实现跟踪用户状态的,Session 用于服务器端,Cookie 用于客户端。
-**浏览器禁用 Cookie 的情况**
+### 5. 浏览器禁用 Cookie 的情况
会使用 URL 重写技术,在 URL 后面加上 sid=xxx 。
-**使用 Cookie 实现用户名和密码的自动填写**
+### 6. 使用 Cookie 实现用户名和密码的自动填写
-网站脚本会自动从 Cookie 中读取用户名和密码,从而实现自动填写。
+网站脚本会自动从保存在浏览器中的 Cookie 读取用户名和密码,从而实现自动填写。
+
+但是如果 Set-Cookie 指定了 HttpOnly 属性,就无法通过 Javascript 脚本获取 Cookie 信息,这是出于安全性考虑。
## 缓存
-有两种缓存方法:让代理服务器进行缓存和让客户端浏览器进行缓存。
+### 1. 优点
-Cache-Control 用于控制缓存的行为。Cache-Control: no-cache 有两种含义,如果是客户端向缓存服务器发送的请求报文中含有该指令,表示客户端不想要缓存的资源;如果是源服务器向缓存服务器发送的响应报文中含有该指令,表示缓存服务器不能对资源进行缓存。
+1. 降低服务器的负担;
+2. 提高响应速度(缓存资源比服务器上的资源离客户端更近)。
-Expires 字段可以用于告知缓存服务器该资源什么时候会过期。当首部字段 Cache-Control 有指定 max-age 指令时,比起首部字段 Expires,会优先处理 max-age 指令。
+### 2. 实现方法
+
+1. 让代理服务器进行缓存;
+2. 让客户端浏览器进行缓存。
+
+### 3. Cache-Control 字段
+
+HTTP 通过 Cache-Control 首部字段来控制缓存。
+
+```html
+Cache-Control: private, max-age=0, no-cache
+```
+
+### 4. no-cache 指令
+
+该指令出现在请求报文的 Cache-Control 字段中,表示缓存服务器需要先向原服务器验证缓存资源是否过期;
+
+该指令出现在响应报文的 Cache-Control 字段中,表示缓存服务器在进行缓存之前需要先验证缓存资源的有效性。
+
+### 5. no-store 指令
+
+该指令表示缓存服务器不能对请求或响应的任何一部分进行缓存。
+
+no-cache 不表示不缓存,而是缓存之前需要先进行验证,no-store 才是不进行缓存。
+
+### 6. max-age 指令
+
+该指令出现在请求报文的 Cache-Control 字段中,如果缓存资源的缓存时间小于该指令指定的时间,那么就能接受该缓存。
+
+该指令出现在响应报文的 Cache-Control 字段中,表示缓存资源在缓存服务器中保存的时间。
+
+Expires 字段也可以用于告知缓存服务器该资源什么时候会过期。在 HTTP/1.1 中,会优先处理 Cache-Control : max-age 指令;而在 HTTP/1.0 中,Cache-Control : max-age 指令会被忽略掉。
## 持久连接
-当浏览器访问一个包含多张图片的 HTML 页面时,除了请求访问 HTML 页面资源,还会请求图片资源,如果每进行一次 HTTP 通信就要断开一次 TCP 连接,连接建立和断开的开销会很大。 **持久连接** 只需要进行一次 TCP 连接就能进行多次 HTTP 通信。HTTP/1.1 开始,所有的连接默认都是持久连接。
+当浏览器访问一个包含多张图片的 HTML 页面时,除了请求访问 HTML 页面资源,还会请求图片资源,如果每进行一次 HTTP 通信就要断开一次 TCP 连接,连接建立和断开的开销会很大。持久连接只需要建立一次 TCP 连接就能进行多次 HTTP 通信。
-
+
-持久连接需要使用 Connection 首部字段进行管理。HTTP/1.1 开始 HTTP 默认是持久化连接的,如果要断开 TCP 连接,需要由客户端或者服务器端提出断开,使用 Connection: close;而在 HTTP/1.1 之前默认是非持久化连接的,如果要维持持续连接,需要使用 Keep-Alive。
+持久连接需要使用 Connection 首部字段进行管理。HTTP/1.1 开始 HTTP 默认是持久化连接的,如果要断开 TCP 连接,需要由客户端或者服务器端提出断开,使用 Connection : close;而在 HTTP/1.1 之前默认是非持久化连接的,如果要维持持续连接,需要使用 Connection : Keep-Alive。
-管线化方式可以同时发送多个请求和响应,而不需要发送一个请求然后等待响应之后再发下一个请求。
+## 管线化处理
-
+HTTP/1.1 支持管线化处理,可以同时发送多个请求和响应,而不需要发送一个请求然后等待响应之后再发下一个请求。
## 编码
编码(Encoding)主要是为了对实体进行压缩。常用的编码有:gzip、compress、deflate、identity,其中 identity 表示不执行压缩的编码格式。
-## 分块传输
+## 分块传输编码
-分块传输(Chunked Transfer Coding)可以把数据分割成多块,让浏览器逐步显示页面。
+Chunked Transfer Coding,可以把数据分割成多块,让浏览器逐步显示页面。
## 多部分对象集合
-一份报文主体内可含有多种类型的实体同时发送,每个部分之间用 boundary 字段定义的分隔符进行分隔;每个部分都可以有首部字段。
+一份报文主体内可含有多种类型的实体同时发送,每个部分之间用 boundary 字段定义的分隔符进行分隔,每个部分都可以有首部字段。
例如,上传多个表单时可以使用如下方式:
-
+```html
+Content-Type: multipart/form-data; boundary=AaB03x
+
+--AaB03x
+Content-Disposition: form-data; name="submit-name"
+
+Larry
+--AaB03x
+Content-Disposition: form-data; name="files"; filename="file1.txt"
+Content-Type: text/plain
+
+... contents of file1.txt ...
+--AaB03x--
+```
## 范围请求
如果网络出现中断,服务器只发送了一部分数据,范围请求使得客户端能够只请求未发送的那部分数据,从而避免服务器端重新发送所有数据。
-在请求报文首部中添加 Range 字段,然后指定请求的范围,例如 Range:bytes=5001-10000。请求成功的话服务器发送 206 Partial Content 状态。
+在请求报文首部中添加 Range 字段指定请求的范围,请求成功的话服务器发送 206 Partial Content 状态。
+
+```html
+GET /z4d4kWk.jpg HTTP/1.1
+Host: i.imgur.com
+Range: bytes=0-1023
+```
+
+```html
+HTTP/1.1 206 Partial Content
+Content-Range: bytes 0-1023/146515
+Content-Length: 1024
+...
+(binary content)
+```
## 内容协商
@@ -336,33 +483,33 @@ Expires 字段可以用于告知缓存服务器该资源什么时候会过期。
## 虚拟主机
-使用虚拟主机技术,使得一台服务器拥有多个域名,并且在逻辑上可以看成多个服务器。
+HTTP/1.1 使用虚拟主机技术,使得一台服务器拥有多个域名,并且在逻辑上可以看成多个服务器。
+
+使用 Host 首部字段进行处理。
## 通信数据转发
-**代理**
+### 1. 代理
代理服务器接受客户端的请求,并且转发给其它服务器。
-代理服务器一般是透明的,不会改变 URL。
-
使用代理的主要目的是:缓存、网络访问控制以及访问日志记录。
-
+代理服务器分为正向代理和反向代理两种,用户察觉得到正向代理的存在,而反向代理一般位于内部网络中,用户察觉不到。
-**网关**
+
+
+
+
+### 2. 网关
与代理服务器不同的是,网关服务器会将 HTTP 转化为其它协议进行通信,从而请求其它非 HTTP 服务器的服务。
-
+### 3. 隧道
-**隧道**
+使用 SSL 等加密手段,为客户端和服务器之间建立一条安全的通信线路。隧道本身不去解析 HTTP 请求。
-使用 SSL 等加密手段,为客户端和服务器之间建立一条安全的通信线路。
-
-
-
-# HTTPs
+# 六、HTTPs
HTTP 有以下安全性问题:
@@ -370,40 +517,334 @@ HTTP 有以下安全性问题:
2. 不验证通信方的身份,通信方的身份有可能遭遇伪装;
3. 无法证明报文的完整性,报文有可能遭篡改。
-HTTPs 并不是新协议,而是 HTTP 先和 SSL(Secure Socket Layer)通信,再由 SSL 和 TCP 通信。通过使用 SSL,HTTPs 提供了加密、认证和完整性保护。
+HTTPs 并不是新协议,而是 HTTP 先和 SSL(Secure Sockets Layer)通信,再由 SSL 和 TCP 通信。也就是说 HTTPs 使用了隧道进行通信。
+
+通过使用 SSL,HTTPs 具有了加密、认证和完整性保护。
+
+
## 加密
-有两种加密方式:对称密钥加密和公开密钥加密。对称密钥加密的加密和解密使用同一密钥,而公开密钥加密使用一对密钥用于加密和解密,分别为公开密钥和私有密钥。公开密钥所有人都可以获得,通信发送方获得接收方的公开密钥之后,就可以使用公开密钥进行加密,接收方收到通信内容后使用私有密钥解密。
+### 1. 对称密钥加密
-对称密钥加密的缺点:无法安全传输密钥;公开密钥加密的缺点:相对来说更耗时。
+对称密钥加密(Symmetric-Key Encryption),加密的加密和解密使用同一密钥。
-HTTPs 采用 **混合的加密机制** ,使用公开密钥加密用于传输对称密钥,之后使用对称密钥加密进行通信。(下图中,共享密钥即对称密钥)
+- 优点:运算速度快;
+- 缺点:密钥容易被获取。
-
+
+
+### 2. 公开密钥加密
+
+公开密钥加密(Public-Key Encryption),也称为非对称密钥加密,使用一对密钥用于加密和解密,分别为公开密钥和私有密钥。公开密钥所有人都可以获得,通信发送方获得接收方的公开密钥之后,就可以使用公开密钥进行加密,接收方收到通信内容后使用私有密钥解密。
+
+- 优点:更为安全;
+- 缺点:运算速度慢;
+
+
+
+### 3. HTTPs 采用的加密方式
+
+HTTPs 采用混合的加密机制,使用公开密钥加密用于传输对称密钥,之后使用对称密钥加密进行通信。(下图中的 Session Key 就是对称密钥)
+
+
## 认证
-通过使用 **证书** 来对通信方进行认证。证书中有公开密钥数据,如果可以验证公开密钥的确属于通信方的,那么就可以确定通信方是可靠的。
+通过使用 **证书** 来对通信方进行认证。
-数字证书认证机构(CA,Certificate Authority)可以对其颁发的公开密钥证书对其进行验证。
+数字证书认证机构(CA,Certificate Authority)是客户端与服务器双方都可信赖的第三方机构。服务器的运营人员向 CA 提出公开密钥的申请,CA 在判明提出申请者的身份之后,会对已申请的公开密钥做数字签名,然后分配这个已签名的公开密钥,并将该公开密钥放入公开密钥证书后绑定在一起。
-进行 HTTPs 通信时,服务器会把证书发送给客户端,客户端取得其中的公开密钥之后,就可以开始通信。
+进行 HTTPs 通信时,服务器会把证书发送给客户端,客户端取得其中的公开密钥之后,先进行验证,如果验证通过,就可以开始通信。
+
+
使用 OpenSSL 这套开源程序,每个人都可以构建一套属于自己的认证机构,从而自己给自己颁发服务器证书。浏览器在访问该服务器时,会显示“无法确认连接安全性”或“该网站的安全证书存在问题”等警告消息。
-客户端证书需要用户自行安装,只有在业务需要非常高的安全性时才使用客户端证书,例如网上银行。
-
## 完整性
-SSL 提供摘要功能来验证完整性。
+SSL 提供报文摘要功能来验证完整性。
-# HTTP/1.0 与 HTTP/1.1 的区别
+# 七、Web 攻击技术
-HTTP/1.1 新增了以下内容:
+## 攻击模式
-- 默认为长连接;
-- 提供了范围请求功能;
-- 提供了虚拟主机的功能;
-- 多了一些缓存处理字段;
-- 多了一些状态码;
+### 1. 主动攻击
+
+直接攻击服务器,具有代表性的有 SQL 注入和 OS 命令注入。
+
+### 2. 被动攻击
+
+设下圈套,让用户发送有攻击代码的 HTTP 请求,用户会泄露 Cookie 等个人信息,具有代表性的有跨站脚本攻击和跨站请求伪造。
+
+## 跨站脚本攻击
+
+### 1. 概念
+
+跨站脚本攻击(Cross-Site Scripting, XSS),可以将代码注入到用户浏览的网页上,这种代码包括 HTML 和 JavaScript。利用网页开发时留下的漏洞,通过巧妙的方法注入恶意指令代码到网页,使用户加载并执行攻击者恶意制造的网页程序。攻击成功后,攻击者可能得到更高的权限(如执行一些操作)、私密网页内容、会话和 Cookie 等各种内容。
+
+例如有一个论坛网站,攻击者可以在上面发表以下内容:
+
+```
+
+```
+
+之后该内容可能会被渲染成以下形式:
+
+```
+
+```
+
+另一个用户浏览了含有这个内容的页面将会跳往 domain.com 并携带了当前作用域的 Cookie。如果这个论坛网站通过 Cookie 管理用户登录状态,那么攻击者就可以通过这个 Cookie 登录被攻击者的账号了。
+
+### 2. 危害
+
+- 伪造虚假的输入表单骗取个人信息
+- 窃取用户的 Cookie 值
+- 显示伪造的文章或者图片
+
+### 3. 防范手段
+
+(一)过滤特殊字符
+
+许多语言都提供了对 HTML 的过滤:
+
+- PHP 的 htmlentities() 或是 htmlspecialchars()。
+- Python 的 cgi.escape()。
+- Java 的 xssprotect (Open Source Library)。
+- Node.js 的 node-validator。
+
+(二)指定 HTTP 的 Content-Type
+
+通过这种方式,可以避免内容被当成 HTML 解析,比如 PHP 语言可以使用以下代码:
+
+```php
+
+```
+
+## 跨站点请求伪造
+
+### 1. 概念
+
+跨站点请求伪造(Cross-site request forgery,CSRF),是攻击者通过一些技术手段欺骗用户的浏览器去访问一个自己曾经认证过的网站并执行一些操作(如发邮件,发消息,甚至财产操作如转账和购买商品)。由于浏览器曾经认证过,所以被访问的网站会认为是真正的用户操作而去执行。这利用了 Web 中用户身份验证的一个漏洞:简单的身份验证只能保证请求发自某个用户的浏览器,却不能保证请求本身是用户自愿发出的。
+
+XSS 利用的是用户对指定网站的信任,CSRF 利用的是网站对用户网页浏览器的信任。
+
+假如一家银行用以执行转账操作的 URL 地址如下:
+
+```
+http://www.examplebank.com/withdraw?account=AccoutName&amount=1000&for=PayeeName。
+```
+
+那么,一个恶意攻击者可以在另一个网站上放置如下代码:
+
+```
+ 。
+```
+
+如果有账户名为 Alice 的用户访问了恶意站点,而她之前刚访问过银行不久,登录信息尚未过期,那么她就会损失 1000 资金。
+
+这种恶意的网址可以有很多种形式,藏身于网页中的许多地方。此外,攻击者也不需要控制放置恶意网址的网站。例如他可以将这种地址藏在论坛,博客等任何用户生成内容的网站中。这意味着如果服务器端没有合适的防御措施的话,用户即使访问熟悉的可信网站也有受攻击的危险。
+
+透过例子能够看出,攻击者并不能通过 CSRF 攻击来直接获取用户的账户控制权,也不能直接窃取用户的任何信息。他们能做到的,是欺骗用户浏览器,让其以用户的名义执行操作。
+
+### 2. 防范手段
+
+(一)检查 Referer 字段
+
+HTTP 头中有一个 Referer 字段,这个字段用以标明请求来源于哪个地址。在处理敏感数据请求时,通常来说,Referer 字段应和请求的地址位于同一域名下。
+
+这种办法简单易行,工作量低,仅需要在关键访问处增加一步校验。但这种办法也有其局限性,因其完全依赖浏览器发送正确的 Referer 字段。虽然 HTTP 协议对此字段的内容有明确的规定,但并无法保证来访的浏览器的具体实现,亦无法保证浏览器没有安全漏洞影响到此字段。并且也存在攻击者攻击某些浏览器,篡改其 Referer 字段的可能。
+
+(二)添加校验 Token
+
+由于 CSRF 的本质在于攻击者欺骗用户去访问自己设置的地址,所以如果要求在访问敏感数据请求时,要求用户浏览器提供不保存在 Cookie 中,并且攻击者无法伪造的数据作为校验,那么攻击者就无法再执行 CSRF 攻击。这种数据通常是表单中的一个数据项。服务器将其生成并附加在表单中,其内容是一个伪乱数。当客户端通过表单提交请求时,这个伪乱数也一并提交上去以供校验。正常的访问时,客户端浏览器能够正确得到并传回这个伪乱数,而通过 CSRF 传来的欺骗性攻击中,攻击者无从事先得知这个伪乱数的值,服务器端就会因为校验 Token 的值为空或者错误,拒绝这个可疑请求。
+
+## SQL 注入攻击
+
+### 1. 概念
+
+服务器上的数据库运行非法的 SQL 语句。
+
+### 2. 攻击原理
+
+例如一个网站登录验证的 SQL 查询代码为:
+
+```sql
+strSQL = "SELECT * FROM users WHERE (name = '" + userName + "') and (pw = '"+ passWord +"');"
+```
+
+如果填入以下内容:
+
+```sql
+userName = "1' OR '1'='1";
+passWord = "1' OR '1'='1";
+```
+
+那么 SQL 查询字符串为:
+
+```sql
+strSQL = "SELECT * FROM users WHERE (name = '1' OR '1'='1') and (pw = '1' OR '1'='1');"
+```
+
+此时无需验证通过就能执行以下查询:
+
+```sql
+strSQL = "SELECT * FROM users;"
+```
+
+### 3. 危害
+
+- 数据表中的数据外泄,例如个人机密数据,账户数据,密码等。
+- 数据结构被黑客探知,得以做进一步攻击(例如 SELECT * FROM sys.tables)。
+- 数据库服务器被攻击,系统管理员账户被窜改(例如 ALTER LOGIN sa WITH PASSWORD='xxxxxx')。
+- 获取系统较高权限后,有可能得以在网页加入恶意链接、恶意代码以及 XSS 等。
+- 经由数据库服务器提供的操作系统支持,让黑客得以修改或控制操作系统(例如 xp_cmdshell "net stop iisadmin" 可停止服务器的 IIS 服务)。
+- 破坏硬盘数据,瘫痪全系统(例如 xp_cmdshell "FORMAT C:")。
+
+### 4. 防范手段
+
+- 在设计应用程序时,完全使用参数化查询(Parameterized Query)来设计数据访问功能。
+- 在组合 SQL 字符串时,先针对所传入的参数作字符取代(将单引号字符取代为连续 2 个单引号字符)。
+- 如果使用 PHP 开发网页程序的话,亦可打开 PHP 的魔术引号(Magic quote)功能(自动将所有的网页传入参数,将单引号字符取代为连续 2 个单引号字符)。
+- 其他,使用其他更安全的方式连接 SQL 数据库。例如已修正过 SQL 注入问题的数据库连接组件,例如 ASP.NET 的 SqlDataSource 对象或是 LINQ to SQL。
+- 使用 SQL 防注入系统。
+
+## 拒绝服务攻击
+
+### 1. 概念
+
+拒绝服务攻击(denial-of-service attack,DoS),亦称洪水攻击,其目的在于使目标电脑的网络或系统资源耗尽,使服务暂时中断或停止,导致其正常用户无法访问。
+
+分布式拒绝服务攻击(distributed denial-of-service attack,DDoS),指攻击者使用网络上两个或以上被攻陷的电脑作为“僵尸”向特定的目标发动“拒绝服务”式攻击。
+
+> [维基百科:拒绝服务攻击](https://zh.wikipedia.org/wiki/%E9%98%BB%E6%96%B7%E6%9C%8D%E5%8B%99%E6%94%BB%E6%93%8A)
+
+# 八、GET 和 POST 的区别
+
+## 参数
+
+GET 和 POST 的请求都能使用额外的参数,但是 GET 的参数是以查询字符串出现在 URL 中,而 POST 的参数存储在内容实体中。
+
+GET 的传参方式相比于 POST 安全性较差,因为 GET 传的参数在 URL 中是可见的,可能会泄露私密信息。并且 GET 只支持 ASCII 字符,如果参数为中文则可能会出现乱码,而 POST 支持标准字符集。
+
+```
+GET /test/demo_form.asp?name1=value1&name2=value2 HTTP/1.1
+```
+
+```
+POST /test/demo_form.asp HTTP/1.1
+Host: w3schools.com
+name1=value1&name2=value2
+```
+
+## 安全
+
+安全的 HTTP 方法不会改变服务器状态,也就是说它只是可读的。
+
+GET 方法是安全的,而 POST 却不是,因为 POST 的目的是传送实体主体内容,这个内容可能是用户上传的表单数据,上传成功之后,服务器可能把这个数据存储到数据库中,因此状态也就发生了改变。
+
+安全的方法除了 GET 之外还有:HEAD、OPTIONS。
+
+不安全的方法除了 POST 之外还有 PUT、DELETE。
+
+## 幂等性
+
+幂等的 HTTP 方法,同样的请求被执行一次与连续执行多次的效果是一样的,服务器的状态也是一样的。换句话说就是,幂等方法不应该具有副作用(统计用途除外)。在正确实现的条件下,GET,HEAD,PUT 和 DELETE 等方法都是幂等的,而 POST 方法不是。所有的安全方法也都是幂等的。
+
+GET /pageX HTTP/1.1 是幂等的。连续调用多次,客户端接收到的结果都是一样的:
+
+```
+GET /pageX HTTP/1.1
+GET /pageX HTTP/1.1
+GET /pageX HTTP/1.1
+GET /pageX HTTP/1.1
+```
+
+POST /add_row HTTP/1.1 不是幂等的。如果调用多次,就会增加多行记录:
+
+```
+POST /add_row HTTP/1.1
+POST /add_row HTTP/1.1 -> Adds a 2nd row
+POST /add_row HTTP/1.1 -> Adds a 3rd row
+```
+
+DELETE /idX/delete HTTP/1.1 是幂等的,即便是不同请求之间接收到的状态码不一样:
+
+```
+DELETE /idX/delete HTTP/1.1 -> Returns 200 if idX exists
+DELETE /idX/delete HTTP/1.1 -> Returns 404 as it just got deleted
+DELETE /idX/delete HTTP/1.1 -> Returns 404
+```
+
+## 可缓存
+
+如果要对响应进行缓存,需要满足以下条件:
+
+1. 请求报文的 HTTP 方法本身是可缓存的,包括 GET 和 HEAD,但是 PUT 和 DELETE 不可缓存,POST 在多数情况下不可缓存的。
+2. 响应报文的状态码是可缓存的,包括:200, 203, 204, 206, 300, 301, 404, 405, 410, 414, and 501。
+3. 响应报文的 Cache-Control 首部字段没有指定不进行缓存。
+
+## XMLHttpRequest
+
+为了阐述 POST 和 GET 的另一个区别,需要先了解 XMLHttpRequest:
+
+> XMLHttpRequest 是一个 API,它为客户端提供了在客户端和服务器之间传输数据的功能。它提供了一个通过 URL 来获取数据的简单方式,并且不会使整个页面刷新。这使得网页只更新一部分页面而不会打扰到用户。XMLHttpRequest 在 AJAX 中被大量使用。
+
+在使用 XMLHttpRequest 的 POST 方法时,浏览器会先发送 Header 再发送 Data。但并不是所有浏览器会这么做,例如火狐就不会。
+
+# 九、各版本比较
+
+## HTTP/1.0 与 HTTP/1.1 的区别
+
+1. HTTP/1.1 默认是持久连接
+2. HTTP/1.1 支持管线化处理
+3. HTTP/1.1 支持虚拟主机
+4. HTTP/1.1 新增状态码 100
+5. HTTP/1.1 支持分块传输编码
+6. HTTP/1.1 新增缓存处理指令 max-age
+
+具体内容见上文
+
+## HTTP/1.1 与 HTTP/2.0 的区别
+
+### 1. 多路复用
+
+HTTP/2.0 使用多路复用技术,使用同一个 TCP 连接来处理多个请求。
+
+### 2. 首部压缩
+
+HTTP/1.1 的首部带有大量信息,而且每次都要重复发送。HTTP/2.0 要求通讯双方各自缓存一份首部字段表,从而避免了重复传输。
+
+### 3. 服务端推送
+
+在客户端请求一个资源时,会把相关的资源一起发送给客户端,客户端就不需要再次发起请求了。例如客户端请求 index.html 页面,服务端就把 index.js 一起发给客户端。
+
+### 4. 二进制格式
+
+HTTP/1.1 的解析是基于文本的,而 HTTP/2.0 采用二进制格式。
+
+# 参考资料
+
+- 上野宣. 图解 HTTP[M]. 人民邮电出版社, 2014.
+- [MDN : HTTP](https://developer.mozilla.org/en-US/docs/Web/HTTP)
+- [Are http:// and www really necessary?](https://www.webdancers.com/are-http-and-www-necesary/)
+- [HTTP (HyperText Transfer Protocol)](https://www.ntu.edu.sg/home/ehchua/programming/webprogramming/HTTP_Basics.html)
+- [Web-VPN: Secure Proxies with SPDY & Chrome](https://www.igvita.com/2011/12/01/web-vpn-secure-proxies-with-spdy-chrome/)
+- [File:HTTP persistent connection.svg](http://en.wikipedia.org/wiki/File:HTTP_persistent_connection.svg)
+- [Proxy server](https://en.wikipedia.org/wiki/Proxy_server)
+- [What Is This HTTPS/SSL Thing And Why Should You Care?](https://www.x-cart.com/blog/what-is-https-and-ssl.html)
+- [What is SSL Offloading?](https://securebox.comodo.com/ssl-sniffing/ssl-offloading/)
+- [Sun Directory Server Enterprise Edition 7.0 Reference - Key Encryption](https://docs.oracle.com/cd/E19424-01/820-4811/6ng8i26bn/index.html)
+- [An Introduction to Mutual SSL Authentication](https://www.codeproject.com/Articles/326574/An-Introduction-to-Mutual-SSL-Authentication)
+- [The Difference Between URLs and URIs](https://danielmiessler.com/study/url-uri/)
+- [维基百科:跨站脚本](https://zh.wikipedia.org/wiki/%E8%B7%A8%E7%B6%B2%E7%AB%99%E6%8C%87%E4%BB%A4%E7%A2%BC)
+- [维基百科:SQL 注入攻击](https://zh.wikipedia.org/wiki/SQL%E8%B3%87%E6%96%99%E9%9A%B1%E7%A2%BC%E6%94%BB%E6%93%8A)
+- [维基百科:跨站点请求伪造](https://zh.wikipedia.org/wiki/%E8%B7%A8%E7%AB%99%E8%AF%B7%E6%B1%82%E4%BC%AA%E9%80%A0)
+- [维基百科:拒绝服务攻击](https://zh.wikipedia.org/wiki/%E9%98%BB%E6%96%B7%E6%9C%8D%E5%8B%99%E6%94%BB%E6%93%8A)
+- [What is the difference between a URI, a URL and a URN?](https://stackoverflow.com/questions/176264/what-is-the-difference-between-a-uri-a-url-and-a-urn)
+- [XMLHttpRequest](https://developer.mozilla.org/zh-CN/docs/Web/API/XMLHttpRequest)
+- [XMLHttpRequest (XHR) Uses Multiple Packets for HTTP POST?](https://blog.josephscott.org/2009/08/27/xmlhttprequest-xhr-uses-multiple-packets-for-http-post/)
+- [Symmetric vs. Asymmetric Encryption – What are differences?](https://www.ssl2buy.com/wiki/symmetric-vs-asymmetric-encryption-what-are-differences)
diff --git a/notes/JDK 中的设计模式.md b/notes/JDK 中的设计模式.md
new file mode 100644
index 00000000..92f53cb3
--- /dev/null
+++ b/notes/JDK 中的设计模式.md
@@ -0,0 +1,283 @@
+
+* [一、创建型](#一创建型)
+ * [1. 单例模式](#1-单例模式)
+ * [2. 简单工厂模式](#2-简单工厂模式)
+ * [3. 工厂方法模式](#3-工厂方法模式)
+ * [4. 抽象工厂](#4-抽象工厂)
+ * [5. 生成器模式](#5-生成器模式)
+ * [6. 原型模式](#6-原型模式)
+* [二、行为型](#二行为型)
+ * [1. 责任链](#1-责任链)
+ * [2. 命令模式](#2-命令模式)
+ * [3. 解释器模式](#3-解释器模式)
+ * [4. 迭代器](#4-迭代器)
+ * [5. 中间人模式](#5-中间人模式)
+ * [6. 备忘录模式](#6-备忘录模式)
+ * [7. 观察者模式](#7-观察者模式)
+ * [8. 策略模式](#8-策略模式)
+ * [9. 模板方法](#9-模板方法)
+ * [10. 访问者模式](#10-访问者模式)
+ * [11. 空对象模式](#11-空对象模式)
+* [三、结构型](#三结构型)
+ * [1. 适配器](#1-适配器)
+ * [2. 桥接模式](#2-桥接模式)
+ * [3. 组合模式](#3-组合模式)
+ * [4. 装饰者模式](#4-装饰者模式)
+ * [5. 蝇量模式](#5-蝇量模式)
+ * [6. 动态代理](#6-动态代理)
+* [参考资料](#参考资料)
+
+
+
+# 一、创建型
+
+## 1. 单例模式
+
+确保只实例化一个对象,并提供一个对象的全局访问点。
+
+```java
+java.lang.Runtime#getRuntime()
+java.awt.Toolkit#getDefaultToolkit()
+java.awt.GraphicsEnvironment#getLocalGraphicsEnvironment()
+java.awt.Desktop#getDesktop()
+```
+
+## 2. 简单工厂模式
+
+在不对用户暴露对象内部逻辑的前提下创建对象。
+
+## 3. 工厂方法模式
+
+定义创建对象的接口,但是让子类来决定应该使用哪个类来创建。
+
+```java
+java.lang.Proxy#newProxyInstance()
+java.lang.Object#toString()
+java.lang.Class#newInstance()
+java.lang.reflect.Array#newInstance()
+java.lang.reflect.Constructor#newInstance()
+java.lang.Boolean#valueOf(String)
+java.lang.Class#forName()
+```
+
+## 4. 抽象工厂
+
+提供一个创建相关对象家族的接口,而没有明确指明它们的类。
+
+```java
+java.util.Calendar#getInstance()
+java.util.Arrays#asList()
+java.util.ResourceBundle#getBundle()
+java.sql.DriverManager#getConnection()
+java.sql.Connection#createStatement()
+java.sql.Statement#executeQuery()
+java.text.NumberFormat#getInstance()
+javax.xml.transform.TransformerFactory#newInstance()
+```
+
+## 5. 生成器模式
+
+定义一个新的类来构造另一个类的实例,以创建一个复杂的对象。
+
+它可以封装一个对象的构造过程,并允许按步骤构造。
+
+```java
+java.lang.StringBuilder#append()
+java.lang.StringBuffer#append()
+java.sql.PreparedStatement
+javax.swing.GroupLayout.Group#addComponent()
+```
+
+## 6. 原型模式
+
+使用原型实例指定要创建对象的类型;通过复制这个原型来创建新对象。
+
+```java
+java.lang.Object#clone()
+java.lang.Cloneable
+```
+
+# 二、行为型
+
+## 1. 责任链
+
+避免将请求的发送者附加到其接收者,从而使其它对象也可以处理请求;将请求以对象的方式发送到链上直到请求被处理完毕。
+
+```java
+java.util.logging.Logger#log()
+javax.servlet.Filter#doFilter()
+```
+
+## 2. 命令模式
+
+将命令封装进对象中;允许使用命令对象对客户对象进行参数化;允许将命令对象存放到队列中。
+
+```java
+java.lang.Runnable
+javax.swing.Action
+```
+
+## 3. 解释器模式
+
+为语言创建解释器,通常由语言的语法和语法分析来定义。
+
+```java
+java.util.Pattern
+java.text.Normalizer
+java.text.Format
+```
+
+## 4. 迭代器
+
+提供一种一致的访问聚合对象元素的方法,并且不暴露聚合对象的内部表示。
+
+```java
+java.util.Iterator
+java.util.Enumeration
+```
+
+## 5. 中间人模式
+
+使用中间人对象来封装对象之间的交互。中间人模式可以让降低交互对象之间的耦合程度。
+
+```java
+java.util.Timer
+java.util.concurrent.Executor#execute()
+java.util.concurrent.ExecutorService#submit()
+java.lang.reflect.Method#invoke()
+```
+
+## 6. 备忘录模式
+
+在不违反封装的情况下获得对象的内部状态,从而在需要时可以将对象恢复到最初状态。
+
+```java
+java.util.Date
+java.io.Serializable
+```
+
+## 7. 观察者模式
+
+定义对象之间的一对多依赖,当一个对象状态改变时,它的所有依赖都会收到通知并且自动更新状态。
+
+```java
+java.util.EventListener
+javax.servlet.http.HttpSessionBindingListener
+javax.servlet.http.HttpSessionAttributeListener
+javax.faces.event.PhaseListener
+```
+
+## 8. 策略模式
+
+定义一系列算法,封装每个算法,并使它们可以互换。策略可以让算法独立于使用它的客户端。
+
+```java
+java.util.Comparator#compare()
+javax.servlet.http.HttpServlet
+javax.servlet.Filter#doFilter()
+```
+
+## 9. 模板方法
+
+定义算法框架,并将一些步骤的实现延迟到子类。通过模板方法,子类可以重新定义算法的某些步骤,而不用改变算法的结构。
+
+```java
+java.util.Collections#sort()
+java.io.InputStream#skip()
+java.io.InputStream#read()
+java.util.AbstractList#indexOf()
+```
+
+## 10. 访问者模式
+
+提供便捷的维护方式来操作一组对象。它使你在不改变操作对象的前提下,可以修改或扩展对象的行为。
+
+例如集合,它可以包含不同类型的元素,访问者模式允许在不知道具体元素类型的前提下对集合元素进行一些操作。
+
+```java
+javax.lang.model.element.Element and javax.lang.model.element.ElementVisitor
+javax.lang.model.type.TypeMirror and javax.lang.model.type.TypeVisitor
+```
+
+## 11. 空对象模式
+
+使用什么都不做的空对象来替代 NULL。
+
+# 三、结构型
+
+## 1. 适配器
+
+把一个类接口转换成另一个用户需要的接口。
+
+```java
+java.util.Arrays#asList()
+javax.swing.JTable(TableModel)
+java.io.InputStreamReader(InputStream)
+java.io.OutputStreamWriter(OutputStream)
+javax.xml.bind.annotation.adapters.XmlAdapter#marshal()
+javax.xml.bind.annotation.adapters.XmlAdapter#unmarshal()
+```
+
+## 2. 桥接模式
+
+将抽象与实现分离开来,使它们可以独立变化。
+
+```java
+AWT (It provides an abstraction layer which maps onto the native OS the windowing support.)
+JDBC
+```
+
+## 3. 组合模式
+
+将对象组合成树形结构来表示整体-部分层次关系,允许用户以相同的方式处理单独对象和组合对象。
+
+```java
+javax.swing.JComponent#add(Component)
+java.awt.Container#add(Component)
+java.util.Map#putAll(Map)
+java.util.List#addAll(Collection)
+java.util.Set#addAll(Collection)
+```
+
+## 4. 装饰者模式
+
+为对象动态添加功能。
+
+```java
+java.io.BufferedInputStream(InputStream)
+java.io.DataInputStream(InputStream)
+java.io.BufferedOutputStream(OutputStream)
+java.util.zip.ZipOutputStream(OutputStream)
+java.util.Collections#checked[List|Map|Set|SortedSet|SortedMap]()
+```
+
+## 5. 蝇量模式
+
+利用共享的方式来支持大量的对象,这些对象一部分内部状态是相同的,而另一份状态可以变化。
+
+Java 利用缓存来加速大量小对象的访问时间。
+
+```java
+java.lang.Integer#valueOf(int)
+java.lang.Boolean#valueOf(boolean)
+java.lang.Byte#valueOf(byte)
+java.lang.Character#valueOf(char)
+```
+
+## 6. 动态代理
+
+提供一个占位符来控制对象的访问。
+
+代理可以是一些轻量级的对象,它控制着对重量级对象的访问,只有在真正实例化这些重量级对象时才会去实例化它。
+
+```java
+java.lang.reflect.Proxy
+RMI
+```
+
+# 参考资料
+
+- [The breakdown of design patterns in JDK](http://www.programering.com/a/MTNxAzMwATY.html)
+- [Design Patterns](http://www.oodesign.com/)
+
+
diff --git a/notes/Java IO.md b/notes/Java IO.md
index 5eb6d462..b48c90c3 100644
--- a/notes/Java IO.md
+++ b/notes/Java IO.md
@@ -1,56 +1,48 @@
-* [概览](#概览)
-* [磁盘操作](#磁盘操作)
-* [字节操作](#字节操作)
-* [字符操作](#字符操作)
-* [对象操作](#对象操作)
-* [网络操作](#网络操作)
- * [1. InetAddress](#1-inetaddress)
- * [2. URL](#2-url)
- * [3. Sockets](#3-sockets)
- * [4. Datagram](#4-datagram)
-* [NIO](#nio)
- * [1. 流与块](#1-流与块)
- * [2. 通道与缓冲区](#2-通道与缓冲区)
- * [2.1 通道](#21-通道)
- * [2.2 缓冲区](#22-缓冲区)
- * [3. 缓冲区状态变量](#3-缓冲区状态变量)
- * [4. 读写文件实例](#4-读写文件实例)
- * [5. 阻塞与非阻塞](#5-阻塞与非阻塞)
- * [5.1 阻塞式 I/O](#51-阻塞式-io)
- * [5.2 非阻塞式 I/O](#52-非阻塞式-io)
- * [6. 套接字实例](#6-套接字实例)
- * [6.1 ServerSocketChannel](#61-serversocketchannel)
- * [6.2 Selectors](#62-selectors)
- * [6.3 主循环](#63-主循环)
- * [6.4 监听新连接](#64-监听新连接)
- * [6.5 接受新的连接](#65-接受新的连接)
- * [6.6 删除处理过的 SelectionKey](#66-删除处理过的-selectionkey)
- * [6.7 传入的 I/O](#67-传入的-io)
-* [参考资料](#参考资料)
+* [一、概览](#一概览)
+* [二、磁盘操作](#二磁盘操作)
+* [三、字节操作](#三字节操作)
+* [四、字符操作](#四字符操作)
+* [五、对象操作](#五对象操作)
+* [六、网络操作](#六网络操作)
+ * [InetAddress](#inetaddress)
+ * [URL](#url)
+ * [Sockets](#sockets)
+ * [Datagram](#datagram)
+* [七、NIO](#七nio)
+ * [流与块](#流与块)
+ * [通道与缓冲区](#通道与缓冲区)
+ * [缓冲区状态变量](#缓冲区状态变量)
+ * [文件 NIO 实例](#文件-nio-实例)
+ * [套接字 NIO 实例](#套接字-nio-实例)
+ * [内存映射文件](#内存映射文件)
+ * [对比](#对比)
+* [八、参考资料](#八参考资料)
-# 概览
+# 一、概览
-Java 的 I/O 大概可以分成以下几类
+Java 的 I/O 大概可以分成以下几类:
1. 磁盘操作:File
2. 字节操作:InputStream 和 OutputStream
3. 字符操作:Reader 和 Writer
4. 对象操作:Serializable
5. 网络操作:Socket
-6. 非阻塞式 IO:NIO
+6. 新的输入/输出:NIO
-# 磁盘操作
+# 二、磁盘操作
File 类可以用于表示文件和目录,但是它只用于表示文件的信息,而不表示文件的内容。
-# 字节操作
+# 三、字节操作
-
+
-Java I/O 使用了装饰者模式来实现。以 InputStream 为例,InputStream 是抽象组件,FileInputStream 是 InputStream 的子类,属于具体组件,提供了字节流的输入操作。FilterInputStream 属于抽象装饰者,装饰者用于装饰组件,为组件提供额外的功能,例如 BufferedInputStream 为 FileInputStream 提供缓存的功能。实例化一个具有缓存功能的字节流对象时,只需要在 FileInputStream 对象上再套一层 BufferedInputStream 对象即可。
+Java I/O 使用了装饰者模式来实现。以 InputStream 为例,InputStream 是抽象组件,FileInputStream 是 InputStream 的子类,属于具体组件,提供了字节流的输入操作。FilterInputStream 属于抽象装饰者,装饰者用于装饰组件,为组件提供额外的功能,例如 BufferedInputStream 为 FileInputStream 提供缓存的功能。
+
+实例化一个具有缓存功能的字节流对象时,只需要在 FileInputStream 对象上再套一层 BufferedInputStream 对象即可。
```java
BufferedInputStream bis = new BufferedInputStream(new FileInputStream(file));
@@ -58,7 +50,7 @@ BufferedInputStream bis = new BufferedInputStream(new FileInputStream(file));
DataInputStream 装饰者提供了对更多数据类型进行输入的操作,比如 int、double 等基本类型。
-批量读入文件中的内容到字节数组中
+批量读入文件内容到字节数组:
```java
byte[] buf = new byte[20*1024];
@@ -69,11 +61,11 @@ while((bytes = in.read(buf, 0 , buf.length)) != -1) {
}
```
-# 字符操作
+# 四、字符操作
-不管是磁盘还是网络传输,最小的存储单元都是字节,而不是字符,所以 I/O 操作的都是字节而不是字符。但是在程序中操作的数据通常是字符形式,因此需要提供对字符进行操作的方法。
+不管是磁盘还是网络传输,最小的存储单元都是字节,而不是字符,所以 I/O 操作的都是字节而不是字符。但是在程序中操作的通常是字符形式的数据,因此需要提供对字符进行操作的方法。
-InputStreamReader 实现从文本文件的字节流解码成字符流;OutputStreamWriter 实现字符流编码成为文本文件的字节流。它们都继承自 Reader 和 Writer。
+InputStreamReader 实现从文本文件的字节流解码成字符流;OutputStreamWriter 实现字符流编码成为文本文件的字节流。它们继承自 Reader 和 Writer。
编码就是把字符转换为字节,而解码是把字节重新组合成字符。
@@ -86,7 +78,7 @@ GBK 编码中,中文占 2 个字节,英文占 1 个字节;UTF-8 编码中
如果编码和解码过程使用不同的编码方式那么就出现了乱码。
-# 对象操作
+# 五、对象操作
序列化就是将一个对象转换成字节序列,方便存储和传输。
@@ -100,11 +92,11 @@ transient 关键字可以使一些属性不会被序列化。
**ArrayList 序列化和反序列化的实现** :ArrayList 中存储数据的数组是用 transient 修饰的,因为这个数组是动态扩展的,并不是所有的空间都被使用,因此就不需要所有的内容都被序列化。通过重写序列化和反序列化方法,使得可以只序列化数组中有内容的那部分数据。
-```
+```java
private transient Object[] elementData;
```
-# 网络操作
+# 六、网络操作
Java 中的网络支持:
@@ -113,18 +105,23 @@ Java 中的网络支持:
3. Sockets:使用 TCP 协议实现网络通信;
4. Datagram:使用 UDP 协议实现网络通信。
-## 1. InetAddress
+## InetAddress
-没有公有构造函数,只能通过静态方法来创建实例,比如 InetAddress.getByName(String host)、InetAddress.getByAddress(byte[] addr)。
+没有公有构造函数,只能通过静态方法来创建实例。
-## 2. URL
+```java
+InetAddress.getByName(String host);
+InetAddress.getByAddress(byte[] addr);
+```
+
+## URL
可以直接从 URL 中读取字节流数据
```java
URL url = new URL("http://www.baidu.com");
-InputStream is = url.openStream(); // 字节流
-InputStreamReader isr = new InputStreamReader(is, "utf-8"); // 字符流
+InputStream is = url.openStream(); // 字节流
+InputStreamReader isr = new InputStreamReader(is, "utf-8"); // 字符流
BufferedReader br = new BufferedReader(isr);
String line = br.readLine();
while (line != null) {
@@ -136,43 +133,40 @@ isr.close();
is.close();
```
-## 3. Sockets
-
-Socket 通信模型
-
-
+## Sockets
- ServerSocket:服务器端类
- Socket:客户端类
+- 服务器和客户端通过 InputStream 和 OutputStream 进行输入输出。
-服务器和客户端通过 InputStream 和 OutputStream 进行输入输出。
+
-## 4. Datagram
+## Datagram
- DatagramPacket:数据包类
- DatagramSocket:通信类
-# NIO
+# 七、NIO
-NIO 将最耗时的 I/O 操作 ( 即填充和提取缓冲区 ) 转移回操作系统,因而 不需要程序员去控制就可以极大地提高运行速度。
+新的输入/输出 (NIO) 库是在 JDK 1.4 中引入的。NIO 弥补了原来的 I/O 的不足,它在标准 Java 代码中提供了高速的、面向块的 I/O。
-## 1. 流与块
+## 流与块
-I/O 与 NIO 最重要的区别是数据打包和传输的方式。正如前面提到的,I/O 以流的方式处理数据,而 NIO 以块的方式处理数据。
+I/O 与 NIO 最重要的区别是数据打包和传输的方式,I/O 以流的方式处理数据,而 NIO 以块的方式处理数据。
-面向流的 I/O 一次一个字节进行处理数据,一个输入流产生一个字节的数据,一个输出流消费一个字节的数据。为流式数据创建过滤器非常容易,链接几个过滤器,以便每个过滤器只负责单个复杂处理机制的一部分,这样也是相对简单的。不利的一面是,面向流的 I/O 通常相当慢。
+面向流的 I/O 一次处理一个字节数据,一个输入流产生一个字节数据,一个输出流消费一个字节数据。为流式数据创建过滤器非常容易,链接几个过滤器,以便每个过滤器只负责单个复杂处理机制的一部分,这样也是相对简单的。不利的一面是,面向流的 I/O 通常相当慢。
-一个面向块的 I/O 系统以块的形式处理数据,每一个操作都在一步中产生或者消费一个数据块。按块处理数据比按流处理数据要快得多。但是面向块的 I/O 缺少一些面向流的 I/O 所具有的优雅性和简单性。
+一个面向块的 I/O 系统以块的形式处理数据,一次处理一个数据块。按块处理数据比按流处理数据要快得多。但是面向块的 I/O 缺少一些面向流的 I/O 所具有的优雅性和简单性。
-I/O 包和 NIO 已经很好地集成了,java.io.\* 已经以 NIO 为基础重新实现了,所以现在它可以利用 NIO 的一些特性。例如, java.io.\* 包中的一些类包含以块的形式读写数据的方法,这使得即使在更面向流的系统中,处理速度也会更快。
+I/O 包和 NIO 已经很好地集成了,java.io.\* 已经以 NIO 为基础重新实现了,所以现在它可以利用 NIO 的一些特性。例如,java.io.\* 包中的一些类包含以块的形式读写数据的方法,这使得即使在面向流的系统中,处理速度也会更快。
-## 2. 通道与缓冲区
+## 通道与缓冲区
-### 2.1 通道
+### 1. 通道
通道 Channel 是对原 I/O 包中的流的模拟,可以通过它读取和写入数据。
-通道与流的不同之处在于,流只能在一个方向上移动,(一个流必须是 InputStream 或者 OutputStream 的子类), 而通道是双向的,可以用于读、写或者同时用于读写。
+通道与流的不同之处在于,流只能在一个方向上移动,(一个流必须是 InputStream 或者 OutputStream 的子类),而通道是双向的,可以用于读、写或者同时用于读写。
通道包括以下类型:
@@ -181,9 +175,9 @@ I/O 包和 NIO 已经很好地集成了,java.io.\* 已经以 NIO 为基础重
- SocketChannel:通过 TCP 读写网络中数据;
- ServerSocketChannel:可以监听新进来的 TCP 连接,对每一个新进来的连接都会创建一个 SocketChannel。
-### 2.2 缓冲区
+### 2. 缓冲区
-发送给一个通道的所有对象都必须首先放到缓冲区中;同样地,从通道中读取的任何数据都要读到缓冲区中。也就是说,不会直接对通道进行读写数据,而是先经过缓冲区。
+发送给一个通道的所有数据都必须首先放到缓冲区中,同样地,从通道中读取的任何数据都要读到缓冲区中。也就是说,不会直接对通道进行读写数据,而是要先经过缓冲区。
缓冲区实质上是一个数组,但它不仅仅是一个数组。缓冲区提供了对数据的结构化访问,而且还可以跟踪系统的读/写进程。
@@ -197,51 +191,50 @@ I/O 包和 NIO 已经很好地集成了,java.io.\* 已经以 NIO 为基础重
- FloatBuffer
- DoubleBuffer
-
-## 3. 缓冲区状态变量
+## 缓冲区状态变量
- capacity:最大容量;
- position:当前已经读写的字节数;
- limit:还可以读写的字节数。
-状态变量的改变过程:
+状态变量的改变过程举例:
-1\. 新建一个大小为 8 个字节的缓冲区,此时 position 为 0,而 limit == capacity == 9。capacity 变量不会改变,下面的讨论会忽略它。
+① 新建一个大小为 8 个字节的缓冲区,此时 position 为 0,而 limit = capacity = 8。capacity 变量不会改变,下面的讨论会忽略它。
-2\. 从输入通道中读取 3 个字节数据写入缓冲区中,此时 position 移动设为 3,limit 保持不变。
+② 从输入通道中读取 3 个字节数据写入缓冲区中,此时 position 移动设为 3,limit 保持不变。
-3\. 在将缓冲区的数据写到输出通道之前,需要先调用 flip() 方法,这个方法将 limit 设置为当前 position,并将 position 设置为 0。
+③ 以下图例为已经从输入通道读取了 5 个字节数据写入缓冲区中。在将缓冲区的数据写到输出通道之前,需要先调用 flip() 方法,这个方法将 limit 设置为当前 position,并将 position 设置为 0。
-4\. 从缓冲区中取 4 个字节到输出缓冲中,此时 position 设为 4。
+④ 从缓冲区中取 4 个字节到输出缓冲中,此时 position 设为 4。
-5\. 最后需要调用 clear() 方法来清空缓冲区,此时 position 和 limit 都被设置为最初位置。
+⑤ 最后需要调用 clear() 方法来清空缓冲区,此时 position 和 limit 都被设置为最初位置。
-## 4. 读写文件实例
+## 文件 NIO 实例
-1\. 为要读取的文件创建 FileInputStream,之后通过 FileInputStream 获取输入 FileChannel;
+① 为要读取的文件创建 FileInputStream,之后通过 FileInputStream 获取输入 FileChannel;
```java
FileInputStream fin = new FileInputStream("readandshow.txt");
FileChannel fic = fin.getChannel();
```
-2\. 创建一个容量为 1024 的 Buffer
+② 创建一个容量为 1024 的 Buffer;
```java
ByteBuffer buffer = ByteBuffer.allocate(1024);
```
-3\. 将数据从输入 FileChannel 写入到 Buffer 中,如果没有数据的话, read() 方法会返回 -1
+③ 将数据从输入 FileChannel 写入到 Buffer 中,如果没有数据的话,read() 方法会返回 -1;
```java
int r = fcin.read(buffer);
@@ -250,56 +243,36 @@ if (r == -1) {
}
```
-4\. 为要写入的文件创建 FileOutputStream,之后通过 FileOutputStream 获取输出 FileChannel
+④ 为要写入的文件创建 FileOutputStream,之后通过 FileOutputStream 获取输出 FileChannel
```java
FileOutputStream fout = new FileOutputStream("writesomebytes.txt");
FileChannel foc = fout.getChannel();
```
-5\. 调用 flip() 切换读写
+⑤ 调用 flip() 切换读写
```java
buffer.flip();
```
-6\. 把 Buffer 中的数据读取到输出 FileChannel 中
+⑥ 把 Buffer 中的数据读取到输出 FileChannel 中
```java
foc.write(buffer);
```
-7\. 最后调用 clear() 重置缓冲区
+⑦ 最后调用 clear() 重置缓冲区
```java
buffer.clear();
```
-## 5. 阻塞与非阻塞
+## 套接字 NIO 实例
-应当注意,FileChannel 不能切换到非阻塞模式,套接字 Channel 可以。
+### 1. ServerSocketChannel
-### 5.1 阻塞式 I/O
-
-阻塞式 I/O 在调用 InputStream.read() 方法时会一直等到数据到来时(或超时)才会返回,在调用 ServerSocket.accept() 方法时,也会一直阻塞到有客户端连接才会返回,每个客户端连接过来后,服务端都会启动一个线程去处理该客户端的请求。
-
-
-
-### 5.2 非阻塞式 I/O
-
-由一个专门的线程来处理所有的 I/O 事件,并负责分发。
-
-事件驱动机制:事件到的时候触发,而不是同步的去监视事件。
-
-线程通信:线程之间通过 wait()、notify() 等方式通信,保证每次上下文切换都是有意义的,减少无谓的线程切换。
-
-
-
-## 6. 套接字实例
-
-### 6.1 ServerSocketChannel
-
-每一个端口都需要有一个 ServerSocketChannel 用来监听连接。
+每一个监听端口都需要有一个 ServerSocketChannel 用来监听连接。
```java
ServerSocketChannel ssc = ServerSocketChannel.open();
@@ -310,11 +283,11 @@ InetSocketAddress address = new InetSocketAddress(ports[i]);
ss.bind(address); // 绑定端口号
```
-### 6.2 Selectors
+### 2. Selectors
异步 I/O 通过 Selector 注册对特定 I/O 事件的兴趣 ― 可读的数据的到达、新的套接字连接等等,在发生这样的事件时,系统将会发送通知。
-创建 Selectors 之后,就可以对不同的通道对象调用 register() 方法。register() 的第一个参数总是这个 Selector。第二个参数是 OP_ACCEPT,这里它指定我们想要监听 accept 事件,也就是在新的连接建立时所发生的事件。
+创建 Selectors 之后,就可以对不同的通道对象调用 register() 方法。register() 的第一个参数总是这个 Selector。第二个参数是 OP_ACCEPT,这里它指定我们想要监听 ACCEPT 事件,也就是在新的连接建立时所发生的事件。
SelectionKey 代表这个通道在此 Selector 上的这个注册。当某个 Selector 通知您某个传入事件时,它是通过提供对应于该事件的 SelectionKey 来进行的。SelectionKey 还可以用于取消通道的注册。
@@ -323,27 +296,27 @@ Selector selector = Selector.open();
SelectionKey key = ssc.register(selector, SelectionKey.OP_ACCEPT);
```
-### 6.3 主循环
+### 3. 主循环
-首先,我们调用 Selector 的 select() 方法。这个方法会阻塞,直到至少有一个已注册的事件发生。当一个或者更多的事件发生时, select() 方法将返回所发生的事件的数量。
+首先,我们调用 Selector 的 select() 方法。这个方法会阻塞,直到至少有一个已注册的事件发生。当一个或者更多的事件发生时,select() 方法将返回所发生的事件的数量。
-接下来,我们调用 Selector 的 selectedKeys() 方法,它返回发生了事件的 SelectionKey 对象的一个 集合 。
+接下来,我们调用 Selector 的 selectedKeys() 方法,它返回发生了事件的 SelectionKey 对象的一个集合。
我们通过迭代 SelectionKeys 并依次处理每个 SelectionKey 来处理事件。对于每一个 SelectionKey,您必须确定发生的是什么 I/O 事件,以及这个事件影响哪些 I/O 对象。
```java
int num = selector.select();
-
+
Set selectedKeys = selector.selectedKeys();
Iterator it = selectedKeys.iterator();
-
+
while (it.hasNext()) {
SelectionKey key = (SelectionKey)it.next();
// ... deal with I/O event ...
}
```
-### 6.4 监听新连接
+### 4. 监听新连接
程序执行到这里,我们仅注册了 ServerSocketChannel,并且仅注册它们“接收”事件。为确认这一点,我们对 SelectionKey 调用 readyOps() 方法,并检查发生了什么类型的事件:
@@ -355,9 +328,9 @@ if ((key.readyOps() & SelectionKey.OP_ACCEPT)
}
```
-可以肯定地说, readOps() 方法告诉我们该事件是新的连接。
+可以肯定地说,readOps() 方法告诉我们该事件是新的连接。
-### 6.5 接受新的连接
+### 5. 接受新的连接
因为我们知道这个服务器套接字上有一个传入连接在等待,所以可以安全地接受它;也就是说,不用担心 accept() 操作会阻塞:
@@ -366,16 +339,16 @@ ServerSocketChannel ssc = (ServerSocketChannel)key.channel();
SocketChannel sc = ssc.accept();
```
-下一步是将新连接的 SocketChannel 配置为非阻塞的。而且由于接受这个连接的目的是为了读取来自套接字的数据,所以我们还必须将 SocketChannel 注册到 Selector上,如下所示:
+下一步是将新连接的 SocketChannel 配置为非阻塞的。而且由于接受这个连接的目的是为了读取来自套接字的数据,所以我们还必须将 SocketChannel 注册到 Selector 上,如下所示:
```java
-sc.configureBlocking( false );
-SelectionKey newKey = sc.register( selector, SelectionKey.OP_READ );
+sc.configureBlocking(false);
+SelectionKey newKey = sc.register(selector, SelectionKey.OP_READ);
```
-注意我们使用 register() 的 OP_READ 参数,将 SocketChannel 注册用于 读取 而不是 接受 新连接。
+注意我们使用 register() 的 OP_READ 参数,将 SocketChannel 注册用于读取而不是接受新连接。
-### 6.6 删除处理过的 SelectionKey
+### 6. 删除处理过的 SelectionKey
在处理 SelectionKey 之后,我们几乎可以返回主循环了。但是我们必须首先将处理过的 SelectionKey 从选定的键集合中删除。如果我们没有删除处理过的键,那么它仍然会在主集合中以一个激活的键出现,这会导致我们尝试再次处理它。我们调用迭代器的 remove() 方法来删除处理过的 SelectionKey:
@@ -383,9 +356,9 @@ SelectionKey newKey = sc.register( selector, SelectionKey.OP_READ );
it.remove();
```
-现在我们可以返回主循环并接受从一个套接字中传入的数据(或者一个传入的 I/O 事件)了。
+现在我们可以返回主循环并接受从一个套接字中传入的数据 (或者一个传入的 I/O 事件) 了。
-### 6.7 传入的 I/O
+### 7. 传入的 I/O
当来自一个套接字的数据到达时,它会触发一个 I/O 事件。这会导致在主循环中调用 Selector.select(),并返回一个或者多个 I/O 事件。这一次, SelectionKey 将被标记为 OP_READ 事件,如下所示:
@@ -398,10 +371,34 @@ it.remove();
}
```
+## 内存映射文件
-# 参考资料
+内存映射文件 I/O 是一种读和写文件数据的方法,它可以比常规的基于流或者基于通道的 I/O 快得多。
-- Eckel B, 埃克尔 , 昊鹏 , 等 . Java 编程思想 [M]. 机械工业出版社 , 2002.
+只有文件中实际读取或者写入的部分才会映射到内存中。
+
+现代操作系统一般会根据需要将文件的部分映射为内存的部分,从而实现文件系统。Java 内存映射机制只不过是在底层操作系统中可以采用这种机制时,提供了对该机制的访问。
+
+向内存映射文件写入可能是危险的,仅只是改变数组的单个元素这样的简单操作,就可能会直接修改磁盘上的文件。修改数据与将数据保存到磁盘是没有分开的。
+
+下面代码行将文件的前 1024 个字节映射到内存中,map() 方法返回一个 MappedByteBuffer,它是 ByteBuffer 的子类。因此,您可以像使用其他任何 ByteBuffer 一样使用新映射的缓冲区,操作系统会在需要时负责执行映射。
+
+```java
+MappedByteBuffer mbb = fc.map(FileChannel.MapMode.READ_WRITE, 0, 1024);
+```
+
+## 对比
+
+NIO 与普通 I/O 的区别主要有以下两点:
+
+- NIO 是非阻塞的。应当注意,FileChannel 不能切换到非阻塞模式,套接字 Channel 可以。
+- NIO 面向块,I/O 面向流。
+
+# 八、参考资料
+
+- Eckel B, 埃克尔, 昊鹏, 等. Java 编程思想 [M]. 机械工业出版社, 2002.
- [IBM: NIO 入门](https://www.ibm.com/developerworks/cn/education/java/j-nio/j-nio.html)
-- [ 深入分析 Java I/O 的工作机制 ](https://www.ibm.com/developerworks/cn/java/j-lo-javaio/index.html)
-- [NIO 与传统 IO 的区别 ](http://blog.csdn.net/shimiso/article/details/24990499)
+- [深入分析 Java I/O 的工作机制](https://www.ibm.com/developerworks/cn/java/j-lo-javaio/index.html)
+- [NIO 与传统 IO 的区别](http://blog.csdn.net/shimiso/article/details/24990499)
+- [Decorator Design Pattern](http://stg-tud.github.io/sedc/Lecture/ws13-14/5.3-Decorator.html#mode=document)
+- [Socket Multicast](http://labojava.blogspot.com/2012/12/socket-multicast.html)
diff --git a/notes/Java 基础.md b/notes/Java 基础.md
index c6ade7a6..aa33ce32 100644
--- a/notes/Java 基础.md
+++ b/notes/Java 基础.md
@@ -1,71 +1,91 @@
-* [关键字](#关键字)
- * [1. final](#1-final)
- * [2. static](#2-static)
-* [Object 通用方法](#object-通用方法)
- * [1. 概览](#1-概览)
- * [2. clone()](#2-clone)
- * [3. equals()](#3-equals)
-* [继承](#继承)
- * [1. 访问权限](#1-访问权限)
- * [2. 抽象类与接口的区别](#2-抽象类与接口的区别)
- * [3. super()](#3-super)
-* [String](#string)
- * [1. String, StringBuffer and StringBuilder](#1-string,-stringbuffer-and-stringbuilder)
- * [2. String 不可变的原因](#2-string-不可变的原因)
- * [3. String.intern()](#3-stringintern)
-* [基本类型与运算](#基本类型与运算)
- * [1. 包装类型](#1-包装类型)
- * [2. switch](#2-switch)
-* [反射](#反射)
-* [异常](#异常)
-* [泛型](#泛型)
-* [特性](#特性)
- * [1. 三大特性](#1-三大特性)
- * [2. Java 各版本的新特性](#2-java-各版本的新特性)
- * [3. Java 与 C++ 的区别](#3-java-与-c++-的区别)
- * [4. JRE or JDK](#4-jre-or-jdk)
+* [一、关键字](#一关键字)
+ * [final](#final)
+ * [static](#static)
+* [二、Object 通用方法](#二object-通用方法)
+ * [概览](#概览)
+ * [equals()](#equals)
+ * [hashCode()](#hashcode)
+ * [toString()](#tostring)
+ * [clone()](#clone)
+* [四、继承](#四继承)
+ * [访问权限](#访问权限)
+ * [抽象类与接口](#抽象类与接口)
+ * [super](#super)
+ * [覆盖与重载](#覆盖与重载)
+* [五、String](#五string)
+ * [String, StringBuffer and StringBuilder](#string,-stringbuffer-and-stringbuilder)
+ * [String 不可变的原因](#string-不可变的原因)
+ * [String.intern()](#stringintern)
+* [六、基本类型与运算](#六基本类型与运算)
+ * [包装类型](#包装类型)
+ * [switch](#switch)
+* [七、反射](#七反射)
+* [八、异常](#八异常)
+* [九、泛型](#九泛型)
+* [十、注解](#十注解)
+* [十一、特性](#十一特性)
+ * [面向对象三大特性](#面向对象三大特性)
+ * [Java 各版本的新特性](#java-各版本的新特性)
+ * [Java 与 C++ 的区别](#java-与-c++-的区别)
+ * [JRE or JDK](#jre-or-jdk)
+* [参考资料](#参考资料)
+# 一、关键字
-# 关键字
+## final
-## 1. final
-
-**数据**
+**1. 数据**
声明数据为常量,可以是编译时常量,也可以是在运行时被初始化后不能被改变的常量。
-对于基本类型,final 使数值不变;对于引用对象,final 使引用不变,也就不能引用其它对象,但是被引用的对象本身是可以修改的。
+- 对于基本类型,final 使数值不变;
+- 对于引用类型,final 使引用不变,也就不能引用其它对象,但是被引用的对象本身是可以修改的。
-**方法**
+```java
+final int x = 1;
+// x = 2; // cannot assign value to final variable 'x'
+final A y = new A();
+y.a = 1;
+```
+
+**2. 方法**
声明方法不能被子类覆盖。
-private 方法隐式地被指定为 final,如果在子类中定义的方法和基类中的一个 private 方法签名相同,此时子类的方法不是覆盖基类方法,而是重载了。
+private 方法隐式地被指定为 final,如果在子类中定义的方法和基类中的一个 private 方法签名相同,此时子类的方法不是覆盖基类方法,而是在子类中定义了一个新的方法。
-**类**
+**3. 类**
声明类不允许被继承。
-## 2. static
+## static
-**变量**
+**1. 静态变量**
-静态变量在内存中只存在一份,只在类第一次实例化时初始化一次,同时类所有的实例都共享静态变量,可以直接通过类名来访问它。
+静态变量在内存中只存在一份,只在类第一次实例化时初始化一次。
-但是实例变量则不同,它是伴随着实例的,每创建一个实例就会产生一个实例变量,它与该实例同生共死。
+- 静态变量:类所有的实例都共享静态变量,可以直接通过类名来访问它;
+- 实例变量:每创建一个实例就会产生一个实例变量,它与该实例同生共死。
-**方法**
+```java
+public class A {
+ private int x; // 实例变量
+ public static int y; // 静态变量
+}
+```
-静态方法在类加载的时候就存在了,它不依赖于任何实例,所以 static 方法必须实现,也就是说他不能是抽象方法 abstract。
+**2. 静态方法**
-**静态语句块**
+静态方法在类加载的时候就存在了,它不依赖于任何实例,所以 static 方法必须实现,也就是说它不能是抽象方法(abstract)。
+
+**3. 静态语句块**
静态语句块和静态变量一样在类第一次实例化时运行一次。
-**初始化顺序**
+**4. 初始化顺序**
静态数据优先于其它数据的初始化,静态变量和静态语句块哪个先运行取决于它们在代码中的顺序。
@@ -75,19 +95,19 @@ public static String staticField = "静态变量";
```java
static {
- System.out.println("静态初始化块");
+ System.out.println("静态语句块");
}
```
实例变量和普通语句块的初始化在静态变量和静态语句块初始化结束之后。
```java
-public String field = "变量";
+public String field = "实例变量";
```
```java
{
- System.out.println("初始化块");
+ System.out.println("普通语句块");
}
```
@@ -95,213 +115,754 @@ public String field = "变量";
```java
public InitialOrderTest() {
- System.out.println("构造器");
+ System.out.println("构造函数");
}
```
存在继承的情况下,初始化顺序为:
-1. 父类(静态变量、静态初始化块)
-2. 子类(静态变量、静态初始化块)
-3. 父类(变量、初始化块)
-4. 父类(构造器)
-5. 子类(变量、初始化块)
-6. 子类(构造器)
+1. 父类(静态变量、静态语句块块)
+2. 子类(静态变量、静态语句块)
+3. 父类(实例变量、普通语句块)
+4. 父类(构造函数)
+5. 子类(实例变量、普通语句块)
+6. 子类(构造函数)
+**5. 静态内部类**
-# Object 通用方法
+内部类的一种,静态内部类不依赖外部类,且不能访问外部类的非 static 变量和方法。
-## 1. 概览
+**6. 静态导包**
-- public final native Class> getClass()
-- public native int hashCode()
-- public boolean equals(Object obj)
-- protected native Object clone() throws CloneNotSupportedException
-- public String toString()
-- public final native void notify()
-- public final native void notifyAll()
-- public final native void wait(long timeout) throws InterruptedException
-- public final void wait(long timeout, int nanos) throws InterruptedException
-- public final void wait() throws InterruptedException
-- protected void finalize() throws Throwable { }
+```source-java
+import static com.xxx.ClassName.*
+```
-## 2. clone()
+在使用静态变量和方法时不用再指明 ClassName,从而简化代码,但可读性大大降低。
-**浅拷贝**
+# 二、Object 通用方法
-引用类型引用的是同一个对象,clone() 方法默认就是浅拷贝实现。
+## 概览
-
+```java
+public final native Class> getClass()
-**深拷贝**
+public native int hashCode()
-可以使用序列化实现。
+public boolean equals(Object obj)
-
+protected native Object clone() throws CloneNotSupportedException
-> [How do I copy an object in Java?](https://stackoverflow.com/questions/869033/how-do-i-copy-an-object-in-java)
+public String toString()
-## 3. equals()
+public final native void notify()
-- 对于基本类型,== 就是判断两个值是否相等;
-- 对于引用类型,== 是判断两个引用是否引用同一个对象,而 equals() 是判断引用的对象是否等价。
+public final native void notifyAll()
-等价性:[ 散列 ](https://github.com/CyC2018/InterviewNotes/blob/master/notes/Java%20%E5%AE%B9%E5%99%A8.md#%E6%95%A3%E5%88%97)
+public final native void wait(long timeout) throws InterruptedException
-# 继承
+public final void wait(long timeout, int nanos) throws InterruptedException
-## 1. 访问权限
+public final void wait() throws InterruptedException
+
+protected void finalize() throws Throwable {}
+```
+
+## equals()
+
+**1. equals() 与 == 的区别**
+
+- 对于基本类型,== 判断两个值是否相等,基本类型没有 equals() 方法。
+- 对于引用类型,== 判断两个实例是否引用同一个对象,而 equals() 判断引用的对象是否等价。
+
+```java
+Integer x = new Integer(1);
+Integer y = new Integer(1);
+System.out.println(x.equals(y)); // true
+System.out.println(x == y); // false
+```
+
+**2. 等价关系**
+
+(一)自反性
+
+```java
+x.equals(x); // true
+```
+
+(二)对称性
+
+```java
+x.equals(y) == y.equals(x) // true
+```
+
+(三)传递性
+
+```java
+if(x.equals(y) && y.equals(z)) {
+ x.equals(z); // true;
+}
+```
+
+(四)一致性
+
+多次调用 equals() 方法结果不变
+
+```java
+x.equals(y) == x.equals(y); // true
+```
+
+(五)与 null 的比较
+
+对任何不是 null 的对象 x 调用 x.equals(null) 结果都为 false
+
+```java
+x.euqals(null); // false;
+```
+
+**3. 实现**
+
+- 检查是否为同一个对象的引用,如果是直接返回 true;
+- 检查是否是同一个类型,如果不是,直接返回 false;
+- 将 Object 实例进行转型;
+- 判断每个关键域是否相等。
+
+```java
+public class EqualExample {
+ private int x;
+ private int y;
+ private int z;
+
+ public EqualExample(int x, int y, int z) {
+ this.x = x;
+ this.y = y;
+ this.z = z;
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ if (this == o) return true;
+ if (o == null || getClass() != o.getClass()) return false;
+
+ EqualExample that = (EqualExample) o;
+
+ if (x != that.x) return false;
+ if (y != that.y) return false;
+ return z == that.z;
+ }
+}
+```
+
+## hashCode()
+
+hasCode() 返回散列值,而 equals() 是用来判断两个实例是否等价。等价的两个实例散列值一定要相同,但是散列值相同的两个实例不一定等价。
+
+在覆盖 equals() 方法时应当总是覆盖 hashCode() 方法,保证相等的两个实例散列值也等价。
+
+下面的代码中,新建了两个等价的实例,并将它们添加到 HashSet 中。我们希望将这两个实例当成一样的,只在集合中添加一个实例,但是因为 EqualExample 没有实现 hasCode() 方法,因此这两个实例的散列值是不同的,最终导致集合添加了两个等价的实例。
+
+```java
+EqualExample e1 = new EqualExample(1, 1, 1);
+EqualExample e2 = new EqualExample(1, 1, 1);
+System.out.println(e1.equals(e2)); // true
+HashSet set = new HashSet<>();
+set.add(e1);
+set.add(e2);
+System.out.println(set.size()); // 2
+```
+
+理想的散列函数应当具有均匀性,即不相等的实例应当均匀分不到所有可能的散列值上。这就要求了散列函数要把所有域的值都考虑进来,可以将每个域都当成 R 进制的某一位,然后组成一个 R 进制的整数。R 一般取 31,因为它是一个奇素数,如果是偶数的话,当出现乘法溢出,信息就会丢失,因为与 2 相乘相当于向左移一位。
+
+一个数与 31 相乘可以转换成移位和减法:31\*x == (x<<5)-x。
+
+```java
+@Override
+public int hashCode() {
+ int result = 17;
+ result = 31 * result + x;
+ result = 31 * result + y;
+ result = 31 * result + z;
+ return result;
+}
+```
+
+## toString()
+
+默认返回 ToStringExample@4554617c 这种形式,其中 @ 后面的数值为散列码的无符号十六进制表示。
+
+```java
+public class ToStringExample {
+ private int number;
+
+ public ToStringExample(int number) {
+ this.number = number;
+ }
+}
+```
+
+```java
+ToStringExample example = new ToStringExample(123);
+System.out.println(example.toString());
+```
+
+```html
+ToStringExample@4554617c
+```
+
+## clone()
+
+**1. cloneable**
+
+clone() 是 Object 的受保护方法,这意味着,如果一个类不显式去覆盖 clone() 就没有这个方法。
+
+```java
+public class CloneExample {
+ private int a;
+ private int b;
+}
+```
+
+```java
+CloneExample e1 = new CloneExample();
+CloneExample e2 = e1.clone(); // 'clone()' has protected access in 'java.lang.Object'
+```
+
+接下来覆盖 Object 的 clone() 得到以下实现:
+
+```java
+public class CloneExample {
+ private int a;
+ private int b;
+
+ @Override
+ protected CloneExample clone() throws CloneNotSupportedException {
+ return (CloneExample)super.clone();
+ }
+}
+```
+
+```java
+CloneExample e1 = new CloneExample();
+try {
+ CloneExample e2 = e1.clone();
+} catch (CloneNotSupportedException e) {
+ e.printStackTrace();
+}
+```
+
+```html
+java.lang.CloneNotSupportedException: CloneTest
+```
+
+以上抛出了 CloneNotSupportedException,这是因为 CloneTest 没有实现 Cloneable 接口。
+
+```java
+public class CloneExample implements Cloneable {
+ private int a;
+ private int b;
+
+ @Override
+ protected Object clone() throws CloneNotSupportedException {
+ return super.clone();
+ }
+}
+```
+
+应该注意的是,clone() 方法并不是 Cloneable 接口的方法,而是 Object 的一个 protected 方法。Cloneable 接口只是规定,如果一个类没有实现 Cloneable 接口又调用了 clone() 方法,就会抛出 CloneNotSupportedException。
+
+**2. 深拷贝与浅拷贝**
+
+- 浅拷贝:拷贝实例和原始实例的引用类型引用同一个对象;
+- 深拷贝:拷贝实例和原始实例的引用类型引用不同对象。
+
+```java
+public class ShallowCloneExample implements Cloneable {
+ private int[] arr;
+
+ public ShallowCloneExample() {
+ arr = new int[10];
+ for (int i = 0; i < arr.length; i++) {
+ arr[i] = i;
+ }
+ }
+
+ public void set(int index, int value) {
+ arr[index] = value;
+ }
+
+ public int get(int index) {
+ return arr[index];
+ }
+
+ @Override
+ protected ShallowCloneExample clone() throws CloneNotSupportedException {
+ return (ShallowCloneExample) super.clone();
+ }
+}
+```
+
+```java
+ShallowCloneExample e1 = new ShallowCloneExample();
+ShallowCloneExample e2 = null;
+try {
+ e2 = e1.clone();
+} catch (CloneNotSupportedException e) {
+ e.printStackTrace();
+}
+e1.set(2, 222);
+System.out.println(e2.get(2)); // 222
+```
+
+```java
+public class DeepCloneExample implements Cloneable {
+ private int[] arr;
+
+ public DeepCloneExample() {
+ arr = new int[10];
+ for (int i = 0; i < arr.length; i++) {
+ arr[i] = i;
+ }
+ }
+
+ public void set(int index, int value) {
+ arr[index] = value;
+ }
+
+ public int get(int index) {
+ return arr[index];
+ }
+
+ @Override
+ protected DeepCloneExample clone() throws CloneNotSupportedException {
+ DeepCloneExample result = (DeepCloneExample) super.clone();
+ result.arr = new int[arr.length];
+ for (int i = 0; i < arr.length; i++) {
+ result.arr[i] = arr[i];
+ }
+ return result;
+ }
+}
+```
+
+```java
+DeepCloneExample e1 = new DeepCloneExample();
+DeepCloneExample e2 = null;
+try {
+ e2 = e1.clone();
+} catch (CloneNotSupportedException e) {
+ e.printStackTrace();
+}
+e1.set(2, 222);
+System.out.println(e2.get(2)); // 2
+```
+
+使用 clone() 方法来拷贝一个对象即复杂又有风险,它会抛出异常,并且还需要类型转换。Effective Java 书上讲到,最好不要去使用 clone(),可以使用拷贝构造函数或者拷贝工厂来拷贝一个对象。
+
+```java
+public class CloneConstructorExample {
+ private int[] arr;
+
+ public CloneConstructorExample() {
+ arr = new int[10];
+ for (int i = 0; i < arr.length; i++) {
+ arr[i] = i;
+ }
+ }
+
+ public CloneConstructorExample(CloneConstructorExample original) {
+ arr = new int[original.arr.length];
+ for (int i = 0; i < original.arr.length; i++) {
+ arr[i] = original.arr[i];
+ }
+ }
+
+ public void set(int index, int value) {
+ arr[index] = value;
+ }
+
+ public int get(int index) {
+ return arr[index];
+ }
+}
+```
+
+```java
+CloneConstructorExample e1 = new CloneConstructorExample();
+CloneConstructorExample e2 = new CloneConstructorExample(e1);
+e1.set(2, 222);
+System.out.println(e2.get(2)); // 2
+```
+
+# 四、继承
+
+## 访问权限
Java 中有三个访问权限修饰符:private、protected 以及 public,如果不加访问修饰符,表示包级可见。
-可以对类或类中的成员(字段以及方法)加上访问修饰符。成员可见表示其它类可以用成员所在类的对象访问到该成员;类可见表示其它类可以用这个类创建对象,可以把类当做包中的一个成员,然后包表示一个类,这样就好理解了。
+可以对类或类中的成员(字段以及方法)加上访问修饰符。
-protected 用于修饰成员,表示在继承体系中成员对于子类可见。但是这个访问修饰符对于类没有意义,因为包没有继承体系。
+- 成员可见表示其它类可以用这个类的实例访问到该成员;
+- 类可见表示其它类可以用这个类创建对象。
-更详细的内容:[ 浅析 Java 中的访问权限控制 ](http://www.importnew.com/18097.html)
+protected 用于修饰成员,表示在继承体系中成员对于子类可见,但是这个访问修饰符对于类没有意义。
-## 2. 抽象类与接口的区别
+设计良好的模块会隐藏所有的实现细节,把它的 API 与它的实现清晰地隔离开来。模块之间只通过它们的 API 进行通信,一个模块不需要知道其他模块的内部工作情况,这个概念被称为信息隐藏或封装。因此访问权限应当尽可能地使每个类或者成员不被外界访问。
-抽象类至少包含一个抽象方法,该抽象方法必须在子类中实现。由于抽象类没有抽象方法的具体实现,因此不能对抽象类进行实例化。
+如果子类的方法覆盖了父类的方法,那么子类中该方法的访问级别不允许低于父类的访问级别。这是为了确保可以使用父类实例的地方都可以使用子类实例,也就是确保满足里式替换原则。
+
+字段决不能是公有的,因为这么做的话就失去了对这个实例域修改行为的控制,客户端可以对其随意修改。可以使用共有的 getter 和 setter 方法来替换共有字段。
```java
-public abstract class GenericServlet implements Servlet, ServletConfig, Serializable {
- // abstract method
- abstract void service(ServletRequest req, ServletResponse res);
-
- void init() {
- // Its implementation
- }
- // other method related to Servlet
+public class AccessExample {
+ public int x;
}
```
-接口定义了一组方法,但是接口都没有方法的实现,也就是说这些方法都是抽象方法。
-
```java
-public interface Externalizable extends Serializable {
+public class AccessExample {
+ private int x;
- void writeExternal(ObjectOutput out) throws IOException;
+ public int getX() {
+ return x;
+ }
- void readExternal(ObjectInput in) throws IOException, ClassNotFoundException;
+ public void setX(int x) {
+ this.x = x;
+ }
}
```
-更详细的内容:[Java 抽象类与接口的区别 ](http://www.importnew.com/12399.html)
-
-## 3. super()
-
-用来访问父类的构造函数父类的方法,第二种情况中,子类需要重载父类的方法。
+但是也有例外,如果是包级私有的类或者私有的嵌套类,那么直接暴露成员不会有特别大的影响。
```java
-public class Subclass extends Superclass {
- // overrides printMethod in Superclass
- public void printMethod() {
- super.printMethod();
- System.out.println("Printed in Subclass");
+public class AccessWithInnerClassExample {
+ private class InnerClass {
+ int x;
}
- public static void main(String[] args) {
- Subclass s = new Subclass();
- s.printMethod();
+
+ private InnerClass innerClass;
+
+ public AccessWithInnerClassExample() {
+ innerClass = new InnerClass();
+ }
+
+ public int getValue() {
+ return innerClass.x; // 直接访问
}
}
```
+## 抽象类与接口
+
+**1. 抽象类**
+
+抽象类和抽象方法都使用 abstract 进行声明。抽象类一般会包含抽象方法,抽象方法一定位于抽象类中。
+
+抽象类和普通类最大的区别是,抽象类不能被实例化,需要继承抽象类才能实例化其子类。
+
+```java
+public abstract class AbstractClassExample {
+
+ protected int x;
+ private int y;
+
+ public abstract void func1();
+
+ public void func2() {
+ System.out.println("func2");
+ }
+}
+```
+
+```java
+public class AbstractExtendClassExample extends AbstractClassExample{
+ @Override
+ public void func1() {
+ System.out.println("func1");
+ }
+}
+```
+
+```java
+// AbstractClassExample ac1 = new AbstractClassExample(); // 'AbstractClassExample' is abstract; cannot be instantiated
+AbstractClassExample ac2 = new AbstractExtendClassExample();
+ac2.func1();
+```
+
+**2. 接口**
+
+接口是抽象类的延伸,在 Java 8 之前,它可以看成是一个完全抽象的类,也就是说它不能有任何的方法实现。
+
+从 Java 8 开始,接口也可以拥有默认的方法实现,这是因为不支持默认方法的接口的维护成本太高了。在 Java 8 之前,如果一个接口想要添加新的方法,那么要修改所有实现了该接口的类。
+
+接口也可以包含字段,并且这些字段隐式都是 static 和 final 的。
+
+接口中的方法默认都是 public 的,并且不允许定义为 private 或者 protected。
+
+```java
+public interface InterfaceExample {
+ void func1();
+
+ default void func2(){
+ System.out.println("func2");
+ }
+
+ int x = 123;
+ //int y; // Variable 'y' might not have been initialized
+ public int z = 0; // Modifier 'public' is redundant for interface fields
+ // private int k = 0; // Modifier 'private' not allowed here
+ // protected int l = 0; // Modifier 'protected' not allowed here
+ // private void fun3(); // Modifier 'private' not allowed here
+}
+```
+
+```java
+public class InterfaceImplementExample implements InterfaceExample {
+ @Override
+ public void func1() {
+ System.out.println("func1");
+ }
+}
+```
+
+```java
+// InterfaceExample ie1 = new InterfaceExample(); // 'InterfaceExample' is abstract; cannot be instantiated
+InterfaceExample ie2 = new InterfaceImplementExample();
+ie2.func1();
+System.out.println(InterfaceExample.x);
+```
+
+**3. 比较**
+
+- 从设计层面上看,抽象类提供了一种 IS-A 关系,那么就必须满足里式替换原则,即子类对象必须能够替换掉所有父类对象。而接口更像是一种 LIKE-A 关系,它只是提供一种方法实现契约,并不要求子类和父类具有 IS-A 关系;
+- 从使用上来看,一个类可以实现多个接口,但是不能继承多个抽象类。
+- 接口的字段只能是 static 和 final 类型的,而抽象类的域可以有多种访问权限。
+- 接口的方法只能是 public 的,而抽象类的方法可以由多种访问权限。
+
+**4. 使用选择**
+
+使用抽象类:
+
+- 需要在几个相关的类中共享代码;
+- 需要能控制继承来的方法和域的访问权限,而不是都为 public。
+- 需要继承非静态(non-static)和非常量(non-final)字段。
+
+使用接口:
+
+- 需要让不相关的类都实现一个方法,例如不相关的类都可以实现 Compareable 接口中的 compareTo() 方法;
+- 需要使用多重继承。
+
+在很多情况下,接口优先于抽象类,因为接口没有抽象类严格的类层次接口要求,可以灵活地为一个类添加行为。并且从 Java 8 开始,接口也可以有默认的方法实现,使得修改接口的成本也变的很低。
+
+> [深入理解 abstract class 和 interface](https://www.ibm.com/developerworks/cn/java/l-javainterface-abstract/) [When to Use Abstract Class and Interface](https://dzone.com/articles/when-to-use-abstract-class-and-intreface)
+
+## super
+
+- 访问父类的构造函数:可以使用 super() 函数访问父类的构造函数,从而完成一些初始化的工作。
+- 访问父类的成员:如果子类覆盖了父类的中某个方法的实现,可以通过使用 super 关键字来引用父类的方法实现。
+
+```java
+public class SuperExample {
+ protected int x;
+ protected int y;
+
+ public SuperExample(int x, int y) {
+ this.x = x;
+ this.y = y;
+ }
+
+ public void func() {
+ System.out.println("SuperExample.func()");
+ }
+}
+```
+
+```java
+public class SuperExtendExample extends SuperExample {
+ private int z;
+
+ public SuperExtendExample(int x, int y, int z) {
+ super(x, y);
+ this.z = z;
+ }
+
+ @Override
+ public void func() {
+ super.func();
+ System.out.println("SuperExtendExample.func()");
+ }
+}
+```
+
+```java
+SuperExample e = new SuperExtendExample(1, 2, 3);
+e.func();
+```
+
+```html
+SuperExample.func()
+SuperExtendExample.func()
+```
+
> [Using the Keyword super](https://docs.oracle.com/javase/tutorial/java/IandI/super.html)
-# String
+## 覆盖与重载
-## 1. String, StringBuffer and StringBuilder
+- 覆盖(Override)存在于继承体系中,指子类实现了一个与父类在方法声明上完全相同的一个方法;
-**是否可变**
+- 重载(Overload)存在于同一个类中,指一个方法与已经存在的方法名称上相同,但是参数类型、个数、顺序至少有一个不同。应该注意的是,返回值不同,其它都相同不算是重载。
-String 不可变,StringBuffer 和 StringBuilder 可变。
+# 五、String
-**是否线程安全**
+## String, StringBuffer and StringBuilder
-String 不可变,因此是线程安全的。
+**1. 是否可变**
-StringBuilder 不是线程安全的;StringBuffer 是线程安全的,使用 synchronized 来同步。
+- String 不可变
+- StringBuffer 和 StringBuilder 可变
+
+**2. 是否线程安全**
+
+- String 不可变,因此是线程安全的
+- StringBuilder 不是线程安全的
+- StringBuffer 是线程安全的,内部使用 synchronized 来同步
> [String, StringBuffer, and StringBuilder](https://stackoverflow.com/questions/2971315/string-stringbuffer-and-stringbuilder)
-## 2. String 不可变的原因
+## String 不可变的原因
-**可以缓存 hash 值**
+**1. 可以缓存 hash 值**
-因为 String 的 hash 值经常被使用,例如 String 用做 HashMap 等。不可变的特性可以使得 hash 值也不可变,因此就只需要进行一次计算。
+因为 String 的 hash 值经常被使用,例如 String 用做 HashMap 的 key。不可变的特性可以使得 hash 值也不可变,因此只需要进行一次计算。
-**String Pool 的需要**
+**2. String Pool 的需要**
-如果 String 已经被创建过了,那么就会从 String Pool 中取得引用。只有 String 是不可变的,才可能使用 String Pool。
+如果一个 String 对象已经被创建过了,那么就会从 String Pool 中取得引用。只有 String 是不可变的,才可能使用 String Pool。
-
+
-**安全性**
+**3. 安全性**
-String 经常作为参数,例如网络连接参数等,在作为网络连接参数的情况下,如果 String 是可变的,那么在网络连接过程中,String 被改变,改变 String 对象的那一方以为现在连接的是其它主机,而实际情况却不一定是。String 不可变性可以保证参数不可变。
+String 经常作为参数,String 不可变性可以保证参数不可变。例如在作为网络连接参数的情况下如果 String 是可变的,那么在网络连接过程中,String 被改变,改变 String 对象的那一方以为现在连接的是其它主机,而实际情况却不一定是。
-**线程安全**
+**4. 线程安全**
-String 不可变性天生具备线程安全,可以在多个线程中使用。
+String 不可变性天生具备线程安全,可以在多个线程中安全地使用。
> [Why String is immutable in Java?](https://www.programcreek.com/2013/04/why-string-is-immutable-in-java/)
-## 3. String.intern()
+## String.intern()
-使用 String.intern() 可以保证所有相同内容的字符串变量引用相同的内存对象。
+使用 String.intern() 可以保证相同内容的字符串实例引用相同的内存对象。
-更详细的内容:[ 揭开 String.intern() 那神秘的面纱 ](https://www.jianshu.com/p/95f516cb75ef)
+下面示例中,s1 和 s2 采用 new String() 的方式新建了两个不同对象,而 s3 是通过 s1.intern() 方法取得一个对象引用,这个方法首先把 s1 引用的对象放到 String Poll(字符串常量池)中,然后返回这个对象引用。因此 s3 和 s1 引用的是同一个字符串常量池的对象。
-# 基本类型与运算
+```java
+String s1 = new String("aaa");
+String s2 = new String("aaa");
+System.out.println(s1 == s2); // false
+String s3 = s1.intern();
+System.out.println(s1.intern() == s3); // true
+```
-## 1. 包装类型
+如果是采用 "bbb" 这种使用双引号的形式创建字符串实例,会自动地将新建的对象放入 String Poll 中。
-八个基本类型:boolean 1 byte 8 char 16 short 16 int 32 float 32 long 64 double 64
+```java
+String s4 = "bbb";
+String s5 = "bbb";
+System.out.println(s4 == s5); // true
+```
-基本类型都有对应的包装类型,它们之间的赋值使用自动装箱与拆箱完成。
+Java 虚拟机将堆划分成新生代、老年代和永久代(PermGen Space)。在 Java 7 之前,字符串常量池被放在永久代中,而在 Java 7,它被放在堆的其它位置。这是因为永久代的空间有限,如果大量使用字符串的场景下会导致 OutOfMemoryError 错误。
+
+> [What is String interning?](https://stackoverflow.com/questions/10578984/what-is-string-interning) [深入解析 String#intern](https://tech.meituan.com/in_depth_understanding_string_intern.html)
+
+# 六、基本类型与运算
+
+## 包装类型
+
+八个基本类型:
+
+- boolean/1
+- byte/8
+- char/16
+- short/16
+- int/32
+- float/32
+- long/64
+- double/64
+
+基本类型都有对应的包装类型,基本类型与其对应的包装类型之间的赋值使用自动装箱与拆箱完成。
```java
Integer x = 2; // 装箱
int y = x; // 拆箱
```
-new Integer(123) 与 Integer.valueOf(123) 的区别在于,Integer.valueOf(123) 可能会使用缓存对象,因此多次使用 Integer.valueOf(123) 会取得同一个对象的引用。
+new Integer(123) 与 Integer.valueOf(123) 的区别在于,new Integer(123) 每次都会新建一个对象,而 Integer.valueOf(123) 可能会使用缓存对象,因此多次使用 Integer.valueOf(123) 会取得同一个对象的引用。
```java
-public static void main(String[] args) {
-
- Integer a = new Integer(1);
- Integer b = new Integer(1);
-
- System.out.println("a==b? " + (a==b));
-
- Integer c = Integer.valueOf(1);
- Integer d = Integer.valueOf(1);
-
- System.out.println("c==d? " + (c==d));
-
- }
+Integer x = new Integer(123);
+Integer y = new Integer(123);
+System.out.println(x == y); // false
+Integer z = Integer.valueOf(123);
+Integer k = Integer.valueOf(123);
+System.out.println(z == k); // true
```
-```html
-a==b? false
-c==d? true
+编译器会在自动装箱过程调用 valueOf() 方法,因此多个 Integer 实例使用自动装箱来创建并且值相同,那么就会引用相同的对象。
+
+```java
+Integer m = 123;
+Integer n = 123;
+System.out.println(m == n); // true
```
valueOf() 方法的实现比较简单,就是先判断值是否在缓存池中,如果在的话就直接使用缓存池的内容。
```java
public static Integer valueOf(int i) {
- final int offset = 128;
- if (i >= -128 && i <= 127) { // must cache
- return IntegerCache.cache[i + offset];
- }
+ if (i >= IntegerCache.low && i <= IntegerCache.high)
+ return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
```
-The following is the list of primitives stored as immutable objects:
+在 Java 8 中,Integer 缓存池的大小默认为 -128\~127。
+
+```java
+static final int low = -128;
+static final int high;
+static final Integer cache[];
+
+static {
+ // high value may be configured by property
+ int h = 127;
+ String integerCacheHighPropValue =
+ sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
+ if (integerCacheHighPropValue != null) {
+ try {
+ int i = parseInt(integerCacheHighPropValue);
+ i = Math.max(i, 127);
+ // Maximum array size is Integer.MAX_VALUE
+ h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
+ } catch( NumberFormatException nfe) {
+ // If the property cannot be parsed into an int, ignore it.
+ }
+ }
+ high = h;
+
+ cache = new Integer[(high - low) + 1];
+ int j = low;
+ for(int k = 0; k < cache.length; k++)
+ cache[k] = new Integer(j++);
+
+ // range [-128, 127] must be interned (JLS7 5.1.7)
+ assert IntegerCache.high >= 127;
+}
+```
+
+Java 还将一些其它基本类型的值放在缓冲池中,包含以下这些:
- boolean values true and false
- all byte values
@@ -309,87 +870,87 @@ The following is the list of primitives stored as immutable objects:
- int values between -128 and 127
- char in the range \u0000 to \u007F
-自动装箱过程编译器会调用 valueOf() 方法,因此多个 Integer 对象使用装箱来创建并且值相同,那么就会引用相同的对象,这样做很显然是为了节省内存开销。
-
-```java
-Integer x = 1;
-Integer y = 1;
-System.out.println(c==d); // true
-```
+因此在使用这些基本类型对应的包装类型时,就可以直接使用缓冲池中的对象。
> [Differences between new Integer(123), Integer.valueOf(123) and just 123
](https://stackoverflow.com/questions/9030817/differences-between-new-integer123-integer-valueof123-and-just-123)
-## 2. switch
+## switch
-A switch works with the byte, short, char, and int primitive data types. It also works with enumerated types (discussed in Classes and Inheritance) and a few special classes that "wrap" certain primitive types: Character, Byte, Short, and Integer (discussed in Simple Data Objects).
+从 Java 7 开始,可以在 switch 条件判断语句中使用 String 对象。
-In the JDK 7 release, you can use a String object in the expression of a switch statement.
+```java
+String s = "a";
+switch (s) {
+ case "a":
+ System.out.println("aaa");
+ break;
+ case "b":
+ System.out.println("bbb");
+ break;
+}
+```
-switch 不支持 long,是因为 swicth 的设计初衷是为那些只需要对少数几个值进行等值判断,如果值过于复杂,那么还是用 if 比较合适。
+switch 不支持 long,是因为 swicth 的设计初衷是为那些只需要对少数的几个值进行等值判断,如果值过于复杂,那么还是用 if 比较合适。
+
+```java
+// long x = 111;
+// switch (x) { // Incompatible types. Found: 'long', required: 'char, byte, short, int, Character, Byte, Short, Integer, String, or an enum'
+// case 111:
+// System.out.println(111);
+// break;
+// case 222:
+// System.out.println(222);
+// break;
+// }
+```
> [Why can't your switch statement data type be long, Java?](https://stackoverflow.com/questions/2676210/why-cant-your-switch-statement-data-type-be-long-java)
-switch 使用查找表的方式来实现,JVM 中使用的指令是 lookupswitch。
-
-```java
-public static void main(String... args) {
- switch (1) {
- case 1:
- break;
- case 2:
- break;
- }
-}
-
-public static void main(java.lang.String[]);
- Code:
- Stack=1, Locals=1, Args_size=1
- 0: iconst_1
- 1: lookupswitch{ //2
- 1: 28;
- 2: 31;
- default: 31 }
- 28: goto 31
- 31: return
-```
-
-> [How does Java's switch work under the hood?](https://stackoverflow.com/questions/12020048/how-does-javas-switch-work-under-the-hood)
-
-
-
-
-
-
-# 反射
+# 七、反射
每个类都有一个 **Class** 对象,包含了与类有关的信息。当编译一个新类时,会产生一个同名的 .class 文件,该文件内容保存着 Class 对象。
-类加载相当于 Class 对象的加载。类在第一次使用时才动态加载到 JVM 中,可以使用 Class.forName('com.mysql.jdbc.Driver.class') 这种方式来控制类的加载,该方法会返回一个 Class 对象。
+类加载相当于 Class 对象的加载。类在第一次使用时才动态加载到 JVM 中,可以使用 Class.forName("com.mysql.jdbc.Driver") 这种方式来控制类的加载,该方法会返回一个 Class 对象。
反射可以提供运行时的类信息,并且这个类可以在运行时才加载进来,甚至在编译时期该类的 .class 不存在也可以加载进来。
-Class 和 java.lang.reflect 一起对反射提供了支持,java.lang.reflect 类库包含了 **Field** 、**Method** 以及 **Constructor** 类。可以使用 get() 和 set() 方法读取和修改 Field 对象关联的字段,可以使用 invoke() 方法调用与 Method 对象关联的方法,可以用 Constructor 创建新的对象。
+Class 和 java.lang.reflect 一起对反射提供了支持,java.lang.reflect 类库主要包含了以下三个类:
+
+1. **Field** :可以使用 get() 和 set() 方法读取和修改 Field 对象关联的字段;
+2. **Method** :可以使用 invoke() 方法调用与 Method 对象关联的方法;
+3. **Constructor** :可以用 Constructor 创建新的对象。
IDE 使用反射机制获取类的信息,在使用一个类的对象时,能够把类的字段、方法和构造函数等信息列出来供用户选择。
-更详细的内容:[ 深入解析 Java 反射(1)- 基础 ](http://www.sczyh30.com/posts/Java/java-reflection-1/)
+**Advantages of Using Reflection:**
-# 异常
+- **Extensibility Features** : An application may make use of external, user-defined classes by creating instances of extensibility objects using their fully-qualified names.
+- **Class Browsers and Visual Development Environments** : A class browser needs to be able to enumerate the members of classes. Visual development environments can benefit from making use of type information available in reflection to aid the developer in writing correct code.
+- **Debuggers and Test Tools** : Debuggers need to be able to examine private members on classes. Test harnesses can make use of reflection to systematically call a discoverable set APIs defined on a class, to insure a high level of code coverage in a test suite.
-Throwable 可以用来表示任何可以作为异常抛出的类,分为两种: **Error** 和 **Exception**,其中 Error 用来表示编译时系统错误。
+**Drawbacks of Reflection:**
-Exception 分为两种: **受检异常** 和 **非受检异常**。受检异常需要用 try...catch... 语句捕获并进行处理,并且可以从异常中恢复;非受检异常是程序运行时错误,例如除 0 会引发 Arithmetic Exception,此时程序奔溃并且无法恢复。
+Reflection is powerful, but should not be used indiscriminately. If it is possible to perform an operation without using reflection, then it is preferable to avoid using it. The following concerns should be kept in mind when accessing code via reflection.
-
+- **Performance Overhead** : Because reflection involves types that are dynamically resolved, certain Java virtual machine optimizations can not be performed. Consequently, reflective operations have slower performance than their non-reflective counterparts, and should be avoided in sections of code which are called frequently in performance-sensitive applications.
+- **Security Restrictions** : Reflection requires a runtime permission which may not be present when running under a security manager. This is in an important consideration for code which has to run in a restricted security context, such as in an Applet.
+- **Exposure of Internals** :Since reflection allows code to perform operations that would be illegal in non-reflective code, such as accessing private fields and methods, the use of reflection can result in unexpected side-effects, which may render code dysfunctional and may destroy portability. Reflective code breaks abstractions and therefore may change behavior with upgrades of the platform.
-更详细的内容:
-- [Java 入门之异常处理 ](https://www.tianmaying.com/tutorial/Java-Exception)
-- [Java 异常的面试问题及答案 -Part 1](http://www.importnew.com/7383.html)
+> [Trail: The Reflection API](https://docs.oracle.com/javase/tutorial/reflect/index.html) [深入解析 Java 反射(1)- 基础](http://www.sczyh30.com/posts/Java/java-reflection-1/)
-# 泛型
+# 八、异常
-泛型提供了编译时的类型检测机制,该机制允许程序员在编译时检测到非法的类型。泛型是 Java 中一个非常重要的知识点,在 Java 集合类框架中泛型被广泛应用。
+Throwable 可以用来表示任何可以作为异常抛出的类,分为两种: **Error** 和 **Exception**。其中 Error 用来表示 JVM 无法处理的错误,Exception 分为两种:
+
+1. **受检异常** :需要用 try...catch... 语句捕获并进行处理,并且可以从异常中恢复;
+2. **非受检异常** :是程序运行时错误,例如除 0 会引发 Arithmetic Exception,此时程序奔溃并且无法恢复。
+
+
+
+> [Java 入门之异常处理](https://www.tianmaying.com/tutorial/Java-Exception) [Java 异常的面试问题及答案 -Part 1](http://www.importnew.com/7383.html)
+
+# 九、泛型
```java
public class Box {
@@ -400,20 +961,23 @@ public class Box {
}
```
-更详细的内容:
+> [Java 泛型详解](https://www.ziwenxie.site/2017/03/01/java-generic/)[10 道 Java 泛型面试题](https://cloud.tencent.com/developer/article/1033693)
-- [Java 泛型详解 ](https://www.ziwenxie.site/2017/03/01/java-generic/)
-- [10 道 Java 泛型面试题 ](https://cloud.tencent.com/developer/article/1033693)
+# 十、注解
-# 特性
+Java 注解是附加在代码中的一些元信息,用于一些工具在编译、运行时进行解析和使用,起到说明、配置的功能。注解不会也不能影响代码的实际逻辑,仅仅起到辅助性的作用。
-## 1. 三大特性
+> [注解 Annotation 实现原理与自定义注解例子](https://www.cnblogs.com/acm-bingzi/p/javaAnnotation.html)
-[ 封装、继承、多态 ](https://github.com/CyC2018/InterviewNotes/blob/master/notes/%E9%9D%A2%E5%90%91%E5%AF%B9%E8%B1%A1%E6%80%9D%E6%83%B3.md#%E5%B0%81%E8%A3%85%E7%BB%A7%E6%89%BF%E5%A4%9A%E6%80%81)
+# 十一、特性
-## 2. Java 各版本的新特性
+## 面向对象三大特性
-New highlights in Java SE 8
+> [封装、继承、多态](https://github.com/CyC2018/Interview-Notebook/blob/master/notes/%E9%9D%A2%E5%90%91%E5%AF%B9%E8%B1%A1%E6%80%9D%E6%83%B3.md#%E5%B0%81%E8%A3%85%E7%BB%A7%E6%89%BF%E5%A4%9A%E6%80%81)
+
+## Java 各版本的新特性
+
+**New highlights in Java SE 8**
1. Lambda Expressions
2. Pipelines and Streams
@@ -425,7 +989,7 @@ New highlights in Java SE 8
8. Parallel operations
9. PermGen Error Removed
-New highlights in Java SE 7
+**New highlights in Java SE 7**
1. Strings in Switch Statement
2. Type Inference for Generic Instance Creation
@@ -436,16 +1000,12 @@ New highlights in Java SE 7
7. Binary Literals, Underscore in literals
8. Diamond Syntax
-> [Difference between Java 1.8 and Java 1.7?](http://www.selfgrowth.com/articles/difference-between-java-18-and-java-17)
+> [Difference between Java 1.8 and Java 1.7?](http://www.selfgrowth.com/articles/difference-between-java-18-and-java-17) [Java 8 特性 ](http://www.importnew.com/19345.html)
-更详细的内容:[Java 8 特性 ](http://www.importnew.com/19345.html)
-
-## 3. Java 与 C++ 的区别
+## Java 与 C++ 的区别
Java 是纯粹的面向对象语言,所有的对象都继承自 java.lang.Object,C++ 为了兼容 C 即支持面向对象也支持面向过程。
-比较详细的内容:
-
| Java | C++ |
| -- | -- |
| Java does not support pointers, templates, unions, operator overloading, structures etc. The Java language promoters initially said "No pointers!", but when many programmers questioned how you can work without pointers, the promoters began saying "Restricted pointers." Java supports what it calls "references". References act a lot like pointers in C++ languages but you cannot perform arithmetic on pointers in Java. References have types, and they're type-safe. These references cannot be interpreted as raw address and unsafe conversion is not allowed. | C++ supports structures, unions, templates, operator overloading, pointers and pointer arithmetic.|
@@ -462,7 +1022,12 @@ Java 是纯粹的面向对象语言,所有的对象都继承自 java.lang.Obje
> [What are the main differences between Java and C++?](http://cs-fundamentals.com/tech-interview/java/differences-between-java-and-cpp.php)
-## 4. JRE or JDK
+## JRE or JDK
- JRE is the JVM program, Java application need to run on JRE.
- JDK is a superset of JRE, JRE + tools for developing java programs. e.g, it provides the compiler "javac"
+
+# 参考资料
+
+- Eckel B. Java 编程思想[M]. 机械工业出版社, 2002.
+- Bloch J. Effective java[M]. Addison-Wesley Professional, 2017.
diff --git a/notes/Java 容器.md b/notes/Java 容器.md
index bc2453b3..1cb19fe6 100644
--- a/notes/Java 容器.md
+++ b/notes/Java 容器.md
@@ -1,139 +1,140 @@
-* [概览](#概览)
- * [1. List](#1-list)
- * [2. Set](#2-set)
- * [3. Queue](#3-queue)
- * [4. Map](#4-map)
- * [5. Java 1.0/1.1 容器](#5-java-1011-容器)
-* [容器中的设计模式](#容器中的设计模式)
- * [1. 迭代器模式](#1-迭代器模式)
- * [2. 适配器模式](#2-适配器模式)
-* [散列](#散列)
-* [源码分析](#源码分析)
- * [1. ArraList](#1-arralist)
- * [2. Vector 与 Stack](#2-vector-与-stack)
- * [3. LinkedList](#3-linkedlist)
- * [4. TreeMap](#4-treemap)
- * [5. HashMap](#5-hashmap)
- * [6. LinkedHashMap](#6-linkedhashmap)
- * [7. ConcurrentHashMap](#7-concurrenthashmap)
+* [一、概览](#一概览)
+ * [Collection](#collection)
+ * [Map](#map)
+* [二、容器中的设计模式](#二容器中的设计模式)
+ * [迭代器模式](#迭代器模式)
+ * [适配器模式](#适配器模式)
+* [三、源码分析](#三源码分析)
+ * [ArrayList](#arraylist)
+ * [Vector](#vector)
+ * [LinkedList](#linkedlist)
+ * [TreeMap](#treemap)
+ * [HashMap](#hashmap)
+ * [LinkedHashMap](#linkedhashmap)
+ * [ConcurrentHashMap - JDK 1.7](#concurrenthashmap---jdk-17)
+ * [ConcurrentHashMap - JDK 1.8](#concurrenthashmap---jdk-18)
* [参考资料](#参考资料)
-# 概览
-
-
+# 一、概览
容器主要包括 Collection 和 Map 两种,Collection 又包含了 List、Set 以及 Queue。
-## 1. List
+## Collection
+
+
+
+### 1. Set
+
+- HashSet:基于哈希实现,支持快速查找,但不支持有序性操作,例如根据一个范围查找元素的操作。并且失去了元素的插入顺序信息,也就是说使用 Iterator 遍历 HashSet 得到的结果是不确定的。
+
+- TreeSet:基于红黑树实现,支持有序性操作,但是查找效率不如 HashSet,HashSet 查找时间复杂度为 O(1),TreeSet 则为 O(logN);
+
+- LinkedHashSet:具有 HashSet 的查找效率,且内部使用链表维护元素的插入顺序。
+
+### 2. List
- ArrayList:基于动态数组实现,支持随机访问;
+- Vector:和 ArrayList 类似,但它是线程安全的;
+
- LinkedList:基于双向循环链表实现,只能顺序访问,但是可以快速地在链表中间插入和删除元素。不仅如此,LinkedList 还可以用作栈、队列和双端队列。
-## 2. Set
+### 3. Queue
-- HashSet:基于 Hash 实现,支持快速查找,但是失去有序性;
+- LinkedList:可以用它来支持双向队列;
-- TreeSet:基于红黑树实现,保持有序,但是查找效率不如 HashSet;
+- PriorityQueue:基于堆结构实现,可以用它来实现优先级队列。
-- LinkedHashSet:具有 HashSet 的查找效率,且内部使用链表维护元素的插入顺序,因此具有有序性。
+## Map
-## 3. Queue
+
-只有两个实现:LinkedList 和 PriorityQueue,其中 LinkedList 支持双向队列,PriorityQueue 是基于堆结构实现。
+- HashMap:基于哈希实现;
-## 4. Map
+- HashTable:和 HashMap 类似,但它是线程安全的,这意味着同一时刻多个线程可以同时写入 HashTable 并且不会导致数据不一致。它是遗留类,不应该去使用它。现在可以使用 ConcurrentHashMap 来支持线程安全,并且 ConcurrentHashMap 的效率会更高,因为 ConcurrentHashMap 引入了分段锁。
-- HashMap:基于 Hash 实现
+- LinkedHashMap:使用链表来维护元素的顺序,顺序为插入顺序或者最近最少使用(LRU)顺序。
-- LinkedHashMap:使用链表来维护元素的顺序,顺序为插入顺序或者最近最少使用(LRU)顺序
+- TreeMap:基于红黑树实现。
-- TreeMap:基于红黑树实现
+# 二、容器中的设计模式
-- ConcurrentHashMap:线程安全 Map,不涉及类似于 HashTable 的同步加锁
+## 迭代器模式
-## 5. Java 1.0/1.1 容器
+
-对于旧的容器,我们决不应该使用它们,只需要对它们进行了解。
+Collection 实现了 Iterable 接口,其中的 iterator() 方法能够产生一个 Iterator 对象,通过这个对象就可以迭代遍历 Collection 中的元素。
-- Vector:和 ArrayList 类似,但它是线程安全的
+从 JDK 1.5 之后可以使用 foreach 方法来遍历实现了 Iterable 接口的聚合对象。
-- HashTable:和 HashMap 类似,但它是线程安全的
+```java
+List list = new ArrayList<>();
+list.add("a");
+list.add("b");
+for (String item : list) {
+ System.out.println(item);
+}
+```
-# 容器中的设计模式
-
-## 1. 迭代器模式
-
-从概览图可以看到,每个集合类都有一个 Iterator 对象,可以通过这个迭代器对象来遍历集合中的元素。
-
-[Java 中的迭代器模式 ](https://github.com/CyC2018/InterviewNotes/blob/master/notes/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F.md#92-java-%E5%86%85%E7%BD%AE%E7%9A%84%E8%BF%AD%E4%BB%A3%E5%99%A8)
-
-## 2. 适配器模式
+## 适配器模式
java.util.Arrays#asList() 可以把数组类型转换为 List 类型。
```java
- List list = Arrays.asList(1, 2, 3);
- int[] arr = {1, 2, 3};
- list = Arrays.asList(arr);
+@SafeVarargs
+public static List asList(T... a)
```
-# 散列
+如果要将数组类型转换为 List 类型,应该注意的是 asList() 的参数为泛型的变长参数,因此不能使用基本类型数组作为参数,只能使用相应的包装类型数组。
-使用 hasCode() 来返回散列值,使用的是对象的地址。
+```java
+Integer[] arr = {1, 2, 3};
+List list = Arrays.asList(arr);
+```
-而 equals() 是用来判断两个对象是否相等的,相等的两个对象散列值一定要相同,但是散列值相同的两个对象不一定相等。
+也可以使用以下方式生成 List。
-相等必须满足以下五个性质:
+```java
+List list = Arrays.asList(1,2,3);
+```
-1. 自反性
-2. 对称性
-3. 传递性
-4. 一致性(多次调用 x.equals(y),结果不变)
-5. 对任何不是 null 的对象 x 调用 x.equals(nul) 结果都为 false
+# 三、源码分析
-# 源码分析
+建议先阅读 [算法-查找](https://github.com/CyC2018/Interview-Notebook/blob/master/notes/%E7%AE%97%E6%B3%95.md#%E6%9F%A5%E6%89%BE) 部分,对容器类源码的理解有很大帮助。
-建议先阅读 [ 算法 - 查找 ](https://github.com/CyC2018/InterviewNotes/blob/master/notes/%E7%AE%97%E6%B3%95.md#%E7%AC%AC%E4%B8%89%E7%AB%A0-%E6%9F%A5%E6%89%BE) 部分,对集合类源码的理解有很大帮助。
+至于 ConcurrentHashMap 的理解,需要有并发方面的知识,建议先阅读:[Java 并发](https://github.com/CyC2018/Interview-Notebook/blob/master/notes/Java%20%E5%B9%B6%E5%8F%91.md)
-源码下载:[OpenJDK 1.7](http://download.java.net/openjdk/jdk7)
+以下源码从 JDK 1.8 提取而来,下载地址:[JDK-Source-Code](https://github.com/CyC2018/JDK-Source-Code)。
-## 1. ArraList
+## ArrayList
[ArraList.java](https://github.com/CyC2018/JDK-Source-Code/tree/master/src/ArrayList.java)
+### 1. 概览
+
实现了 RandomAccess 接口,因此支持随机访问,这是理所当然的,因为 ArrayList 是基于数组实现的。
```java
public class ArrayList extends AbstractList
- implements List, RandomAccess, Cloneable, java.io.Serializable
+ implements List, RandomAccess, Cloneable, java.io.Serializable
```
-基于数组实现,保存元素的数组使用 transient 修饰,这是因为该数组不一定所有位置都占满元素,因此也就没必要全部都进行序列化。需要重写 writeObject() 和 readObject()。
+基于数组实现,保存元素的数组使用 transient 修饰,该关键字声明数组默认不会被序列化。ArrayList 具有动态扩容特性,因此保存元素的数组不一定都会被使用,那么就没必要全部进行序列化。ArrayList 重写了 writeObject() 和 readObject() 来控制只序列化数组中有元素填充那部分内容。
```java
-private transient Object[] elementData;
+transient Object[] elementData; // non-private to simplify nested class access
```
-数组的默认大小为 10
+数组的默认大小为 10。
```java
-public ArrayList(int initialCapacity) {
- super();
- if (initialCapacity < 0)
- throw new IllegalArgumentException("Illegal Capacity: "+ initialCapacity);
- this.elementData = new Object[initialCapacity];
-}
-
-public ArrayList() {
- this(10);
-}
+private static final int DEFAULT_CAPACITY = 10;
```
-删除元素时调用 System.arraycopy() 对元素进行复制,因此删除操作成本很高,最好在创建时就指定大概的容量大小,减少复制操作的执行次数。
+删除元素时需要调用 System.arraycopy() 对元素进行复制,因此删除操作成本很高。
```java
public E remove(int index) {
@@ -145,33 +146,23 @@ public E remove(int index) {
int numMoved = size - index - 1;
if (numMoved > 0)
System.arraycopy(elementData, index+1, elementData, index, numMoved);
- elementData[--size] = null; // Let gc do its work
+ elementData[--size] = null; // clear to let GC do its work
return oldValue;
}
```
-添加元素时使用 ensureCapacity() 方法来保证容量足够,如果不够时,需要进行扩容,使得新容量为旧容量的 1.5 倍。
-
-modCount 用来记录 ArrayList 结构发生变化的次数,因为每次在进行 add() 和 addAll() 时都需要调用 ensureCapacity(),因此直接在 ensureCapacity() 中对 modCount 进行修改。
-
-结构发生变化:添加或者删除至少一个元素的所有操作,或者是调整内部数组的大小,仅仅只是设置元素的值不算结构发生变化。
+添加元素时使用 ensureCapacity() 方法来保证容量足够,如果不够时,需要使用 grow() 方法进行扩容,使得新容量为旧容量的 1.5 倍(oldCapacity + (oldCapacity >> 1))。扩容操作需要把原数组整个复制到新数组中,因此最好在创建 ArrayList 对象时就指定大概的容量大小,减少扩容操作的次数。
```java
-public void ensureCapacity(int minCapacity) {
- if (minCapacity > 0)
- ensureCapacityInternal(minCapacity);
-}
-
-private void ensureCapacityInternal(int minCapacity) {
+private void ensureExplicitCapacity(int minCapacity) {
modCount++;
+
// overflow-conscious code
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}
-private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;
-
private void grow(int minCapacity) {
// overflow-conscious code
int oldCapacity = elementData.length;
@@ -183,16 +174,12 @@ private void grow(int minCapacity) {
// minCapacity is usually close to size, so this is a win:
elementData = Arrays.copyOf(elementData, newCapacity);
}
-
-private static int hugeCapacity(int minCapacity) {
- if (minCapacity < 0) // overflow
- throw new OutOfMemoryError();
- return (minCapacity > MAX_ARRAY_SIZE) ?
- Integer.MAX_VALUE :
- MAX_ARRAY_SIZE;
-}
```
+### 2. Fail-Fast
+
+modCount 用来记录 ArrayList 结构发生变化的次数。结构发生变化是指添加或者删除至少一个元素的所有操作,或者是调整内部数组的大小,仅仅只是设置元素的值不算结构发生变化。
+
在进行序列化或者迭代等操作时,需要比较操作前后 modCount 是否改变,如果改变了需要抛出 ConcurrentModificationException。
```java
@@ -202,56 +189,146 @@ private void writeObject(java.io.ObjectOutputStream s)
int expectedModCount = modCount;
s.defaultWriteObject();
- // Write out array length
- s.writeInt(elementData.length);
+ // Write out size as capacity for behavioural compatibility with clone()
+ s.writeInt(size);
// Write out all elements in the proper order.
- for (int i=0; i()); 返回一个线程安全的 ArrayList,也可以使用 concurrent 并发包下的 CopyOnWriteArrayList 类;
+为了获得线程安全的 ArrayList,可以调用 Collections.synchronizedList(new ArrayList<>()); 返回一个线程安全的 ArrayList,也可以使用 concurrent 并发包下的 CopyOnWriteArrayList 类;
-**和 LinkedList 的区别**
+### 4. 和 LinkedList 的区别
-1. ArrayList 基于动态数组实现,LinkedList 基于双向循环链表实现;
-2. ArrayList 支持随机访问,LinkedList 不支持;
-3. LinkedList 在任意位置添加删除元素更快。
+- ArrayList 基于动态数组实现,LinkedList 基于双向循环链表实现;
+- ArrayList 支持随机访问,LinkedList 不支持;
+- LinkedList 在任意位置添加删除元素更快。
-## 2. Vector 与 Stack
+## Vector
[Vector.java](https://github.com/CyC2018/JDK-Source-Code/tree/master/src/Vector.java)
-## 3. LinkedList
+## LinkedList
[LinkedList.java](https://github.com/CyC2018/JDK-Source-Code/tree/master/src/LinkedList.java)
-## 4. TreeMap
+## TreeMap
[TreeMap.java](https://github.com/CyC2018/JDK-Source-Code/tree/master/src/TreeMap.java)
-## 5. HashMap
+## HashMap
[HashMap.java](https://github.com/CyC2018/JDK-Source-Code/tree/master/src/HashMap.java)
-使用拉链法来解决冲突。
+### 1. 存储结构
-默认容量 capacity 为 16,需要注意的是容量必须保证为 2 的次方。容量就是 Entry[] table 数组的长度,size 是数组的实际使用量。
+使用拉链法来解决冲突,内部包含了一个 Entry 类型的数组 table,数组中的每个位置被当成一个桶。
-threshold 规定了一个 size 的临界值,size 必须小于 threshold,如果大于等于,就必须进行扩容操作。
+```java
+transient Entry[] table;
+```
-threshold = capacity * load_factor,其中 load_factor 为 table 数组能够使用的比例,load_factor 过大会导致聚簇的出现,从而影响查询和插入的效率,详见算法笔记。
+其中,Entry 就是存储数据的键值对,它包含了四个字段。从 next 字段我们可以看出 Entry 是一个链表,即每个桶会存放一个链表。
+
+
+
+JDK 1.8 使用 Node 类型存储一个键值对,它依然继承自 Entry,因此可以按照上面的存储结构来理解。
+
+```java
+static class Node implements Map.Entry {
+ final int hash;
+ final K key;
+ V value;
+ Node next;
+
+ Node(int hash, K key, V value, Node next) {
+ this.hash = hash;
+ this.key = key;
+ this.value = value;
+ this.next = next;
+ }
+
+ public final K getKey() { return key; }
+ public final V getValue() { return value; }
+ public final String toString() { return key + "=" + value; }
+
+ public final int hashCode() {
+ return Objects.hashCode(key) ^ Objects.hashCode(value);
+ }
+
+ public final V setValue(V newValue) {
+ V oldValue = value;
+ value = newValue;
+ return oldValue;
+ }
+
+ public final boolean equals(Object o) {
+ if (o == this)
+ return true;
+ if (o instanceof Map.Entry) {
+ Map.Entry,?> e = (Map.Entry,?>)o;
+ if (Objects.equals(key, e.getKey()) &&
+ Objects.equals(value, e.getValue()))
+ return true;
+ }
+ return false;
+ }
+}
+```
+
+### 2. 拉链法的工作原理
+
+```java
+HashMap map = new HashMap<>();
+map.put("K1", "V1");
+map.put("K2", "V2");
+map.put("K3", "V3");
+```
+
+- 新建一个 HashMap,默认大小为 16;
+- 插入 <K1,V1> 键值对,先计算 K1 的 hashCode 为 115,使用除留余数法得到所在的桶下标 115%16=3。
+- 插入 <K2,V2> 键值对,先计算 K2 的 hashCode 为 118,使用除留余数法得到所在的桶下标 118%16=6。
+- 插入 <K3,V3> 键值对,先计算 K3 的 hashCode 为 118,使用除留余数法得到所在的桶下标 118%16=6,插在 <K2,V2> 后面。
+
+
+
+查找需要分成两步进行:
+
+- 计算键值对所在的桶;
+- 在链表上顺序查找,时间复杂度显然和链表的长度成正比。
+
+### 3. 链表转红黑树
+
+应该注意到,从 JDK 1.8 开始,一个桶存储的链表长度大于 8 时会将链表转换为红黑树。
+
+### 4. 扩容
+
+因为从 JDK 1.8 开始引入了红黑树,因此扩容操作较为复杂,为了便于理解,以下内容使用 JDK 1.7 的内容。
+
+设 HashMap 的 table 长度为 M,需要存储的键值对数量为 N,如果哈希函数满足均匀性的要求,那么每条链表的长度大约为 N/M,因此平均查找次数的数量级为 O(N/M)。
+
+为了让查找的成本降低,应该尽可能使得 N/M 尽可能小,因此需要保证 M 尽可能大,也就是说 table 要尽可能大。HashMap 采用动态扩容来根据当前的 N 值来调整 M 值,使得空间效率和时间效率都能得到保证。
+
+和扩容相关的参数主要有:capacity、size、threshold 和 load_factor。
+
+| 参数 | 含义 |
+| :--: | :-- |
+| capacity | table 的容量大小,默认为 16,需要注意的是 capacity 必须保证为 2 的次方。|
+| size | table 的实际使用量。 |
+| threshold | size 的临界值,size 必须小于 threshold,如果大于等于,就必须进行扩容操作。 |
+| load_factor | table 能够使用的比例,threshold = capacity * load_factor。|
```java
static final int DEFAULT_INITIAL_CAPACITY = 16;
@@ -282,83 +359,357 @@ void addEntry(int hash, K key, V value, int bucketIndex) {
}
```
-Entry 用来表示一个键值对元素,其中的 next 指针在序列化时会使用。
+扩容使用 resize() 实现,需要注意的是,扩容操作同样需要把旧 table 的所有键值对重新插入新的 table 中,因此这一步是很费时的。
```java
-static class Entry implements Map.Entry {
- final K key;
- V value;
- Entry next;
- final int hash;
-}
-```
-
-get() 操作需要分成两种情况,key 为 null 和 不为 null,从中可以看出 HashMap 允许插入 null 作为键。
-
-```java
-public V get(Object key) {
- if (key == null)
- return getForNullKey();
- int hash = hash(key.hashCode());
- for (Entry e = table[indexFor(hash, table.length)]; e != null; e = e.next) {
- Object k;
- if (e.hash == hash && ((k = e.key) == key || key.equals(k)))
- return e.value;
+void resize(int newCapacity) {
+ Entry[] oldTable = table;
+ int oldCapacity = oldTable.length;
+ if (oldCapacity == MAXIMUM_CAPACITY) {
+ threshold = Integer.MAX_VALUE;
+ return;
}
- return null;
+
+ Entry[] newTable = new Entry[newCapacity];
+ transfer(newTable);
+ table = newTable;
+ threshold = (int)(newCapacity * loadFactor);
}
-```
-put() 操作也需要根据 key 是否为 null 做不同的处理,需要注意的是如果本来没有 key 为 null 的键值对,新插入一个 key 为 null 的键值对时默认是放在数组的 0 位置,这是因为 null 不能计算 hash 值,也就无法知道应该放在哪个链表上。
-
-```java
-public V put(K key, V value) {
- if (key == null)
- return putForNullKey(value);
- int hash = hash(key.hashCode());
- int i = indexFor(hash, table.length);
- for (Entry e = table[i]; e != null; e = e.next) {
- Object k;
- if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
- V oldValue = e.value;
- e.value = value;
- e.recordAccess(this);
- return oldValue;
+void transfer(Entry[] newTable) {
+ Entry[] src = table;
+ int newCapacity = newTable.length;
+ for (int j = 0; j < src.length; j++) {
+ Entry e = src[j];
+ if (e != null) {
+ src[j] = null;
+ do {
+ Entry next = e.next;
+ int i = indexFor(e.hash, newCapacity);
+ e.next = newTable[i];
+ newTable[i] = e;
+ e = next;
+ } while (e != null);
}
}
-
- modCount++;
- addEntry(hash, key, value, i);
- return null;
}
```
+### 5. 确定桶下标
+
+很多操作都需要先确定一个键值对所在的桶下标,需要分三步进行。
+
+(一)hashCode()
+
+调用 Key 的 hashCode() 方法得到 hashCode。
+
```java
-private V putForNullKey(V value) {
- for (Entry e = table[0]; e != null; e = e.next) {
- if (e.key == null) {
- V oldValue = e.value;
- e.value = value;
- e.recordAccess(this);
- return oldValue;
- }
- }
- modCount++;
- addEntry(0, null, value, 0);
- return null;
+public final int hashCode() {
+ return Objects.hashCode(key) ^ Objects.hashCode(value);
}
```
-## 6. LinkedHashMap
+(二)高位运算
+
+将 hashCode 的高 16 位和低 16 位进行异或操作,使得在数组比较小时,也能保证高低位都参与到了哈希计算中。
+
+```java
+static final int hash(Object key) {
+ int h;
+ return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
+}
+```
+
+(三)除留余数法
+
+令 x = 1<<4,即 x 为 2 的 4 次方,它具有以下性质:
+
+```
+x : 00010000
+x-1 : 00001111
+```
+
+令一个数 y 与 x-1 做与运算,可以去除 y 位级表示的第 4 位及以上数:
+
+```
+y : 10110010
+x-1 : 00001111
+y&(x-1) : 00000010
+```
+
+这个性质和 y 对 x 取模效果是一样的:
+
+```
+x : 00010000
+y : 10110010
+y%x : 00000010
+```
+
+我们知道,位运算的代价比求模运算小的多,因此在进行这种计算时能用位运算的话能带来更高的性能。
+
+拉链法需要使用除留余数法来得到桶下标,也就是需要进行以下计算:hash%capacity,如果能保证 capacity 为 2 的幂次方,那么就可以将这个操作转换位位运算。
+
+以下操作在 Java 8 中没有,但是原理上相同。
+
+```java
+static int indexFor(int h, int length) {
+ return h & (length-1);
+}
+```
+
+### 6. 扩容-重新计算桶下标
+
+在进行扩容时,需要把 Node 重新放到对应的桶上。HashMap 使用了一个特殊的机制,可以降低重新计算桶下标的操作。
+
+假设原数组长度 capacity 为 8,扩容之后 new capacity 为 16:
+
+```html
+capacity : 00010000
+new capacity : 00100000
+```
+
+对于一个 Key,它的 hashCode 如果在第 6 位上为 0,那么除留余数得到的结果和之前一样;如果为 1,那么得到的结果为原来的结果 + 8。
+
+### 7. 扩容-计算数组容量
+
+先考虑如何求一个数的补码,对于 10010000,它的掩码为 11111111,可以使用以下方法得到:
+
+```
+mask |= mask >> 1 11011000
+mask |= mask >> 2 11111100
+mask |= mask >> 4 11111111
+```
+
+如果最后令 mask+1,得到就是大于原始数字的最小的 2 次方。
+
+以下是 HashMap 中计算一个大小所需要的数组容量的代码:
+
+```java
+static final int tableSizeFor(int cap) {
+ int n = cap - 1;
+ n |= n >>> 1;
+ n |= n >>> 2;
+ n |= n >>> 4;
+ n |= n >>> 8;
+ n |= n >>> 16;
+ return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
+}
+```
+
+### 7. null 值
+
+HashMap 允许有一个 Node 的 Key 为 null,该 Node 一定会放在第 0 个桶的位置,因为这个 Key 无法计算 hashCode(),因此只能规定一个桶让它存放。
+
+### 8. 与 HashTable 的区别
+
+- HashTable 是同步的,它使用了 synchronized 来进行同步。它也是线程安全的,多个线程可以共享同一个 HashTable。HashMap 不是同步的,但是可以使用 ConcurrentHashMap,它是 HashTable 的替代,而且比 HashTable 可扩展性更好。
+- HashMap 可以插入键为 null 的 Entry。
+- HashMap 的迭代器是 fail-fast 迭代器,而 Hashtable 的 enumerator 迭代器不是 fail-fast 的。
+- 由于 Hashtable 是线程安全的也是 synchronized,所以在单线程环境下它比 HashMap 要慢。
+- HashMap 不能保证随着时间的推移 Map 中的元素次序是不变的。
+
+## LinkedHashMap
[LinkedHashMap.java](https://github.com/CyC2018/JDK-Source-Code/tree/master/src/HashMap.java)
-## 7. ConcurrentHashMap
+## ConcurrentHashMap - JDK 1.7
-[ConcurrentHashMap.java](https://github.com/CyC2018/JDK-Source-Code/tree/master/src/HashMap.java)
+[ConcurrentHashMap.java](https://github.com/CyC2018/JDK-Source-Code/blob/master/src/1.7/ConcurrentHashMap.java)
-[ 探索 ConcurrentHashMap 高并发性的实现机制 ](https://www.ibm.com/developerworks/cn/java/java-lo-concurrenthashmap/)
+ConcurrentHashMap 和 HashMap 实现上类似,最主要的差别是 ConcurrentHashMap 采用了了分段锁,每个分段锁维护着几个桶,多个线程可以同时访问不同分段锁上的桶。
+
+相比于 HashTable 和用同步包装器包装的 HashMap(Collections.synchronizedMap(new HashMap())),ConcurrentHashMap 拥有更高的并发性。在 HashTable 和由同步包装器包装的 HashMap 中,使用一个全局的锁来同步不同线程间的并发访问。同一时间点,只能有一个线程持有锁,也就是说在同一时间点,只能有一个线程能访问容器。这虽然保证多线程间的安全并发访问,但同时也导致对容器的访问变成串行化的了。
+
+### 1. 存储结构
+
+和 HashMap 类似。
+
+```java
+static final class HashEntry {
+ final int hash;
+ final K key;
+ volatile V value;
+ volatile HashEntry next;
+}
+```
+
+继承自 ReentrantLock,每个 Segment 维护着多个 HashEntry。
+
+```java
+static final class Segment extends ReentrantLock implements Serializable {
+
+ private static final long serialVersionUID = 2249069246763182397L;
+
+ static final int MAX_SCAN_RETRIES =
+ Runtime.getRuntime().availableProcessors() > 1 ? 64 : 1;
+
+ transient volatile HashEntry[] table;
+
+ transient int count;
+
+ transient int modCount;
+
+ transient int threshold;
+
+ final float loadFactor;
+}
+```
+
+```java
+final Segment[] segments;
+```
+
+默认的并发级别为 16,也就是说默认创建 16 个 Segment。
+
+```java
+static final int DEFAULT_CONCURRENCY_LEVEL = 16;
+```
+
+
+
+### 2. HashEntry 的不可变性
+
+HashEntry 中的 key,hash,next 都声明为 final 型。这意味着,不能把节点添加到链接的中间和尾部,也不能在链接的中间和尾部删除节点。这个特性可以保证:在访问某个节点时,这个节点之后的链接不会被改变。这个特性可以大大降低处理链表时的复杂性。
+
+同时,HashEntry 类的 value 域被声明为 Volatile 型,Java 的内存模型可以保证:某个写线程对 value 域的写入马上可以被后续的某个读线程 “看” 到。在 ConcurrentHashMap 中,不允许用 null 作为键和值,当读线程读到某个 HashEntry 的 value 域的值为 null 时,便知道产生了冲突——发生了重排序现象,需要加锁后重新读入这个 value 值。这些特性互相配合,使得读线程即使在不加锁状态下,也能正确访问 ConcurrentHashMap。
+
+```java
+final V remove(Object key, int hash, Object value) {
+ if (!tryLock())
+ scanAndLock(key, hash);
+ V oldValue = null;
+ try {
+ HashEntry[] tab = table;
+ int index = (tab.length - 1) & hash;
+ HashEntry e = entryAt(tab, index);
+ HashEntry pred = null;
+ while (e != null) {
+ K k;
+ HashEntry next = e.next;
+ if ((k = e.key) == key ||
+ (e.hash == hash && key.equals(k))) {
+ V v = e.value;
+ if (value == null || value == v || value.equals(v)) {
+ if (pred == null)
+ setEntryAt(tab, index, next);
+ else
+ pred.setNext(next);
+ ++modCount;
+ --count;
+ oldValue = v;
+ }
+ break;
+ }
+ pred = e;
+ e = next;
+ }
+ } finally {
+ unlock();
+ }
+ return oldValue;
+}
+```
+
+在以下链表中删除 C 节点,C 节点之后的所有节点都原样保留,C 节点之前的所有节点都被克隆到新的链表中,并且顺序被反转。可以注意到,在执行 remove 操作时,原始链表并没有被修改,也就是说,读线程不会受到执行 remove 操作的并发写线程的干扰。
+
+
+
+
+
+除了 remove 操作,其它操作也类似。可以得出一个结论:写线程对某个链表的结构性修改不会影响其他的并发读线程对这个链表的遍历访问。
+
+### 3. Volatile 变量
+
+```java
+static final class HashEntry {
+ final int hash;
+ final K key;
+ volatile V value;
+ volatile HashEntry next;
+}
+```
+
+由于内存可见性问题,未正确同步的情况下,写线程写入的值可能并不为后续的读线程可见。
+
+下面以写线程 M 和读线程 N 来说明 ConcurrentHashMap 如何协调读 / 写线程间的内存可见性问题。
+
+
+
+假设线程 M 在写入了 volatile 型变量 count 后,线程 N 读取了这个 volatile 型变量 count。
+
+根据 happens-before 关系法则中的程序次序法则,A appens-before 于 B,C happens-before D。
+
+根据 Volatile 变量法则,B happens-before C。
+
+根据传递性,连接上面三个 happens-before 关系得到:A appens-before 于 B; B appens-before C;C happens-before D。也就是说:写线程 M 对链表做的结构性修改,在读线程 N 读取了同一个 volatile 变量后,对线程 N 也是可见的了。
+
+虽然线程 N 是在未加锁的情况下访问链表。Java 的内存模型可以保证:只要之前对链表做结构性修改操作的写线程 M 在退出写方法前写 volatile 型变量 count,读线程 N 在读取这个 volatile 型变量 count 后,就一定能 “看到” 这些修改。
+
+ConcurrentHashMap 中,每个 Segment 都有一个变量 count。它用来统计 Segment 中的 HashEntry 的个数。这个变量被声明为 volatile。
+
+```java
+transient volatile int count;
+```
+
+所有不加锁读方法,在进入读方法时,首先都会去读这个 count 变量。比如下面的 get 方法:
+
+```java
+V get(Object key, int hash) {
+ if(count != 0) { // 首先读 count 变量
+ HashEntry e = getFirst(hash);
+ while(e != null) {
+ if(e.hash == hash && key.equals(e.key)) {
+ V v = e.value;
+ if(v != null)
+ return v;
+ // 如果读到 value 域为 null,说明发生了重排序,加锁后重新读取
+ return readValueUnderLock(e);
+ }
+ e = e.next;
+ }
+ }
+ return null;
+}
+```
+
+在 ConcurrentHashMap 中,所有执行写操作的方法(put, remove, clear),在对链表做结构性修改之后,在退出写方法前都会去写这个 count 变量。所有未加锁的读操作(get, contains, containsKey)在读方法中,都会首先去读取这个 count 变量。
+
+根据 Java 内存模型,对 同一个 volatile 变量的写 / 读操作可以确保:写线程写入的值,能够被之后未加锁的读线程 “看到”。
+
+这个特性和前面介绍的 HashEntry 对象的不变性相结合,使得在 ConcurrentHashMap 中,读线程在读取散列表时,基本不需要加锁就能成功获得需要的值。这两个特性相配合,不仅减少了请求同一个锁的频率(读操作一般不需要加锁就能够成功获得值),也减少了持有同一个锁的时间(只有读到 value 域的值为 null 时 ,读线程才需要加锁后重读)。
+
+### 4. 小结
+
+ConcurrentHashMap 的高并发性主要来自于三个方面:
+
+- 用分离锁实现多个线程间的更深层次的共享访问。
+- 用 HashEntery 对象的不变性来降低执行读操作的线程在遍历链表期间对加锁的需求。
+- 通过对同一个 Volatile 变量的写 / 读访问,协调不同线程间读 / 写操作的内存可见性。
+
+## ConcurrentHashMap - JDK 1.8
+
+[ConcurrentHashMap.java](https://github.com/CyC2018/JDK-Source-Code/blob/master/src/ConcurrentHashMap.java)
+
+JDK 1.7 分段锁机制来实现并发更新操作,核心类为 Segment,它继承自重入锁 ReentrantLock。
+
+JDK 1.8 的实现不是用了 Segment,Segment 属于重入锁 ReentrantLock。而是使用了内置锁 synchronized,主要是出于以下考虑:
+
+1. synchronized 的锁粒度更低;
+2. synchronized 优化空间更大;
+3. 在大量数据操作的情况下,ReentrantLock 会开销更多的内存。
+
+并且 JDK 1.8 的实现也在链表过长时会转换为红黑树。
# 参考资料
-- Java 编程思想
+- Eckel B. Java 编程思想 [M]. 机械工业出版社, 2002.
+- [Java Collection Framework](https://www.w3resource.com/java-tutorial/java-collections.php)
+- [Iterator 模式](https://openhome.cc/Gossip/DesignPattern/IteratorPattern.htm)
+- [Java 8 系列之重新认识 HashMap](https://tech.meituan.com/java-hashmap.html)
+- [What is difference between HashMap and Hashtable in Java?](http://javarevisited.blogspot.hk/2010/10/difference-between-hashmap-and.html)
+- [Java 集合之 HashMap](http://www.zhangchangle.com/2018/02/07/Java%E9%9B%86%E5%90%88%E4%B9%8BHashMap/)
+- [The principle of ConcurrentHashMap analysis](http://www.programering.com/a/MDO3QDNwATM.html)
+- [探索 ConcurrentHashMap 高并发性的实现机制](https://www.ibm.com/developerworks/cn/java/java-lo-concurrenthashmap/)
+- [HashMap 相关面试题及其解答](https://www.jianshu.com/p/75adf47958a7)
+- [Java 集合细节(二):asList 的缺陷](http://wiki.jikexueyuan.com/project/java-enhancement/java-thirtysix.html)
+
diff --git a/notes/Java 并发.md b/notes/Java 并发.md
index 7b13cff7..a1131c9c 100644
--- a/notes/Java 并发.md
+++ b/notes/Java 并发.md
@@ -1,451 +1,1658 @@
-* [使用线程](#使用线程)
- * [1. 实现 Runnable 接口](#1-实现-runnable-接口)
- * [2. 实现 Callable 接口](#2-实现-callable-接口)
- * [3. 继承 Tread 类](#3-继承-tread-类)
- * [4. 实现接口 vs 继承 Thread](#4-实现接口-vs-继承-thread)
-* [Executor](#executor)
-* [基础线程机制](#基础线程机制)
- * [1. sleep()](#1-sleep)
- * [2. yield()](#2-yield)
- * [3. join()](#3-join)
- * [4. deamon](#4-deamon)
-* [线程之间的协作](#线程之间的协作)
- * [1. 线程通信](#1-线程通信)
- * [2. 线程同步](#2-线程同步)
- * [2.1 synchronized](#21-synchronized)
- * [2.2 Lock](#22-lock)
- * [2.3 BlockingQueue](#23-blockingqueue)
-* [线程状态](#线程状态)
-* [结束线程](#结束线程)
- * [1. 阻塞](#1-阻塞)
- * [2. 中断](#2-中断)
-* [原子性](#原子性)
-* [volatile](#volatile)
- * [1. 内存可见性](#1-内存可见性)
- * [2. 禁止指令重排](#2-禁止指令重排)
-* [多线程开发良好的实践](#多线程开发良好的实践)
-* [未完待续](#未完待续)
+* [一、线程状态转换](#一线程状态转换)
+ * [新建(New)](#新建new)
+ * [可运行(Runnable)](#可运行runnable)
+ * [阻塞(Blocking)](#阻塞blocking)
+ * [无限期等待(Waiting)](#无限期等待waiting)
+ * [限期等待(Timed Waiting)](#限期等待timed-waiting)
+ * [死亡(Terminated)](#死亡terminated)
+* [二、使用线程](#二使用线程)
+ * [实现 Runnable 接口](#实现-runnable-接口)
+ * [实现 Callable 接口](#实现-callable-接口)
+ * [继承 Thread 类](#继承-thread-类)
+ * [实现接口 VS 继承 Thread](#实现接口-vs-继承-thread)
+* [三、基础线程机制](#三基础线程机制)
+ * [Executor](#executor)
+ * [Daemon](#daemon)
+ * [sleep()](#sleep)
+ * [yield()](#yield)
+* [四、中断](#四中断)
+ * [InterruptedException](#interruptedexception)
+ * [interrupted()](#interrupted)
+ * [Executor 的中断操作](#executor-的中断操作)
+* [五、互斥同步](#五互斥同步)
+ * [synchronized](#synchronized)
+ * [ReentrantLock](#reentrantlock)
+ * [synchronized 和 ReentrantLock 比较](#synchronized-和-reentrantlock-比较)
+* [六、线程之间的协作](#六线程之间的协作)
+ * [join()](#join)
+ * [wait() notify() notifyAll()](#wait-notify-notifyall)
+ * [await() signal() signalAll()](#await-signal-signalall)
+* [七、J.U.C - AQS](#七juc---aqs)
+ * [CountdownLatch](#countdownlatch)
+ * [CyclicBarrier](#cyclicbarrier)
+ * [Semaphore](#semaphore)
+* [八、J.U.C - 其它组件](#八juc---其它组件)
+ * [FutureTask](#futuretask)
+ * [BlockingQueue](#blockingqueue)
+ * [ForkJoin](#forkjoin)
+* [九、线程不安全示例](#九线程不安全示例)
+* [十、Java 内存模型](#十java-内存模型)
+ * [主内存与工作内存](#主内存与工作内存)
+ * [内存间交互操作](#内存间交互操作)
+ * [内存模型三大特性](#内存模型三大特性)
+ * [先行发生原则](#先行发生原则)
+* [十一、线程安全](#十一线程安全)
+ * [线程安全分类](#线程安全分类)
+ * [线程安全的实现方法](#线程安全的实现方法)
+* [十二、锁优化](#十二锁优化)
+ * [自旋锁与自适应自旋](#自旋锁与自适应自旋)
+ * [锁消除](#锁消除)
+ * [锁粗化](#锁粗化)
+ * [轻量级锁](#轻量级锁)
+ * [偏向锁](#偏向锁)
+* [十三、多线程开发良好的实践](#十三多线程开发良好的实践)
* [参考资料](#参考资料)
-# 使用线程
+# 一、线程状态转换
+
+
+
+## 新建(New)
+
+创建后尚未启动。
+
+## 可运行(Runnable)
+
+可能正在运行,也可能正在等待 CPU 时间片。
+
+包含了操作系统线程状态中的 Running 和 Ready。
+
+## 阻塞(Blocking)
+
+等待获取一个排它锁,如果其线程释放了锁就会结束此状态。
+
+## 无限期等待(Waiting)
+
+等待其它线程显式地唤醒,否则不会被分配 CPU 时间片。
+
+| 进入方法 | 退出方法 |
+| --- | --- |
+| 没有设置 Timeout 参数的 Object.wait() 方法 | Object.notify() / Object.notifyAll() |
+| 没有设置 Timeout 参数的 Thread.join() 方法 | 被调用的线程执行完毕 |
+| LockSupport.park() 方法 | - |
+
+## 限期等待(Timed Waiting)
+
+无需等待其它线程显示地唤醒,在一定时间之后会被系统自动唤醒。
+
+调用 Thread.sleep() 方法使线程进入限期等待状态时,常常用“使一个线程睡眠”进行描述。
+
+调用 Object.wait() 方法使线程进入限期等待或者无限期等待时,常常用“挂起一个线程”进行描述。
+
+| 进入方法 | 退出方法 |
+| --- | --- |
+| Thread.sleep() 方法 | 时间结束 |
+| 设置了 Timeout 参数的 Object.wait() 方法 | 时间结束 / Object.notify() / Object.notifyAll() |
+| 设置了 Timeout 参数的 Thread.join() 方法 | 时间结束 / 被调用的线程执行完毕 |
+| LockSupport.parkNanos() 方法 | - |
+| LockSupport.parkUntil() 方法 | - |
+
+## 死亡(Terminated)
+
+可以是线程结束任务之后自己结束,或者产生了异常而结束。
+
+# 二、使用线程
有三种使用线程的方法:
1. 实现 Runnable 接口;
2. 实现 Callable 接口;
-3. 继承 Tread 类;
+3. 继承 Thread 类。
实现 Runnable 和 Callable 接口的类只能当做一个可以在线程中运行的任务,不是真正意义上的线程,因此最后还需要通过 Thread 来调用。可以说任务是通过线程驱动从而执行的。
+## 实现 Runnable 接口
-## 1. 实现 Runnable 接口
+需要实现 run() 方法。
-需要实现 run() 方法
-
-通过 Thread 调用 start() 方法来启动线程
+通过 Thread 调用 start() 方法来启动线程。
```java
public class MyRunnable implements Runnable {
public void run() {
// ...
}
- public static void main(String[] args) {
- MyRunnable instance = new MyRunnable();
- Tread thread = new Thread(instance);
- thread.start();
- }
}
```
-## 2. 实现 Callable 接口
+```java
+public static void main(String[] args) {
+ MyRunnable instance = new MyRunnable();
+ Thread thread = new Thread(instance);
+ thread.start();
+}
+```
+
+## 实现 Callable 接口
与 Runnable 相比,Callable 可以有返回值,返回值通过 FutureTask 进行封装。
```java
-public class MyCallable implements Callable {
+public class MyCallable implements Callable {
public Integer call() {
- // ...
- }
- public static void main(String[] args) {
- MyCallable mc = new MyCallable();
- FutureTask ft = new FutureTask<>(mc);
- Thread thread = new Thread(ft);
- thread.start();
- System.out.println(ft.get());
+ return 123;
}
}
```
-## 3. 继承 Tread 类
+```java
+public static void main(String[] args) throws ExecutionException, InterruptedException {
+ MyCallable mc = new MyCallable();
+ FutureTask ft = new FutureTask<>(mc);
+ Thread thread = new Thread(ft);
+ thread.start();
+ System.out.println(ft.get());
+}
+```
+
+## 继承 Thread 类
同样也是需要实现 run() 方法,并且最后也是调用 start() 方法来启动线程。
```java
-class MyThread extends Thread {
+public class MyThread extends Thread {
public void run() {
// ...
}
- public static void main(String[] args) {
- MyThread mt = new MyThread();
- mt.start();
- }
}
```
-## 4. 实现接口 vs 继承 Thread
+```java
+public static void main(String[] args) {
+ MyThread mt = new MyThread();
+ mt.start();
+}
+```
+
+## 实现接口 VS 继承 Thread
实现接口会更好一些,因为:
-1. Java 不支持多重继承,因此继承了 Thread 类就无法继承其它类,但是可以实现多个接口。
-2. 类可能只要求可执行即可,继承整个 Thread 类开销会过大。
+1. Java 不支持多重继承,因此继承了 Thread 类就无法继承其它类,但是可以实现多个接口;
+2. 类可能只要求可执行就行,继承整个 Thread 类开销会过大。
-# Executor
+# 三、基础线程机制
+
+## Executor
Executor 管理多个异步任务的执行,而无需程序员显示地管理线程的生命周期。
-主要有三种 Excutor:
+主要有三种 Executor:
1. CachedTreadPool:一个任务创建一个线程;
2. FixedThreadPool:所有任务只能使用固定大小的线程;
3. SingleThreadExecutor:相当于大小为 1 的 FixedThreadPool。
```java
-ExecutorService exec = Executors.newCachedThreadPool();
-for(int i = 0; i < 5; i++) {
- exec.execute(new MyRunnable());
+public static void main(String[] args) {
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ for (int i = 0; i < 5; i++) {
+ executorService.execute(new MyRunnable());
+ }
+ executorService.shutdown();
}
```
-# 基础线程机制
+## Daemon
-## 1. sleep()
+守护线程是程序运行时在后台提供服务的线程,不属于程序中不可或缺的部分。
-**Thread.sleep(millisec)** 方法会休眠当前正在执行的线程,millisec 单位为毫秒。也可以使用 TimeUnit.TILLISECONDS.sleep(millisec)。
+当所有非守护线程结束时,程序也就终止,同时会杀死所有守护线程。
-sleep() 可能会抛出 InterruptedException。因为异常不能跨线程传播回 main() 中,因此必须在本地进行处理。线程中抛出的其它异常也同样需要在本地进行处理。
+main() 属于非守护线程。
+
+使用 setDaemon() 方法将一个线程设置为守护线程。
+
+```java
+public static void main(String[] args) {
+ Thread thread = new Thread(new MyRunnable());
+ thread.setDaemon(true);
+}
+```
+
+## sleep()
+
+Thread.sleep(millisec) 方法会休眠当前正在执行的线程,millisec 单位为毫秒。
+
+sleep() 可能会抛出 InterruptedException,因为异常不能跨线程传播回 main() 中,因此必须在本地进行处理。线程中抛出的其它异常也同样需要在本地进行处理。
```java
public void run() {
try {
- // ...
- Thread.sleep(1000);
- // ...
- } catch(InterruptedException e) {
- System.err.println(e);
+ Thread.sleep(3000);
+ } catch (InterruptedException e) {
+ e.printStackTrace();
}
}
```
-## 2. yield()
+## yield()
-对静态方法 **Thread.yield()** 的调用声明了当前线程已经完成了生命周期中最重要的部分,可以切换给其它线程来执行。
+对静态方法 Thread.yield() 的调用声明了当前线程已经完成了生命周期中最重要的部分,可以切换给其它线程来执行。该方法只是对线程调度器的一个建议,而且也只是建议具有相同优先级的其它线程可以运行。
```java
public void run() {
- // ...
Thread.yield();
}
```
-## 3. join()
+# 四、中断
-在线程中调用另一个线程的 **join()** 方法,会将当前线程挂起,直到目标线程结束。
+一个线程执行完毕之后会自动结束,如果在运行过程中发生异常也会提前结束。
-可以加一个超时参数。
+## InterruptedException
-## 4. deamon
+通过调用一个线程的 interrupt() 来中断该线程,如果该线程处于阻塞、限期等待或者无限期等待状态,那么就会抛出 InterruptedException,从而提前结束该线程。但是不能中断 I/O 阻塞和 synchronized 锁阻塞。
-后台线程( **deamon** )是程序运行时在后台提供服务的线程,并不属于程序中不可或缺的部分。
-
-当所有非后台线程结束时,程序也就终止,同时会杀死所有后台线程。
-
-main() 属于非后台线程。
-
-使用 setDaemon() 方法将一个线程设置为后台线程。
-
-# 线程之间的协作
-
-- **线程通信** :保证线程以一定的顺序执行;
-- **线程同步** :保证线程对临界资源的互斥访问。
-
-线程通信往往是基于线程同步的基础上完成的,因此很多线程通信问题也是线程同步问题。
-
-## 1. 线程通信
-
-**wait()、notify() 和 notifyAll()** 三者实现了线程之间的通信。
-
-wait() 会在等待时将线程挂起,而不是忙等待,并且只有在 notify() 或者 notifyAll() 到达时才唤醒。
-
-sleep() 和 yield() 并没有释放锁,但是 wait() 会释放锁。实际上,只有在同步控制方法或同步控制块里才能调用 wait() 、notify() 和 notifyAll()。
-
-这几个方法属于基类的一部分,而不属于 Thread。
+对于以下代码,在 Main 中启动一个线程之后再中断它,由于线程中调用了 Thread.sleep() 方法,因此会抛出一个 InterruptedException,从而提前结束线程,不执行之后的语句。
```java
-private boolean flag = false;
-
-public synchronized void after() {
- while(flag == false) {
- wait();
- // ...
- }
-}
-
-public synchronized void before() {
- flag = true;
- notifyAll();
-}
-```
-
-**wait() 和 sleep() 的区别**
-
-1. wait() 是 Object 类的方法,而 sleep() 是 Thread 的静态方法;
-2. wait() 会放弃锁,而 sleep() 不会。
-
-## 2. 线程同步
-
-给定一个进程内的所有线程,都共享同一存储空间,这样有好处又有坏处。这些线程就可以共享数据,非常有用。不过,在两个线程同时修改某一资源时,这也会造成一些问题。Java 提供了同步机制,以控制对共享资源的互斥访问。
-
-### 2.1 synchronized
-
-**同步一个方法**
-
-使多个线程不能同时访问该方法。
-
-```java
-public synchronized void func(String name) {
- // ...
-}
-```
-
-**同步一个代码块**
-
-```java
-public void func(String name) {
- synchronized(this) {
- // ...
- }
-}
-```
-
-### 2.2 Lock
-
-若要实现更细粒度的控制,我们可以使用锁(lock)。
-
-```java
-private Lock lock;
-public int func(int value) {
- lock.lock();
- // ...
- lock.unlock();
-}
-```
-
-### 2.3 BlockingQueue
-
-java.util.concurrent.BlockingQueue 接口有以下阻塞队列的实现:
-
-- **FIFO 队列** :LinkedBlockingQueue、ArrayListBlockingQueue(固定长度)
-- **优先级队列** :PriorityBlockingQueue
-
-提供了阻塞的 take() 和 put() 方法:如果队列为空 take() 将一直阻塞到队列中有内容,如果队列为满 put() 将阻塞到队列有空闲位置。它们响应中断,当收到中断请求的时候会抛出 InterruptedException,从而提前结束阻塞状态。
-
-**使用 BlockingQueue 实现生产者消费者问题**
-
-```java
-// 生产者
-import java.util.concurrent.BlockingQueue;
-
-public class Producer implements Runnable {
- private BlockingQueue queue;
-
- public Producer(BlockingQueue queue) {
- this.queue = queue;
+public class InterruptExample {
+ public static void main(String[] args) throws InterruptedException {
+ Thread thread1 = new MyThread1();
+ thread1.start();
+ thread1.interrupt();
+ System.out.println("Main run");
}
- @Override
- public void run() {
- System.out.println(Thread.currentThread().getName() + " is making product...");
- String product = "made by " + Thread.currentThread().getName();
- try {
- queue.put(product);
- } catch (InterruptedException e) {
- e.printStackTrace();
+ private static class MyThread1 extends Thread {
+ @Override
+ public void run() {
+ try {
+ Thread.sleep(2000);
+ System.out.println("Thread run");
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ }
}
}
}
+
```
-```java
-// 消费者
-import java.util.concurrent.BlockingQueue;
-
-public class Consumer implements Runnable{
- private BlockingQueue queue;
-
- public Consumer(BlockingQueue queue) {
- this.queue = queue;
- }
-
- @Override
- public void run() {
- try {
- String product = queue.take();
- System.out.println(Thread.currentThread().getName() + " is consuming product " + product + "...");
- } catch (InterruptedException e) {
- e.printStackTrace();
- }
- }
-}
+```html
+Main run
+java.lang.InterruptedException: sleep interrupted
+ at java.lang.Thread.sleep(Native Method)
+ at InterruptExample.lambda$main$0(InterruptExample.java:5)
+ at InterruptExample$$Lambda$1/713338599.run(Unknown Source)
+ at java.lang.Thread.run(Thread.java:745)
```
-```java
-// 客户端
-import java.util.concurrent.BlockingQueue;
-import java.util.concurrent.LinkedBlockingQueue;
+## interrupted()
-public class Client {
- public static void main(String[] args) {
- BlockingQueue queue = new LinkedBlockingQueue<>(5);
- for (int i = 0; i < 2; i++) {
- new Thread(new Consumer(queue), "Producer" + i).start();
- }
- for (int i = 0; i < 5; i++) {
- // 只有两个 Product,因此只能消费两个,其它三个消费者被阻塞
- new Thread(new Producer(queue), "Consumer" + i).start();
- }
- for (int i = 2; i < 5; i++) {
- new Thread(new Consumer(queue), "Producer" + i).start();
+如果一个线程的 run() 方法执行一个无限循环,并且没有执行 sleep() 等会抛出 InterruptedException 的操作,那么调用线程的 interrupt() 方法就无法使线程提前结束。
+
+但是调用 interrupt() 方法会设置线程的中断标记,此时调用 interrupted() 方法会返回 true。因此可以在循环体中使用 interrupted() 方法来判断线程是否处于中断状态,从而提前结束线程。
+
+```java
+public class InterruptExample {
+ public static void main(String[] args) throws InterruptedException {
+ Thread thread2 = new MyThread2();
+ thread2.start();
+ thread2.interrupt();
+ }
+
+ private static class MyThread2 extends Thread {
+ @Override
+ public void run() {
+ while (!interrupted()) {
+ // ..
+ }
+ System.out.println("Thread end");
}
}
}
```
```html
-// 运行结果
-Consumer0 is making product...
-Producer0 is consuming product made by Consumer0...
-Consumer1 is making product...
-Producer1 is consuming product made by Consumer1...
-Consumer2 is making product...
-Consumer3 is making product...
-Consumer4 is making product...
-Producer2 is consuming product made by Consumer2...
-Producer3 is consuming product made by Consumer3...
-Producer4 is consuming product made by Consumer4...
+Thread end
```
-# 线程状态
+## Executor 的中断操作
-JDK 从 1.5 开始在 Thread 类中增添了 State 枚举,包含以下六种状态:
+调用 Executor 的 shutdown() 方法会等待线程都执行完毕之后再关闭,但是如果调用的是 shutdownNow() 方法,则相当于调用每个线程的 interrupt() 方法。
-1. **NEW** (新建)
-2. **RUNNABLE** (当线程正在运行或者已经就绪正等待 CPU 时间片)
-3. **BLOCKED** (阻塞,线程在等待获取对象同步锁)
-4. **Waiting** (调用不带超时的 wait() 或 join())
-5. **TIMED_WAITING** (调用 sleep()、带超时的 wait() 或者 join())
-6. **TERMINATED** (死亡)
-
-
-
-# 结束线程
-
-## 1. 阻塞
-
-一个线程进入阻塞状态可能有以下原因:
-
-1. 调用 Thread.sleep() 方法进入休眠状态;
-2. 通过 wait() 使线程挂起,直到线程得到 notify() 或 notifyAll() 消息(或者 java.util.concurrent 类库中等价的 signal() 或 signalAll() 消息;
-3. 等待某个 I/O 的完成;
-4. 试图在某个对象上调用其同步控制方法,但是对象锁不可用,因为另一个线程已经获得了这个锁。
-
-## 2. 中断
-
-使用中断机制即可终止阻塞的线程。
-
-使用 **interrupt()** 方法来中断某个线程,它会设置线程的中断状态。Object.wait(), Thread.join() 和 Thread.sleep() 三种方法在收到中断请求的时候会清除中断状态,并抛出 InterruptedException。
-
-应当捕获这个 InterruptedException 异常,从而做一些清理资源的操作。
-
-**不可中断的阻塞**
-
-不能中断 I/O 阻塞和 synchronized 锁阻塞。
-
-**Executor 的中断操作**
-
-Executor 避免对 Thread 对象的直接操作,但是使用 interrupt() 方法必须持有 Thread 对象。Executor 使用 shutdownNow() 方法来中断所有它里面的所有线程,shutdownNow() 方法会发送 interrupt() 调用给所有线程。
-
-如果只想中断一个线程,那么使用 Executor 的 submit() 而不是 executor() 来启动线程,就可以持有线程的上下文。submit() 将返回一个泛型 Futrue,可以在它之上调用 cancel(),如果将 true 传递给 cancel(),那么它将会发送 interrupt() 调用给特定的线程。
-
-**检查中断**
-
-通过中断的方法来终止线程,需要线程进入阻塞状态才能终止。如果编写的 run() 方法循环条件为 true,但是该线程不发生阻塞,那么线程就永远无法终止。
-
-interrupt() 方法会设置中断状态,可以通过 interrupted() 方法来检查中断状,从而判断一个线程是否已经被中断。
-
-interrupted() 方法在检查完中断状态之后会清除中断状态,这样做是为了确保一次中断操作只会产生一次影响。
-
-# 原子性
-
-对于除 long 和 double 之外的基本类型变量的读写,可以看成是具有原子性的,以不可分割的步骤操作内存。
-
-JVM 将 64 位变量(long 和 double)的读写当做两个分离的 32 位操作来执行,在两个操作之间可能会发生上下文切换,因此不具有原子性。可以使用 **volatile** 关键字来定义 long 和 double 变量,从而获得原子性。
-
-**AtomicInteger、AtomicLong、AtomicReference** 等特殊的原子性变量类提供了下面形式的原子性条件更新语句,使得比较和更新这两个操作能够不可分割地执行。
+以下使用 Lambda 创建线程,相当于创建了一个匿名内部线程。
```java
-boolean compareAndSet(expectedValue, updateValue);
-```
-
-AtomicInteger 使用举例:
-
-```java
-private AtomicInteger ai = new AtomicInteger(0);
-
-public int next() {
- return ai.addAndGet(2)
+public class ExecutorInterruptExample {
+ public static void main(String[] args) {
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ executorService.execute(() -> {
+ try {
+ Thread.sleep(2000);
+ System.out.println("Thread run");
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ }
+ });
+ executorService.shutdownNow();
+ System.out.println("Main run");
+ }
}
```
-原子性具有很多复杂问题,应当尽量使用同步而不是原子性。
+```html
+Main run
+java.lang.InterruptedException: sleep interrupted
+ at java.lang.Thread.sleep(Native Method)
+ at ExecutorInterruptExample.lambda$main$0(ExecutorInterruptExample.java:9)
+ at ExecutorInterruptExample$$Lambda$1/1160460865.run(Unknown Source)
+ at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
+ at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
+ at java.lang.Thread.run(Thread.java:745)
+```
-# volatile
+如果只想中断 Executor 中的一个线程,可以通过使用 submit() 方法来提交一个线程,它会返回一个 Future> 对象,通过调用该对象的 cancel(true) 方法就可以中断线程。
-保证了内存可见性和禁止指令重排,没法保证原子性。
+```java
+Future> future = executorService.submit(() -> {
+ // ..
+});
+future.cancel(true);
+```
-## 1. 内存可见性
+# 五、互斥同步
-普通共享变量被修改之后,什么时候被写入主存是不确定的。
+Java 提供了两种锁机制来控制多个线程对共享资源的互斥访问,第一个是 JVM 实现的 synchronized,而另一个是 JDK 实现的 ReentrantLock。
-volatile 关键字会保证每次修改共享变量之后该值会立即更新到内存中,并且在读取时会从内存中读取值。
+## synchronized
-synchronized 和 Lock 也能够保证内存可见性。它们能保证同一时刻只有一个线程获取锁然后执行同步代码,并且在释放锁之前会将对变量的修改刷新到主存当中。不过只有对共享变量的 set() 和 get() 方法都加上 synchronized 才能保证可见性,如果只有 set() 方法加了 synchronized,那么 get() 方法并不能保证会从内存中读取最新的数据。
+**1. 同步一个代码块**
-## 2. 禁止指令重排
+```java
+public void func () {
+ synchronized (this) {
+ // ...
+ }
+}
+```
+
+它只作用于同一个对象,如果调用两个对象上的同步代码块,就不会进行同步。
+
+对于以下代码,使用 ExecutorService 执行了两个线程(这两个线程使用 Lambda 创建),由于调用的是同一个对象的同步语句块,因此这两个线程就需要进行同步,当一个线程进入同步语句块时,另一个线程就必须等待。
+
+```java
+public class SynchronizedExample {
+
+ public void func1() {
+ synchronized (this) {
+ for (int i = 0; i < 10; i++) {
+ System.out.print(i + " ");
+ }
+ }
+ }
+
+ public static void main(String[] args) {
+ SynchronizedExample e1 = new SynchronizedExample();
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ executorService.execute(() -> e1.func1());
+ executorService.execute(() -> e1.func1());
+ }
+}
+```
+
+```html
+0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9
+```
+
+对于以下代码,两个线程调用了不同对象的同步代码块,因此这两个线程就不需要同步。从输出结果可以看出,两个线程交叉执行。
+
+```java
+public static void main(String[] args) {
+ SynchronizedExample e1 = new SynchronizedExample();
+ SynchronizedExample e2 = new SynchronizedExample();
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ executorService.execute(() -> e1.func1());
+ executorService.execute(() -> e2.func1());
+}
+```
+
+```html
+0 0 1 1 2 2 3 3 4 4 5 5 6 6 7 7 8 8 9 9
+```
+
+
+**2. 同步一个方法**
+
+```java
+public synchronized void func () {
+ // ...
+}
+```
+
+它和同步代码块一样,只作用于同一个对象。
+
+**3. 同步一个类**
+
+```java
+public void func() {
+ synchronized (SynchronizedExample.class) {
+ // ...
+ }
+}
+```
+
+作用于整个类,也就是说两个线程调用同一个类的不同对象上的这种同步语句,也需要进行同步。
+
+```java
+public class SynchronizedExample {
+
+ public void func2() {
+ synchronized (SynchronizedExample.class) {
+ for (int i = 0; i < 10; i++) {
+ System.out.print(i + " ");
+ }
+ }
+ }
+
+ public static void main(String[] args) {
+ SynchronizedExample e1 = new SynchronizedExample();
+ SynchronizedExample e2 = new SynchronizedExample();
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ executorService.execute(() -> e1.func2());
+ executorService.execute(() -> e2.func2());
+ }
+}
+```
+
+```html
+0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9
+```
+
+**4. 同步一个静态方法**
+
+```java
+public synchronized static void fun() {
+ // ...
+}
+```
+
+作用于整个类。
+
+## ReentrantLock
+
+```java
+public class LockExample {
+
+ private Lock lock = new ReentrantLock();
+
+ public void func() {
+ lock.lock();
+ try {
+ for (int i = 0; i < 10; i++) {
+ System.out.print(i + " ");
+ }
+ } finally {
+ lock.unlock(); // 确保释放锁,从而避免发生死锁。
+ }
+ }
+}
+```
+
+```java
+public static void main(String[] args) {
+ LockExample lockExample = new LockExample();
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ executorService.execute(() -> lockExample.func());
+ executorService.execute(() -> lockExample.func());
+}
+```
+
+```html
+0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9
+```
+
+ReentrantLock 是 java.util.concurrent(J.U.C)包中的锁,相比于 synchronized,它多了一些高级功能:
+
+**1. 等待可中断**
+
+当持有锁的线程长期不释放锁的时候,正在等待的线程可以选择放弃等待,改为处理其他事情,可中断特性对处理执行时间非常长的同步块很有帮助。
+
+**2. 可实现公平锁**
+
+公平锁是指多个线程在等待同一个锁时,必须按照申请锁的时间顺序来依次获得锁;而非公平锁则不保证这一点,在锁被释放时,任何一个等待锁的线程都有机会获得锁。synchronized 中的锁是非公平的,ReentrantLock 默认情况下也是非公平的,但可以通过带布尔值的构造函数要求使用公平锁。
+
+**3. 锁绑定多个条件**
+
+一个 ReentrantLock 对象可以同时绑定多个 Condition 对象,而在 synchronized 中,锁对象的 wait() 和 notify() 或 notifyAll() 方法可以实现一个隐含的条件,如果要和多于一个的条件关联的时候,就不得不额外地添加一个锁,而 ReentrantLock 则无须这样做,只需要多次调用 newCondition() 方法即可。
+
+## synchronized 和 ReentrantLock 比较
+
+**1. 锁的实现**
+
+synchronized 是 JVM 实现的,而 ReentrantLock 是 JDK 实现的。
+
+**2. 性能**
+
+从性能上来看,在新版本的 JDK 中对 synchronized 进行了很多优化,例如自旋锁等。目前来看它和 ReentrantLock 的性能基本持平了,因此性能因素不再是选择 ReentrantLock 的理由,而且 synchronized 有更大的优化空间,因此优先考虑 synchronized。
+
+**3. 功能**
+
+ReentrantLock 多了一些高级功能。
+
+**4. 使用选择**
+
+除非需要使用 ReentrantLock 的高级功能,否则优先使用 synchronized。这是因为 synchronized 是 JVM 实现的一种锁机制,JVM 原生地支持它,而 ReentrantLock 不是所有的 JDK 版本都支持。并且使用 synchronized 不用担心没有释放锁而导致死锁问题,因为 JVM 会确保锁的释放。
+
+# 六、线程之间的协作
+
+当多个线程可以一起工作去解决某个问题时,如果某些部分必须在其它部分之前完成,那么就需要对线程进行协调。
+
+## join()
+
+在线程中调用另一个线程的 join() 方法,会将当前线程挂起,而不是忙等待,直到目标线程结束。
+
+对于以下代码,虽然 b 线程先启动,但是因为在 b 线程中调用了 a 线程的 join() 方法,因此 b 线程会等待 a 线程结束才继续执行,因此最后能够保证 a 线程的输出先与 b 线程的输出。
+
+```java
+public class JoinExample {
+
+ private class A extends Thread {
+ @Override
+ public void run() {
+ System.out.println("A");
+ }
+ }
+
+ private class B extends Thread {
+
+ private A a;
+
+ B(A a) {
+ this.a = a;
+ }
+
+ @Override
+ public void run() {
+ try {
+ a.join();
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ }
+ System.out.println("B");
+ }
+ }
+
+ public void test() {
+ A a = new A();
+ B b = new B(a);
+ b.start();
+ a.start();
+ }
+
+ public static void main(String[] args) {
+ JoinExample example = new JoinExample();
+ example.test();
+ }
+}
+```
+
+```
+A
+B
+```
+
+## wait() notify() notifyAll()
+
+调用 wait() 使得线程等待某个条件满足,线程在等待时会被挂起,当其他线程的运行使得这个条件满足时,其它线程会调用 notify() 或者 notifyAll() 来唤醒挂起的线程。
+
+它们都属于 Object 的一部分,而不属于 Thread。
+
+只能用在同步方法或者同步控制块中使用,否则会在运行时抛出 IllegalMonitorStateExeception。
+
+使用 wait() 挂起期间,线程会释放锁。这是因为,如果没有释放锁,那么其它线程就无法进入对象的同步方法或者同步控制块中,那么就无法执行 notify() 或者 notifyAll() 来唤醒挂起的线程,造成死锁。
+
+```java
+public class WaitNotifyExample {
+ public synchronized void before() {
+ System.out.println("before");
+ notifyAll();
+ }
+
+ public synchronized void after() {
+ try {
+ wait();
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ }
+ System.out.println("after");
+ }
+
+ public static void main(String[] args) {
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ WaitNotifyExample example = new WaitNotifyExample();
+ executorService.execute(() -> example.after());
+ executorService.execute(() -> example.before());
+ }
+}
+```
+
+```html
+before
+after
+```
+
+**wait() 和 sleep() 的区别**
+
+1. wait() 是 Object 的方法,而 sleep() 是 Thread 的静态方法;
+2. wait() 会释放锁,sleep() 不会。
+
+## await() signal() signalAll()
+
+java.util.concurrent 类库中提供了 Condition 类来实现线程之间的协调,可以在 Condition 上调用 await() 方法使线程等待,其它线程调用 signal() 或 signalAll() 方法唤醒等待的线程。相比于 wait() 这种等待方式,await() 可以指定等待的条件,因此更加灵活。
+
+使用 Lock 来获取一个 Condition 对象。
+
+```java
+public class AwaitSignalExample {
+ private Lock lock = new ReentrantLock();
+ private Condition condition = lock.newCondition();
+
+ public void before() {
+ lock.lock();
+ try {
+ System.out.println("before");
+ condition.signalAll();
+ } finally {
+ lock.unlock();
+ }
+ }
+
+ public void after() {
+ lock.lock();
+ try {
+ condition.await();
+ System.out.println("after");
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ } finally {
+ lock.unlock();
+ }
+ }
+
+ public static void main(String[] args) {
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ AwaitSignalExample example = new AwaitSignalExample();
+ executorService.execute(() -> example.after());
+ executorService.execute(() -> example.before());
+ }
+}
+```
+
+```html
+before
+after
+```
+
+# 七、J.U.C - AQS
+
+java.util.concurrent(J.U.C)大大提高了并发性能,AQS 被认为是 J.U.C 的核心。
+
+## CountdownLatch
+
+用来控制一个线程等待多个线程。
+
+维护了一个计数器 cnt,每次调用 countDown() 方法会让计数器的值减 1,减到 0 的时候,那些因为调用 await() 方法而在等待的线程就会被唤醒。
+
+
+
+```java
+public class CountdownLatchExample {
+
+ public static void main(String[] args) throws InterruptedException {
+ final int totalTread = 10;
+ CountDownLatch countDownLatch = new CountDownLatch(totalTread);
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ for (int i = 0; i < totalTread; i++) {
+ executorService.execute(() -> {
+ System.out.print("run..");
+ countDownLatch.countDown();
+ });
+ }
+ countDownLatch.await();
+ System.out.println("end");
+ executorService.shutdown();
+ }
+}
+```
+
+```html
+run..run..run..run..run..run..run..run..run..run..end
+```
+
+## CyclicBarrier
+
+用来控制多个线程互相等待,只有当多个线程都到达时,这些线程才会继续执行。
+
+和 CountdownLatch 相似,都是通过维护计数器来实现的。但是它的计数器是递增的,每次执行 await() 方法之后,计数器会加 1,直到计数器的值和设置的值相等,等待的所有线程才会继续执行。和 CountdownLatch 的另一个区别是,CyclicBarrier 的计数器可以循环使用,所以它才叫做循环屏障。
+
+下图应该从下往上看才正确。
+
+
+
+```java
+public class CyclicBarrierExample {
+
+ public static void main(String[] args) throws InterruptedException {
+ final int totalTread = 10;
+ CyclicBarrier cyclicBarrier = new CyclicBarrier(totalTread);
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ for (int i = 0; i < totalTread; i++) {
+ executorService.execute(() -> {
+ System.out.print("before..");
+ try {
+ cyclicBarrier.await();
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ } catch (BrokenBarrierException e) {
+ e.printStackTrace();
+ }
+ System.out.print("after..");
+ });
+ }
+ executorService.shutdown();
+ }
+}
+```
+
+```html
+before..before..before..before..before..before..before..before..before..before..after..after..after..after..after..after..after..after..after..after..
+```
+
+## Semaphore
+
+Semaphore 就是操作系统中的信号量,可以控制对互斥资源的访问线程数。
+
+
+
+以下代码模拟了对某个服务的并发请求,每次只能有 3 个客户端同时访问,请求总数为 10。
+
+```java
+public class SemaphoreExample {
+ public static void main(String[] args) {
+ final int clientCount = 3;
+ final int totalRequestCount = 10;
+ Semaphore semaphore = new Semaphore(clientCount);
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ for (int i = 0; i < totalRequestCount; i++) {
+ executorService.execute(()->{
+ try {
+ semaphore.acquire();
+ System.out.print(semaphore.availablePermits() + " ");
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ } finally {
+ semaphore.release();
+ }
+ });
+ }
+ executorService.shutdown();
+ }
+}
+```
+
+```html
+2 1 2 2 2 2 2 1 2 2
+```
+
+# 八、J.U.C - 其它组件
+
+## FutureTask
+
+在介绍 Callable 时我们知道它可以有返回值,返回值通过 Future 进行封装。FutureTask 实现了 RunnableFuture 接口,该接口继承自 Runnable 和 Future 接口,这使得 FutureTask 既可以当做一个任务执行,也可以有返回值。
+
+```java
+public class FutureTask implements RunnableFuture
+```
+
+```java
+public interface RunnableFuture extends Runnable, Future
+```
+
+当一个计算任务需要执行很长时间,那么就可以用 FutureTask 来封装这个任务,用一个线程去执行该任务,然后其它线程继续执行其它任务。当需要该任务的计算结果时,再通过 FutureTask 的 get() 方法获取。
+
+```java
+public class FutureTaskExample {
+ public static void main(String[] args) throws ExecutionException, InterruptedException {
+ FutureTask futureTask = new FutureTask(new Callable() {
+ @Override
+ public Integer call() throws Exception {
+ int result = 0;
+ for (int i = 0; i < 100; i++) {
+ Thread.sleep(10);
+ result += i;
+ }
+ return result;
+ }
+ });
+
+ Thread computeThread = new Thread(futureTask);
+ computeThread.start();
+
+ Thread otherThread = new Thread(() -> {
+ System.out.println("other task is running...");
+ try {
+ Thread.sleep(1000);
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ }
+ });
+ otherThread.start();
+ System.out.println(futureTask.get());
+ }
+}
+```
+
+```html
+other task is running...
+4950
+```
+
+## BlockingQueue
+
+java.util.concurrent.BlockingQueue 接口有以下阻塞队列的实现:
+
+- **FIFO 队列** :LinkedBlockingQueue、ArrayListBlockingQueue(固定长度)
+- **优先级队列** :PriorityBlockingQueue
+
+提供了阻塞的 take() 和 put() 方法:如果队列为空 take() 将阻塞,直到队列中有内容;如果队列为满 put() 将阻塞,指到队列有空闲位置。
+
+**使用 BlockingQueue 实现生产者消费者问题**
+
+```java
+import java.util.concurrent.ArrayBlockingQueue;
+import java.util.concurrent.BlockingQueue;
+
+public class ProducerConsumer {
+
+ private static BlockingQueue queue = new ArrayBlockingQueue<>(5);
+
+ private static class Producer extends Thread {
+ @Override
+ public void run() {
+ try {
+ queue.put("product");
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ }
+ System.out.print("produce..");
+ }
+ }
+
+ private static class Consumer extends Thread {
+
+ @Override
+ public void run() {
+ try {
+ String product = queue.take();
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ }
+ System.out.print("consume..");
+ }
+ }
+
+ public static void main(String[] args) {
+ for (int i = 0; i < 2; i++) {
+ Producer producer = new Producer();
+ producer.start();
+ }
+ for (int i = 0; i < 5; i++) {
+ Consumer consumer = new Consumer();
+ consumer.start();
+ }
+ for (int i = 0; i < 3; i++) {
+ Producer producer = new Producer();
+ producer.start();
+ }
+ }
+}
+```
+
+```html
+produce..produce..consume..consume..produce..consume..produce..consume..produce..consume..
+```
+
+## ForkJoin
+
+主要用于并行计算中,和 MapReduce 原理类似,都是把大的计算任务拆分成多个小任务并行计算。
+
+```java
+public class ForkJoinExample extends RecursiveTask {
+ private final int threhold = 5;
+ private int first;
+ private int last;
+
+ public ForkJoinExample(int first, int last) {
+ this.first = first;
+ this.last = last;
+ }
+
+ @Override
+ protected Integer compute() {
+ int result = 0;
+ if (last - first <= threhold) {
+ // 任务足够小则直接计算
+ for (int i = first; i <= last; i++) {
+ result += i;
+ }
+ } else {
+ // 拆分成小任务
+ int middle = first + (last - first) / 2;
+ ForkJoinExample leftTask = new ForkJoinExample(first, middle);
+ ForkJoinExample rightTask = new ForkJoinExample(middle + 1, last);
+ leftTask.fork();
+ rightTask.fork();
+ result = leftTask.join() + rightTask.join();
+ }
+ return result;
+ }
+}
+```
+
+```java
+public static void main(String[] args) throws ExecutionException, InterruptedException {
+ ForkJoinExample example = new ForkJoinExample(1, 10000);
+ ForkJoinPool forkJoinPool = new ForkJoinPool();
+ Future result = forkJoinPool.submit(example);
+ System.out.println(result.get());
+}
+```
+
+ForkJoin 使用 ForkJoinPool 来启动,它是一个特殊的线程池,线程数量取决于 CPU 核数。
+
+```java
+public class ForkJoinPool extends AbstractExecutorService
+```
+
+ForkJoinPool 实现了工作窃取算法来提高 CPU 的利用率。每个线程都维护了一个双端队列,用来存储需要执行的任务。工作窃取算法允许空闲的线程从其它线程的双端队列中窃取一个任务来执行。窃取的任务必须是最晚的任务,避免和队列所属线程发生竞争。例如下图中,Thread2 从 Thread1 的队列中拿出最晚的 Task1 任务,Thread1 会拿出 Task2 来执行,这样就避免发生竞争。但是如果队列中只有一个任务时还是会发生竞争。
+
+
+
+# 九、线程不安全示例
+
+如果多个线程对同一个共享数据进行访问而不采取同步操作的话,那么操作的结果是不一致的。
+
+以下代码演示了 1000 个线程同时对 cnt 执行自增操作,操作结束之后它的值为 997 而不是 1000。
+
+```java
+public class ThreadUnsafeExample {
+
+ private int cnt = 0;
+
+ public void add() {
+ cnt++;
+ }
+
+ public int get() {
+ return cnt;
+ }
+
+ public static void main(String[] args) throws InterruptedException {
+ final int threadSize = 1000;
+ ThreadUnsafeExample example = new ThreadUnsafeExample();
+ final CountDownLatch countDownLatch = new CountDownLatch(threadSize);
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ for (int i = 0; i < threadSize; i++) {
+ executorService.execute(() -> {
+ example.add();
+ countDownLatch.countDown();
+ });
+ }
+ countDownLatch.await();
+ executorService.shutdown();
+ System.out.println(example.get());
+ }
+}
+```
+
+```html
+997
+```
+
+# 十、Java 内存模型
+
+Java 内存模型视图屏蔽各种硬件和操作系统的内存访问差异,以实现让 Java 程序在各种平台下都能达到一致的内存访问效果。
+
+## 主内存与工作内存
+
+处理器上的寄存器的读写的速度比内存快几个数量级,为了解决这种速度矛盾,在它们之间加入了高速缓存。
+
+加入高速缓存带来了一个新的问题:缓存一致性。如果多个缓存共享同一块主内存区域,那么多个缓存的数据可能会不一致。CPU 使用一致性协议来解决一致性问题。
+
+
+
+所有的变量都存储在主内存中,每个线程还有自己的工作内存,工作内存存储在高速缓存或者寄存器中,保存了该线程使用的变量的主内存副本拷贝。
+
+线程只能直接操作工作内存中的变量,不同线程之间的变量值传递需要通过主内存来完成。
+
+
+
+## 内存间交互操作
+
+Java 内存模型定义了 8 个操作来完成主内存和工作内存的交互操作。
+
+
+
+- read:把一个变量的值从主内存传输到工作内存中
+- load:在 read 之后执行,把 read 得到的值放入工作内存的变量副本中
+- use:把工作内存中一个变量的值传递给执行引擎
+- assign:把一个从执行引擎接收到的值赋给工作内存的变量
+- store:把工作内存的一个变量的值传送到主内存中
+- write:在 store 之后执行,把 store 得到的值放入主内存的变量中
+- lock:作用于主内存的变量
+- unlock
+
+## 内存模型三大特性
+
+### 1. 原子性
+
+Java 内存模型保证了 read、load、use、assign、store、write、lock 和 unlock 操作具有原子性,例如对一个 int 类型的变量执行 assign 赋值操作,这个操作就是原子性的。但是 Java 内存模型允许虚拟机将没有被 volatile 修饰的 64 位数据(long,double)的读写操作划分为两次 32 位的操作来进行,即 load、store、read 和 write 操作可以不具备原子性。
+
+有一个错误认识就是,int 等原子性的变量在多线程环境中不会出现线程安全问题。前面的线程不安全示例代码中,cnt 变量属于 int 类型变量,1000 个线程对它进行自增操作之后,得到的值为 997 而不是 1000。
+
+为了方便讨论,将内存间的交互操作简化为 3 个:load、assign、store。
+
+下图演示了两个线程同时对 cnt 变量进行操作,load、assign、store 这一系列操作整体上看不具备原子性,那么在 T1 修改 cnt 并且还没有将修改后的值写入主内存,T2 依然可以读入该变量的值。可以看出,这两个线程虽然执行了两次自增运算,但是主内存中 cnt 的值最后为 1 而不是 2。因此对 int 类型读写操作满足原子性只是说明 load、assign、store 这些单个操作具备原子性。
+
+
+
+AtomicInteger 能保证多个线程修改的原子性。
+
+
+
+使用 AtomicInteger 重写之前线程不安全的代码之后得到以下线程安全实现:
+
+```java
+public class AtomicExample {
+ private AtomicInteger cnt = new AtomicInteger();
+
+ public void add() {
+ cnt.incrementAndGet();
+ }
+
+ public int get() {
+ return cnt.get();
+ }
+
+ public static void main(String[] args) throws InterruptedException {
+ final int threadSize = 1000;
+ AtomicExample example = new AtomicExample();
+ final CountDownLatch countDownLatch = new CountDownLatch(threadSize);
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ for (int i = 0; i < threadSize; i++) {
+ executorService.execute(() -> {
+ example.add();
+ countDownLatch.countDown();
+ });
+ }
+ countDownLatch.await();
+ executorService.shutdown();
+ System.out.println(example.get());
+ }
+}
+
+```
+
+```html
+1000
+```
+
+除了使用原子类之外,也可以使用 synchronized 互斥锁来保证操作的完整性,它对应的内存间交互操作为:lock 和 unlock,在虚拟机实现上对应的字节码指令为 monitorenter 和 monitorexit。
+
+```java
+public class AtomicSynchronizedExample {
+ private int cnt = 0;
+
+ public synchronized void add() {
+ cnt++;
+ }
+
+ public synchronized int get() {
+ return cnt;
+ }
+
+ public static void main(String[] args) throws InterruptedException {
+ final int threadSize = 1000;
+ AtomicSynchronizedExample example = new AtomicSynchronizedExample();
+ final CountDownLatch countDownLatch = new CountDownLatch(threadSize);
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ for (int i = 0; i < threadSize; i++) {
+ executorService.execute(() -> {
+ example.add();
+ countDownLatch.countDown();
+ });
+ }
+ countDownLatch.await();
+ executorService.shutdown();
+ System.out.println(example.get());
+ }
+}
+```
+
+```html
+1000
+```
+
+### 2. 可见性
+
+可见性指当一个线程修改了共享变量的值,其它线程能够立即得知这个修改。Java 内存模型是通过在变量修改后将新值同步回主内存,在变量读取前从主内存刷新变量值来实现可见性的。
+
+volatile 可保证可见性。synchronized 也能够保证可见性,对一个变量执行 unlock 操作之前,必须把变量值同步回主内存。final 关键字也能保证可见性:被 final 关键字修饰的字段在构造器中一旦初始化完成,并且没有发生 this 逃逸(其它线程可以通过 this 引用访问到初始化了一般的对象),那么其它线程就能看见 final 字段的值。
+
+对前面的线程不安全示例中的 cnt 变量用 volatile 修饰,不能解决线程不安全问题。因为 volatile 并不能保证操作的原子性。
+
+// TODO:volatile 不能解决线程不安全问题的示例代码。
+
+### 3. 有序性
+
+有序性是指:在本线程内观察,所有操作都是有序的。在一个线程观察另一个线程,所有操作都是无序的,无序是因为发生了指令重排序。
在 Java 内存模型中,允许编译器和处理器对指令进行重排序,重排序过程不会影响到单线程程序的执行,却会影响到多线程并发执行的正确性。
-volatile 关键字通过添加内存屏障的方式来进制指令重排,即重排序时不能把后面的指令放到内存屏障之前。
+volatile 关键字通过添加内存屏障的方式来禁止指令重排,即重排序时不能把后面的指令放到内存屏障之前。
-可以通过 synchronized 和 Lock 来保证有序性,它们保证每个时刻只有一个线程执行同步代码,相当于是让线程顺序执行同步代码,自然就保证了有序性。
+也可以通过 synchronized 来保证有序性,它保证每个时刻只有一个线程执行同步代码,相当于是让线程顺序执行同步代码。
-# 多线程开发良好的实践
+## 先行发生原则
-- 给线程命名;
-- 最小化同步范围;
-- 优先使用 volatile;
-- 尽可能使用更高层次的并发工具而非 wait 和 notify() 来实现线程通信,如 BlockingQueue, Semeaphore;
-- 多用并发容器,少用同步容器,并发容器比同步容器的可扩展性更好。
-- 考虑使用线程池
-- 最低限度的使用同步和锁,缩小临界区。因此相对于同步方法,同步块会更好。
+上面提到了可以用 volatile 和 synchronized 来保证有序性。除此之外,JVM 还规定了先行发生原则,让一个操作无需控制就能先于另一个操作完成。
-# 未完待续
+主要有以下这些原则:
+
+### 1. 单一线程原则
+
+> Single Thread rule
+
+在一个线程内,在程序前面的操作先行发生于后面的操作。
+
+
+
+### 2. 管程锁定规则
+
+> Monitor Lock Rule
+
+一个 unlock 操作先行发生于后面对同一个锁的 lock 操作。
+
+
+
+### 3. volatile 变量规则
+
+> Volatile Variable Rule
+
+对一个 volatile 变量的写操作先行发生于后面对这个变量的读操作。
+
+
+
+### 4. 线程启动规则
+
+> Thread Start Rule
+
+Thread 对象的 start() 方法先行发生于此线程的每一个动作。
+
+
+
+### 5. 线程加入规则
+
+> Thread Join Rule
+
+join() 方法返回先行发生于 Thread 对象的结束。
+
+
+
+### 6. 线程中断规则
+
+> Thread Interruption Rule
+
+对线程 interrupt() 方法的调用先行发生于被中断线程的代码检测到中断事件的发生,可以通过 Thread.interrupted() 方法检测到是否有中断发生。
+
+### 7. 对象终结规则
+
+> Finalizer Rule
+
+一个对象的初始化完成(构造函数执行结束)先行发生于它的 finalize() 方法的开始。
+
+### 8. 传递性
+
+> Transitivity
+
+如果操作 A 先行发生于操作 B,操作 B 先行发生于操作 C,那么操作 A 先行发生于操作 C。
+
+# 十一、线程安全
+
+## 线程安全分类
+
+线程安全不是一个非真即假的命题,可以将共享数据按照安全程度的强弱顺序分成以下五类:不可变、绝对线程安全、相对线程安全、线程兼容和线程对立。
+
+### 1. 不可变
+
+不可变(Immutable)的对象一定是线程安全的,无论是对象的方法实现还是方法的调用者,都不需要再采取任何的线程安全保障措施,只要一个不可变的对象被正确地构建出来,那其外部的可见状态永远也不会改变,永远也不会看到它在多个线程之中处于不一致的状态。
+
+不可变的类型:
+
+- final 关键字修饰的基本数据类型;
+- String
+- 枚举类型
+- Number 部分子类,如 Long 和 Double 等数值包装类型,BigInteger 和 BigDecimal 等大数据类型。但同为 Number 的子类型的原子类 AtomicInteger 和 AtomicLong 则并非不可变的。
+
+对于集合类型,可以使用 Collections.unmodifiableXXX() 方法来获取一个不可变的集合。
+
+```java
+public class ImmutableExample {
+ public static void main(String[] args) {
+ Map map = new HashMap<>();
+ Map unmodifiableMap = Collections.unmodifiableMap(map);
+ unmodifiableMap.put("a", 1);
+ }
+}
+```
+
+```html
+Exception in thread "main" java.lang.UnsupportedOperationException
+ at java.util.Collections$UnmodifiableMap.put(Collections.java:1457)
+ at ImmutableExample.main(ImmutableExample.java:9)
+```
+
+Collections.unmodifiableXXX() 先对原始的集合进行拷贝,需要对集合进行修改的方法都直接抛出异常。
+
+```java
+public V put(K key, V value) {
+ throw new UnsupportedOperationException();
+}
+```
+
+多线程环境下,应当尽量使对象称为不可变,来满足线程安全。
+
+### 2. 绝对线程安全
+
+不管运行时环境如何,调用者都不需要任何额外的同步措施。
+
+### 3. 相对线程安全
+
+相对的线程安全需要保证对这个对象单独的操作是线程安全的,在调用的时候不需要做额外的保障措施,但是对于一些特定顺序的连续调用,就可能需要在调用端使用额外的同步手段来保证调用的正确性。
+
+在 Java 语言中,大部分的线程安全类都属于这种类型,例如 Vector、HashTable、Collections 的 synchronizedCollection() 方法包装的集合等。
+
+对于下面的代码,如果删除元素的线程删除了一个元素,而获取元素的线程试图访问一个已经被删除的元素,那么就会抛出 ArrayIndexOutOfBoundsException。
+
+```java
+public class VectorUnsafeExample {
+ private static Vector vector = new Vector<>();
+
+ public static void main(String[] args) {
+ while (true) {
+ for (int i = 0; i < 100; i++) {
+ vector.add(i);
+ }
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ executorService.execute(() -> {
+ for (int i = 0; i < vector.size(); i++) {
+ vector.remove(i);
+ }
+ });
+ executorService.execute(() -> {
+ for (int i = 0; i < vector.size(); i++) {
+ vector.get(i);
+ }
+ });
+ executorService.shutdown();
+ }
+ }
+}
+```
+
+```html
+Exception in thread "Thread-159738" java.lang.ArrayIndexOutOfBoundsException: Array index out of range: 3
+ at java.util.Vector.remove(Vector.java:831)
+ at VectorUnsafeExample.lambda$main$0(VectorUnsafeExample.java:14)
+ at VectorUnsafeExample$$Lambda$1/713338599.run(Unknown Source)
+ at java.lang.Thread.run(Thread.java:745)
+```
+
+
+如果要保证上面的代码能正确执行下去,就需要对删除元素和获取元素的代码进行同步。
+
+```java
+executorService.execute(() -> {
+ synchronized (vector) {
+ for (int i = 0; i < vector.size(); i++) {
+ vector.remove(i);
+ }
+ }
+});
+executorService.execute(() -> {
+ synchronized (vector) {
+ for (int i = 0; i < vector.size(); i++) {
+ vector.get(i);
+ }
+ }
+});
+```
+
+### 4. 线程兼容
+
+线程兼容是指对象本身并不是线程安全的,但是可以通过在调用端正确地使用同步手段来保证对象在并发环境中可以安全地使用,我们平常说一个类不是线程安全的,绝大多数时候指的是这一种情况。Java API 中大部分的类都是属于线程兼容的,如与前面的 Vector 和 HashTable 相对应的集合类 ArrayList 和 HashMap 等。
+
+### 5. 线程对立
+
+线程对立是指无论调用端是否采取了同步措施,都无法在多线程环境中并发使用的代码。由于 Java 语言天生就具备多线程特性,线程对立这种排斥多线程的代码是很少出现的,而且通常都是有害的,应当尽量避免。
+
+## 线程安全的实现方法
+
+### 1. 互斥同步
+
+synchronized 和 ReentrantLock。
+
+### 2. 非阻塞同步
+
+互斥同步最主要的问题就是进行线程阻塞和唤醒所带来的性能问题,因此这种同步也称为阻塞同步(Blocking Synchronization)。
+
+从处理问题的方式上说,互斥同步属于一种悲观的并发策略,总是认为只要不去做正确的同步措施(例如加锁),那就肯定会出现问题,无论共享数据是否真的会出现竞争,它都要进行加锁(这里讨论的是概念模型,实际上虚拟机会优化掉很大一部分不必要的加锁)、用户态核心态转换、维护锁计数器和检查是否有被阻塞的线程需要唤醒等操作。随着硬件指令集的发展,我们有了另外一个选择:基于冲突检测的乐观并发策略,通俗地说,就是先进行操作,如果没有其他线程争用共享数据,那操作就成功了;如果共享数据有争用,产生了冲突,那就再采取其他的补偿措施(最常见的补偿措施就是不断地重试,直到成功为止),这种乐观的并发策略的许多实现都不需要把线程挂起,因此这种同步操作称为非阻塞同步(Non-Blocking Synchronization)。
+
+乐观锁需要操作和冲突检测这两个步骤具备原子性,这里就不能再使用互斥同步来保证了,只能靠硬件来完成。硬件支持的原子性操作最典型的是:比较并交换(Compare-and-Swap,CAS)。
+
+CAS 指令需要有 3 个操作数,分别是内存位置(在 Java 中可以简单理解为变量的内存地址,用 V 表示)、旧的预期值(用 A 表示)和新值(用 B 表示)。CAS 指令执行时,当且仅当 V 符合旧预期值 A 时,处理器用新值 B 更新 V 的值,否则它就不执行更新。但是无论是否更新了 V 的值,都会返回 V 的旧值,上述的处理过程是一个原子操作。
+
+J.U.C 包里面的整数原子类 AtomicInteger,其中的 compareAndSet() 和 getAndIncrement() 等方法都使用了 Unsafe 类的 CAS 操作。
+
+在下面的代码 1 中,使用了 AtomicInteger 执行了自增的操作。代码 2 是 incrementAndGet() 的源码,它调用了 unsafe 的 getAndAddInt() 。代码 3 是 getAndAddInt() 源码,var1 指示内存位置,var2 指示新值,var4 指示操作需要加的数值,这里为 1。在代码 3 的实现中,通过 getIntVolatile(var1, var2) 得到旧的预期值。通过调用 compareAndSwapInt() 来进行 CAS 比较,如果 var2=var5,那么就更新内存地址为 var1 的变量为 var5+var4。可以看到代码 3 是在一个循环中进行,发生冲突的做法是不断的进行重试。
+
+```java
+// 代码 1
+private AtomicInteger cnt = new AtomicInteger();
+
+public void add() {
+ cnt.incrementAndGet();
+}
+```
+
+```java
+// 代码 2
+public final int incrementAndGet() {
+ return unsafe.getAndAddInt(this, valueOffset, 1) + 1;
+}
+```
+
+```java
+// 代码 3
+public final int getAndAddInt(Object var1, long var2, int var4) {
+ int var5;
+ do {
+ var5 = this.getIntVolatile(var1, var2);
+ } while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));
+
+ return var5;
+}
+```
+
+ABA :如果一个变量 V 初次读取的时候是 A 值,它的值被改成了 B,后来又被改回为 A,那 CAS 操作就会误认为它从来没有被改变过。J.U.C 包提供了一个带有标记的原子引用类“AtomicStampedReference”来解决这个问题,它可以通过控制变量值的版本来保证 CAS 的正确性。大部分情况下 ABA 问题不会影响程序并发的正确性,如果需要解决 ABA 问题,改用传统的互斥同步可能会比原子类更高效。
+
+### 3. 无同步方案
+
+要保证线程安全,并不是一定就要进行同步,两者没有因果关系。同步只是保证共享数据争用时的正确性的手段,如果一个方法本来就不涉及共享数据,那它自然就无须任何同步措施去保证正确性,因此会有一些代码天生就是线程安全的。
+
+**(一)可重入代码(Reentrant Code)**
+
+这种代码也叫做纯代码(Pure Code),可以在代码执行的任何时刻中断它,转而去执行另外一段代码(包括递归调用它本身),而在控制权返回后,原来的程序不会出现任何错误。相对线程安全来说,可重入性是更基本的特性,它可以保证线程安全,即所有的可重入的代码都是线程安全的,但是并非所有的线程安全的代码都是可重入的。
+
+可重入代码有一些共同的特征,例如不依赖存储在堆上的数据和公用的系统资源、用到的状态量都由参数中传入、不调用非可重入的方法等。我们可以通过一个简单的原则来判断代码是否具备可重入性:如果一个方法,它的返回结果是可以预测的,只要输入了相同的数据,就都能返回相同的结果,那它就满足可重入性的要求,当然也就是线程安全的。
+
+**(二)栈封闭**
+
+多个线程访问同一个方法的局部变量时,不会出现线程安全问题,因为局部变量存储在栈中,属于线程私有的。
+
+```java
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.Executors;
+
+public class StackClosedExample {
+ public void add100() {
+ int cnt = 0;
+ for (int i = 0; i < 100; i++) {
+ cnt++;
+ }
+ System.out.println(cnt);
+ }
+
+ public static void main(String[] args) {
+ StackClosedExample example = new StackClosedExample();
+ ExecutorService executorService = Executors.newCachedThreadPool();
+ executorService.execute(() -> example.add100());
+ executorService.execute(() -> example.add100());
+ executorService.shutdown();
+ }
+}
+```
+
+```html
+100
+100
+```
+
+**(三)线程本地存储(Thread Local Storage)**
+
+如果一段代码中所需要的数据必须与其他代码共享,那就看看这些共享数据的代码是否能保证在同一个线程中执行。如果能保证,我们就可以把共享数据的可见范围限制在同一个线程之内,这样,无须同步也能保证线程之间不出现数据争用的问题。
+
+符合这种特点的应用并不少见,大部分使用消费队列的架构模式(如“生产者-消费者”模式)都会将产品的消费过程尽量在一个线程中消费完,其中最重要的一个应用实例就是经典 Web 交互模型中的“一个请求对应一个服务器线程”(Thread-per-Request)的处理方式,这种处理方式的广泛应用使得很多 Web 服务端应用都可以使用线程本地存储来解决线程安全问题。
+
+可以使用 java.lang.ThreadLocal 类来实现线程本地存储功能。
+
+对于以下代码,thread1 中设置 threadLocal 为 1,而 thread2 设置 threadLocal 为 2。过了一段时间之后,thread1 读取 threadLocal 依然是 1,不受 thread2 的影响。
+
+```java
+public class ThreadLocalExample {
+ public static void main(String[] args) {
+ ThreadLocal threadLocal = new ThreadLocal();
+ Thread thread1 = new Thread(() -> {
+ threadLocal.set(1);
+ try {
+ Thread.sleep(1000);
+ } catch (InterruptedException e) {
+ e.printStackTrace();
+ }
+ System.out.println(threadLocal.get());
+ threadLocal.remove();
+ });
+ Thread thread2 = new Thread(() -> {
+ threadLocal.set(2);
+ threadLocal.remove();
+ });
+ thread1.start();
+ thread2.start();
+ }
+}
+```
+
+```html
+1
+```
+
+为了理解 ThreadLocal,先看以下代码:
+
+```java
+public class ThreadLocalExample1 {
+ public static void main(String[] args) {
+ ThreadLocal threadLocal1 = new ThreadLocal();
+ ThreadLocal threadLocal2 = new ThreadLocal();
+ Thread thread1 = new Thread(() -> {
+ threadLocal1.set(1);
+ threadLocal2.set(1);
+ });
+ Thread thread2 = new Thread(() -> {
+ threadLocal1.set(2);
+ threadLocal2.set(2);
+ });
+ thread1.start();
+ thread2.start();
+ }
+}
+```
+
+它所对应的底层结构图为:
+
+
+
+每个 Thread 都有一个 TreadLocal.ThreadLocalMap 对象,Thread 类中就定义了 ThreadLocal.ThreadLocalMap 成员。
+
+```java
+/* ThreadLocal values pertaining to this thread. This map is maintained
+ * by the ThreadLocal class. */
+ThreadLocal.ThreadLocalMap threadLocals = null;
+```
+
+当调用一个 ThreadLocal 的 set(T value) 方法时,先得到当前线程的 ThreadLocalMap 对象,然后将 ThreadLocal->value 键值对插入到该 Map 中。
+
+```java
+public void set(T value) {
+ Thread t = Thread.currentThread();
+ ThreadLocalMap map = getMap(t);
+ if (map != null)
+ map.set(this, value);
+ else
+ createMap(t, value);
+}
+```
+
+get() 方法类似。
+
+```java
+public T get() {
+ Thread t = Thread.currentThread();
+ ThreadLocalMap map = getMap(t);
+ if (map != null) {
+ ThreadLocalMap.Entry e = map.getEntry(this);
+ if (e != null) {
+ @SuppressWarnings("unchecked")
+ T result = (T)e.value;
+ return result;
+ }
+ }
+ return setInitialValue();
+}
+```
+
+ThreadLocal 从理论上讲并不是用来解决多线程并发问题的,因为根本不存在多线程竞争。在一些场景 (尤其是使用线程池) 下,由于 ThreadLocal.ThreadLocalMap 的底层数据结构导致 ThreadLocal 有内存泄漏的情况,尽可能在每次使用 ThreadLocal 后手动调用 remove(),以避免出现 ThreadLocal 经典的内存泄漏甚至是造成自身业务混乱的风险。
+
+# 十二、锁优化
+
+高效并发是从 JDK 1.5 到 JDK 1.6 的一个重要改进,HotSpot 虚拟机开发团队在这个版本上花费了大量的精力去实现各种锁优化技术,如适应性自旋(Adaptive Spinning)、锁消除(Lock Elimination)、锁粗化(Lock Coarsening)、轻量级锁(Lightweight Locking)和偏向锁(Biased Locking)等。这些技术都是为了在线程之间更高效地共享数据,以及解决竞争问题,从而提高程序的执行效率。
+
+## 自旋锁与自适应自旋
+
+前面我们讨论互斥同步的时候,提到了互斥同步对性能最大的影响是阻塞的实现,挂起线程和恢复线程的操作都需要转入内核态完成,这些操作给系统的并发性能带来了很大的压力。同时,虚拟机的开发团队也注意到在许多应用上,共享数据的锁定状态只会持续很短的一段时间,为了这段时间去挂起和恢复线程并不值得。如果物理机器有一个以上的处理器,能让两个或以上的线程同时并行执行,我们就可以让后面请求锁的那个线程 “稍等一下”,但不放弃处理器的执行时间,看看持有锁的线程是否很快就会释放锁。为了让线程等待,我们只需让线程执行一个忙循环(自旋),这项技术就是所谓的自旋锁。
+
+自旋等待本身虽然避免了线程切换的开销,但它是要占用处理器时间的,因此,如果锁被占用的时间很短,自旋等待的效果就会非常好。反之,如果锁被占用的时候很长,那么自旋的线程只会白白消耗处理器资源,而不会做任何有用的工作,反而会带来性能上的浪费。因此,自旋等待的时间必须要有一定的限度,如果自旋超过了限定的次数仍然没有成功获得锁,就应当使用传统的方式去挂起线程了。自旋次数的默认值是 10 次,用户可以使用参数 -XX:PreBlockSpin 来更改。
+
+在 JDK 1.6 中引入了自适应的自旋锁。自适应意味着自旋的时间不再固定了,而是由前一次在同一个锁上的自旋时间及锁的拥有者的状态来决定。如果在同一个锁对象上,自旋等待刚刚成功获得过锁,并且持有锁的线程正在运行中,那么虚拟机就会认为这次自旋也很有可能再次成功,进而它将允许自旋等待持续相对更长的时间,比如 100 个循环。另外,如果对于某个锁,自旋很少成功获得过,那在以后要获取这个锁时将可能省略掉自旋过程,以避免浪费处理器资源。有了自适应自旋,随着程序运行和性能监控信息的不断完善,虚拟机对程序锁的状况预测就会越来越准确,虚拟机就会变得越来越“聪明”了。
+
+## 锁消除
+
+锁消除是指虚拟机即时编译器在运行时,对一些代码上要求同步,但是被检测到不可能存在共享数据竞争的锁进行消除。锁消除的主要判定依据来源于逃逸分析的数据支持,如果判定在一段代码中,堆上的所有数据都不会逃逸出去从而被其他线程访问到,那就可以把他们当做栈上数据对待,认为它们是线程私有的,同步加锁自然就无须进行。
+
+也许读者会有疑问,变量是否逃逸,对于虚拟机来说需要使用数据流分析来确定,但是程序自己应该是很清楚的,怎么会在明知道不存在数据争用的情况下要求同步呢?答案是有许多同步措施并不是程序员自己加入的。同步的代码在 Java 程序中的普遍程度也许超过了大部分读者的想象。下面段非常简单的代码仅仅是输出 3 个字符串相加的结果,无论是源码字面上还是程序语义上都没有同步。
+
+```java
+public static String concatString(String s1, String s2, String s3) {
+ return s1 + s2 + s3;
+}
+```
+
+我们也知道,由于 String 是一个不可变的类,对字符串的连接操作总是通过生成新的 String 对象来进行的,因此 Javac 编译器会对 String 连接做自动优化。在 JDK 1.5 之前,会转化为 StringBuffer 对象的连续 append() 操作,在 JDK 1.5 及以后的版本中,会转化为 StringBuilder 对象的连续 append() 操作,即上面的代码可能会变成下面的样子:
+
+```java
+public static String concatString(String s1, String s2, String s3) {
+ StringBuffer sb = new StringBuffer();
+ sb.append(s1);
+ sb.append(s2);
+ sb.append(s3);
+ return sb.toString();
+}
+```
+每个 StringBuffer.append() 方法中都有一个同步块,锁就是 sb 对象。虚拟机观察变量 sb,很快就会发现它的动态作用域被限制在 concatString() 方法内部。也就是说,sb 的所有引用永远不会 “逃逸” 到 concatString() 方法之外,其他线程无法访问到它。因此,虽然这里有锁,但是可以被安全地消除掉,在即时编译之后,这段代码就会忽略掉所有的同步而直接执行了。
+
+## 锁粗化
+
+原则上,我们在编写代码的时候,总是推荐将同步块的作用范围限制得尽量小:只在共享数据的实际作用域中才进行同步。这样是为了使得需要同步的操作数量尽可能变小,如果存在锁竞争,那等待锁的线程也能尽快拿到锁。
+
+大部分情况下,上面的原则都是正确的,但是如果一系列的连续操作都对同一个对象反复加锁和解锁,甚至加锁操作是出现在循环体中,那即使没有线程竞争,频繁地进行互斥同步操作也会导致不必要的性能损耗。
+
+上一节的示例代码中连续的 append() 方法就属于这类情况。如果虚拟机探测到由这样的一串零碎的操作都对同一个对象加锁,将会把加锁同步的范围扩展(粗化)到整个操作序列的外部。对于上一节的示例代码就是扩展到第一个 append() 操作之前直至最后一个 append() 操作之后,这样只需要加锁一次就可以了。
+
+## 轻量级锁
+
+轻量级锁是 JDK 1.6 之中加入的新型锁机制,它名字中的“轻量级”是相对于使用操作系统互斥量来实现的传统锁而言的,因此传统的锁机制就称为“重量级”锁。首先需要强调一点的是,轻量级锁并不是用来代替重要级锁的,它的本意是在没有多线程竞争的前提下,减少传统的重量级锁使用操作系统互斥量产生的性能消耗。
+
+要理解轻量级锁,以及后面会讲到的偏向锁的原理和运作过程,必须从 HotSpot 虚拟机的对象(对象头部分)的内存布局开始介绍。HotSpot 虚拟机的对象头(Object Header)分为两部分信息,第一部分用于存储对象自身的运行时数据,如哈希码(HashCode)、GC 分代年龄(Generational GC Age)等,这部分数据是长度在 32 位和 64 位的虚拟机中分别为 32 bit 和 64 bit,官方称它为“Mark Word”,它是实现轻量级锁和偏向锁的关键。另外一部分用于存储指向方法区对象类型数据的指针,如果是数组对象的话,还会有一个额外的部分用于存储数组长度。
+
+简单地介绍了对象的内存布局后,我们把话题返回到轻量级锁的执行过程上。在代码进入同步块的时候,如果此同步对象没有被锁定(锁标志位为 “01” 状态)虚拟机首先将在当前线程的栈帧中建立一个名为锁记录(Lock Record)的空间,用于存储锁对象目前的 Mark Word 的拷贝(官方把这份拷贝加上了一个 Displaced 前缀,即 Displaced Mark Word)。然后,虚拟机将使用 CAS 操作尝试将对象的 Mark Word 更新为指向 Lock Record 的指针。如果这个更新动作成功了,那么这个线程就拥有了该对象的锁,并且对象 Mark Word 的锁标志位(Mark Word 的最后 2bit)将转变为 “00”,即表示此对象处于轻量级锁定状态。
+
+
+
+如果这个更新操作失败了,虚拟机首先会检查对象的 Mark Word 是否指向当前线程的栈帧,如果是的话只说明当前线程已经拥有了这个对象的锁,那就可以直接进入同步块继续执行,否则说明这个锁对象已经被其他线程线程抢占了。如果有两条以上的线程争用同一个锁,那轻量级锁就不再有效,要膨胀为重量级锁,所标志的状态变为“10”,Mark Word 中存储的就是指向重量级锁(互斥量)的指针,后面等待锁的线程也要进入阻塞状态。
+
+上面描述的是轻量级锁的加锁过程,它的解锁过程也是通过 CAS 操作来进行的,如果对象的 Mark Word 仍然指向着线程的锁记录,那就用 CAS 操作把对象当前的 Mark Word 和线程中复制的 Displaced Mark Word 替换回来,如果替换成功,整个同步过程就完成了。如果替换失败,说明有其他线程尝试过获取该锁,那就要释放锁的同时,唤醒被挂起的线程。
+
+轻量级锁能提升程序同步性能的依据是“对于绝大部分的锁,在整个同步周期内都是不存在竞争的”,这是一个经验数据。如果没有竞争,轻量级锁使用 CAS 操作避免了使用互斥量的开销,但如果存在锁竞争,除了互斥量的开销外,还额外发生了 CAS 操作,因此在有竞争的情况下,轻量级锁会比传统的重量级锁更慢。
+
+## 偏向锁
+
+偏向锁也是 JDK 1.6 中引入的一项锁优化,它的目的是消除数据在无竞争情况下的同步原语,进一步提高程序的运行性能。如果说轻量级锁是在无竞争的情况下使用 CAS 操作去消除同步使用的互斥量,那偏向锁就是在无竞争的情况下把整个同步都消除掉,连 CAS 操作都不做了。
+
+偏向锁的“偏”,就是偏心的“偏”、偏袒的“偏”,它的意思是这个锁会偏向于第一个获得它的线程,如果在接下来的执行过程中,该锁没有被其他的线程获取,则持有偏向锁的线程将永远不需要再进行同步。
+
+假设当前虚拟机启用了偏向锁(启用参数 -XX:+UseBiasedLocking,这是 JDK 1.6 的默认值),那么,当锁对象第一次被线程获取的时候,虚拟机将会把对象头中的标志位设为“01”,即偏向模式。同时使用 CAS 操作把获取到这个锁的线程 ID 记录在对象的 Mark Word 之中,如果 CAS 操作成功,持有偏向锁的线程以后每次进入这个锁相关的同步块时,虚拟机都可以不再进行如何同步操作(例如 Locking、Unlocking 及对 Mark Word 的 Update 等)。
+
+当有另外一个线程去尝试获取这个锁时,偏向模式就宣告结束。根据锁对象目前是否处于被锁定的状态,撤销偏向(Revoke Bias)后恢复到未锁定(标志位为“01”)或轻量级锁定(标志位为“00”)的状态,后续的同步操作就如上面介绍的轻量级锁那样执行。偏向锁、轻量级锁的状态转换及对象 Mark Word 的关系如图 13-5 所示。
+
+
+
+偏向锁可以提高带有同步但无竞争的程序性能。它同样是一个带有效益权衡(Trade Off)性质的优化,也就是说,它并不一定总是对程序运行有利,如果程序中大多数的锁总是被多个不同的线程访问,那偏向模式就是多余的。在具体问题具体分析的前提下,有时候使用参数 -XX:-UseBiasedLocking 来禁止偏向锁优化反而可以提升性能。
+
+
+# 十三、多线程开发良好的实践
+
+- 给线程起个有意义的名字,这样可以方便找 Bug。
+
+- 缩小同步范围,例如 对于 synchronized,应该尽量使用同步块而不是同步方法。
+
+- 多用同步类少用 wait 和 notify。首先,CountDownLatch, Semaphore, CyclicBarrier 和 Exchanger 这些同步类简化了编码操作,而用 wait 和 notify 很难实现对复杂控制流的控制。其次,这些类是由最好的企业编写和维护在后续的 JDK 中它们还会不断优化和完善,使用这些更高等级的同步工具你的程序可以不费吹灰之力获得优化。
+
+- 多用并发集合少用同步集合。并发集合比同步集合的可扩展性更好,例如应该使用 ConcurrentHashMap 而不是 Hashttable。
+
+- 使用本地变量和不可变类来保证线程安全。
+
+- 使用线程池而不是直接创建 Thread 对象,这是因为创建线程代价很高,线程池可以有效地利用有限的线程来启动任务。
+
+- 使用 BlockingQueue 实现生产者消费者问题。
# 参考资料
-- Java 编程思想
+- BruceEckel. Java 编程思想: 第 4 版 [M]. 机械工业出版社, 2007.
+- 周志明. 深入理解 Java 虚拟机 [M]. 机械工业出版社, 2011.
+- [Threads and Locks](https://docs.oracle.com/javase/specs/jvms/se6/html/Threads.doc.html)
+- [线程通信](http://ifeve.com/thread-signaling/#missed_signal)
- [Java 线程面试题 Top 50](http://www.importnew.com/12773.html)
-- [Java 面试专题 - 多线程 & 并发编程 ](https://www.jianshu.com/p/e0c8d3dced8a)
-- [可重入内置锁](https://github.com/francistao/LearningNotes/blob/master/Part2/JavaConcurrent/%E5%8F%AF%E9%87%8D%E5%85%A5%E5%86%85%E7%BD%AE%E9%94%81.md)
+- [BlockingQueue](http://tutorials.jenkov.com/java-util-concurrent/blockingqueue.html)
+- [thread state java](https://stackoverflow.com/questions/11265289/thread-state-java)
+- [CSC 456 Spring 2012/ch7 MN](http://wiki.expertiza.ncsu.edu/index.php/CSC_456_Spring_2012/ch7_MN)
+- [Java - Understanding Happens-before relationship](https://www.logicbig.com/tutorials/core-java-tutorial/java-multi-threading/happens-before.html)
+- [6장 Thread Synchronization](https://www.slideshare.net/novathinker/6-thread-synchronization)
+- [How is Java's ThreadLocal implemented under the hood?](https://stackoverflow.com/questions/1202444/how-is-javas-threadlocal-implemented-under-the-hood/15653015)
+- [Concurrent](https://sites.google.com/site/webdevelopart/21-compile/06-java/javase/concurrent?tmpl=%2Fsystem%2Fapp%2Ftemplates%2Fprint%2F&showPrintDialog=1)
+- [JAVA FORK JOIN EXAMPLE](http://www.javacreed.com/java-fork-join-example/ "Java Fork Join Example")
+- [聊聊并发(八)——Fork/Join 框架介绍](http://ifeve.com/talk-concurrency-forkjoin/)
diff --git a/notes/JVM.md b/notes/Java 虚拟机.md
similarity index 61%
rename from notes/JVM.md
rename to notes/Java 虚拟机.md
index 215ef216..6fc0432f 100644
--- a/notes/JVM.md
+++ b/notes/Java 虚拟机.md
@@ -1,144 +1,132 @@
-* [内存模型](#内存模型)
- * [1. 程序计数器](#1-程序计数器)
- * [2. Java 虚拟机栈](#2-java-虚拟机栈)
- * [3. 本地方法栈](#3-本地方法栈)
- * [4. Java 堆](#4-java-堆)
- * [5. 方法区](#5-方法区)
- * [6. 运行时常量池](#6-运行时常量池)
- * [7. 直接内存](#7-直接内存)
-* [垃圾收集](#垃圾收集)
- * [1. 判断一个对象是否可回收](#1-判断一个对象是否可回收)
- * [1.1 引用计数](#11-引用计数)
- * [1.2 可达性](#12-可达性)
- * [1.3 引用类型](#13-引用类型)
- * [1.3.1 强引用](#131-强引用)
- * [1.3.2 软引用](#132-软引用)
- * [1.3.3 弱引用](#133-弱引用)
- * [1.3.4 虚引用](#134-虚引用)
- * [1.3 方法区的回收](#13-方法区的回收)
- * [1.4 finalize()](#14-finalize)
- * [2. 垃圾收集算法](#2-垃圾收集算法)
- * [2.1 标记-清除算法](#21-标记-清除算法)
- * [2.2 复制算法](#22-复制算法)
- * [2.3 标记-整理算法](#23-标记-整理算法)
- * [2.4 分代收集算法](#24-分代收集算法)
- * [3. 垃圾收集器](#3-垃圾收集器)
- * [3.1 Serial 收集器](#31-serial-收集器)
- * [3.2 ParNew 收集器](#32-parnew-收集器)
- * [3.3 Parallel Scavenge 收集器](#33-parallel-scavenge-收集器)
- * [3.4 Serial Old 收集器](#34-serial-old-收集器)
- * [3.5 Parallel Old 收集器](#35-parallel-old-收集器)
- * [3.6 CMS 收集器](#36-cms-收集器)
- * [3.7 G1 收集器](#37-g1-收集器)
- * [3.8 七种垃圾收集器的比较](#38-七种垃圾收集器的比较)
- * [4. 内存分配与回收策略](#4-内存分配与回收策略)
- * [4.1 优先在 Eden 分配](#41-优先在-eden-分配)
- * [4.2 大对象直接进入老年代](#42-大对象直接进入老年代)
- * [4.3 长期存活的对象进入老年代](#43-长期存活的对象进入老年代)
- * [4.4 动态对象年龄判定](#44-动态对象年龄判定)
- * [4.5 空间分配担保](#45-空间分配担保)
- * [4.6 Full GC 的触发条件](#46-full-gc-的触发条件)
- * [4.6.1 调用 System.gc()](#461-调用-systemgc)
- * [4.6.2 老年代空间不足](#462-老年代空间不足)
- * [4.6.3 空间分配担保失败](#463-空间分配担保失败)
- * [4.6.4 JDK 1.7 及以前的永久代空间不足](#464-jdk-17-及以前的永久代空间不足)
- * [4.6.5 Concurrent Mode Failure](#465-concurrent-mode-failure)
-* [类加载机制](#类加载机制)
- * [1 类的生命周期](#1-类的生命周期)
- * [2. 类初始化时机](#2-类初始化时机)
- * [3. 类加载过程](#3-类加载过程)
- * [3.1 加载](#31-加载)
- * [3.2 验证](#32-验证)
- * [3.3 准备](#33-准备)
- * [3.4 解析](#34-解析)
- * [3.5 初始化](#35-初始化)
- * [4. 类加载器](#4-类加载器)
- * [4.1 类与类加载器](#41-类与类加载器)
- * [4.2 类加载器分类](#42-类加载器分类)
- * [4.3 双亲委派模型](#43-双亲委派模型)
-* [JVM 参数](#jvm-参数)
+* [一、运行时数据区域](#一运行时数据区域)
+ * [程序计数器](#程序计数器)
+ * [虚拟机栈](#虚拟机栈)
+ * [本地方法栈](#本地方法栈)
+ * [堆](#堆)
+ * [方法区](#方法区)
+ * [运行时常量池](#运行时常量池)
+ * [直接内存](#直接内存)
+* [二、垃圾收集](#二垃圾收集)
+ * [判断一个对象是否可回收](#判断一个对象是否可回收)
+ * [垃圾收集算法](#垃圾收集算法)
+ * [垃圾收集器](#垃圾收集器)
+ * [内存分配与回收策略](#内存分配与回收策略)
+ * [Full GC 的触发条件](#full-gc-的触发条件)
+* [三、类加载机制](#三类加载机制)
+ * [类的生命周期](#类的生命周期)
+ * [类初始化时机](#类初始化时机)
+ * [类加载过程](#类加载过程)
+ * [类加载器](#类加载器)
+* [四、JVM 参数](#四jvm-参数)
* [GC 优化配置](#gc-优化配置)
* [GC 类型设置](#gc-类型设置)
+* [参考资料](#参考资料)
-# 内存模型
+# 一、运行时数据区域
-
+
-注:白色区域为线程私有的,蓝色区域为线程共享的。
+## 程序计数器
-## 1. 程序计数器
+记录正在执行的虚拟机字节码指令的地址(如果正在执行的是本地方法则为空)。
-记录正在执行的虚拟机字节码指令的地址(如果正在执行的是 Native 方法则为空)。
+## 虚拟机栈
-## 2. Java 虚拟机栈
+每个 Java 方法在执行的同时会创建一个栈帧用于存储局部变量表、操作数栈、常量池引用等信息。每一个方法从调用直至执行完成的过程,就对应着一个栈帧在 Java 虚拟机栈中入栈和出栈的过程。
-每个 Java 方法在执行的同时会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每一个方法从调用直至执行完成的过程,就对应着一个栈帧在 Java 虚拟机栈中入栈和出栈的过程。
+
+
+可以通过 -Xss 这个虚拟机参数来指定一个程序的 Java 虚拟机栈内存大小:
+
+```java
+java -Xss=512M HackTheJava
+```
该区域可能抛出以下异常:
1. 当线程请求的栈深度超过最大值,会抛出 StackOverflowError 异常;
2. 栈进行动态扩展时如果无法申请到足够内存,会抛出 OutOfMemoryError 异常。
-## 3. 本地方法栈
+## 本地方法栈
+
+本地方法不是用 Java 实现,对待这些方法需要特别处理。
与 Java 虚拟机栈类似,它们之间的区别只不过是本地方法栈为本地方法服务。
-## 4. Java 堆
+
+
+## 堆
所有对象实例都在这里分配内存。
-这块区域是垃圾收集器管理的主要区域("GC 堆 ")。现在收集器基本都是采用分代收集算法,Java 堆还可以分成:新生代和老年代(新生代还可以分成 Eden 空间、From Survivor 空间、To Survivor 空间等)。
+是垃圾收集的主要区域("GC 堆"),现代的垃圾收集器基本都是采用分代收集算法,该算法的思想是针对不同的对象采取不同的垃圾回收算法,因此虚拟机把 Java 堆分成以下三块:
-不需要连续内存,可以通过 -Xmx 和 -Xms 来控制动态扩展内存大小,如果动态扩展失败会抛出 OutOfMemoryError 异常。
+- 新生代(Young Generation)
+- 老年代(Old Generation)
+- 永久代(Permanent Generation)
-## 5. 方法区
+当一个对象被创建时,它首先进入新生代,之后有可能被转移到老年代中。新生代存放着大量的生命很短的对象,因此新生代在三个区域中垃圾回收的频率最高。为了更高效地进行垃圾回收,把新生代继续划分成以下三个空间:
-用于存放已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
+- Eden
+- From Survivor
+- To Survivor
+
+
+
+Java 堆不需要连续内存,并且可以动态增加其内存,增加失败会抛出 OutOfMemoryError 异常。
+
+可以通过 -Xms 和 -Xmx 两个虚拟机参数来指定一个程序的 Java 堆内存大小,第一个参数设置初始值,第二个参数设置最大值。
+
+```java
+java -Xms=1M -Xmx=2M HackTheJava
+```
+
+## 方法区
+
+用于存放已被加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
和 Java 堆一样不需要连续的内存,并且可以动态扩展,动态扩展失败一样会抛出 OutOfMemoryError 异常。
对这块区域进行垃圾回收的主要目标是对常量池的回收和对类的卸载,但是一般比较难实现,HotSpot 虚拟机把它当成永久代来进行垃圾回收。
-## 6. 运行时常量池
+## 运行时常量池
运行时常量池是方法区的一部分。
-类加载后,Class 文件中的常量池(用于存放编译期生成的各种字面量和符号引用)就会被放到这个区域。
+Class 文件中的常量池(编译器生成的各种字面量和符号引用)会在类加载后被放入这个区域。
-在运行期间也可以用过 String 类的 intern() 方法将新的常量放入该区域。
+除了在编译期生成的常量,还允许动态生成,例如 String 类的 intern()。这部分常量也会被放入运行时常量池。
-## 7. 直接内存
+## 直接内存
-在 JDK 1.4 中新加入了 NIO 类,引入了一种基于通道(Channel)与缓冲区(Buffer)的 I/O 方式,它可以使用 Native 函数库直接分配堆外内存,然后通过一个存储在 Java 堆里的 DirectByteBuffer 对象作为这块内存的引用进行操作。这样能在一些场景中显著提高性能,因为避免了在 Java 堆和 Native 堆中来回复制数据。
+在 JDK 1.4 中新加入了 NIO 类,它可以使用 Native 函数库直接分配堆外内存,然后通过一个存储在 Java 堆里的 DirectByteBuffer 对象作为这块内存的引用进行操作。这样能在一些场景中显著提高性能,因为避免了在 Java 堆和 Native 堆中来回复制数据。
-# 垃圾收集
+# 二、垃圾收集
-程序计数器、虚拟机栈和本地方法栈这三个区域属于线程私有的,只存在于线程的生命周期内,线程结束之后也会消失,因此不需要对这三个区域进行垃圾回收。
+程序计数器、虚拟机栈和本地方法栈这三个区域属于线程私有的,只存在于线程的生命周期内,线程结束之后也会消失,因此不需要对这三个区域进行垃圾回收。垃圾回收主要是针对 Java 堆和方法区进行。
-垃圾回收主要是针对 Java 堆和方法区进行。
+## 判断一个对象是否可回收
-## 1. 判断一个对象是否可回收
+### 1. 引用计数
-### 1.1 引用计数
+给对象添加一个引用计数器,当对象增加一个引用时计数器加 1,引用失效时计数器减 1。引用计数为 0 的对象可被回收。
-给对象添加一个引用计数器,当对象增加一个引用时计数器加 1,引用失效时计数器减 1。
-
-引用计数为 0 的对象可被回收。
-
-两个对象会出现循环引用问题,此时引用计数器永远不为 0,导致 GC 收集器无法回收。
+两个对象出现循环引用的情况下,此时引用计数器永远不为 0,导致无法对它们进行回收。
```java
objA.instance = objB;
objB.instance = objA;
```
-### 1.2 可达性
+### 2. 可达性
通过 GC Roots 作为起始点进行搜索,能够到达到的对象都是都是可用的,不可达的对象可被回收。
+
+
GC Roots 一般包含以下内容:
1. 虚拟机栈中引用的对象
@@ -146,64 +134,66 @@ GC Roots 一般包含以下内容:
3. 方法区中的常量引用的对象
4. 本地方法栈中引用的对象
-### 1.3 引用类型
+### 3. 引用类型
-无论是通过引用计算算法判断对象的引用数量,还是通过可达性分析算法判断对象的引用链是否可达,判定独享是否存活都与“引用”有关。
+无论是通过引用计算算法判断对象的引用数量,还是通过可达性分析算法判断对象的引用链是否可达,判定对象是否存活都与引用有关。
-#### 1.3.1 强引用
+Java 对引用的概念进行了扩充,引入四种强度不同的引用类型。
+
+**(一)强引用**
只要强引用存在,垃圾回收器永远不会回收调掉被引用的对象。
+使用 new 一个新对象的方式来创建强引用。
+
```java
Object obj = new Object();
```
-#### 1.3.2 软引用
+**(二)软引用**
+用来描述一些还有用但是并非必需的对象。
-非必须引用,内存溢出之前进行回收。
+在系统将要发生内存溢出异常之前,将会对这些对象列进回收范围之中进行第二次回收。
+
+软引用主要用来实现类似缓存的功能,在内存足够的情况下直接通过软引用取值,无需从繁忙的真实来源获取数据,提升速度;当内存不足时,自动删除这部分缓存数据,从真正的来源获取这些数据。
+
+使用 SoftReference 类来实现软引用。
```java
Object obj = new Object();
SoftReference sf = new SoftReference(obj);
-obj = null;
-sf.get();
```
-sf 是对 obj 的一个软引用,通过 sf.get() 方法可以取到这个对象,当然,当这个对象被标记为需要回收的对象时,则返回 null;
-
-软引用主要用来实现类似缓存的功能,在内存足够的情况下直接通过软引用取值,无需从繁忙的真实来源查询数据,提升速度;当内存不足时,自动删除这部分缓存数据,从真正的来源查询这些数据。
-
-
-#### 1.3.3 弱引用
+**(三)弱引用**
只能生存到下一次垃圾收集发生之前,当垃圾收集器工作时,无论当前内存是否足够,都会被回收。
+使用 WeakReference 类来实现弱引用。
+
```java
Object obj = new Object();
WeakReference wf = new WeakReference(obj);
-obj = null;
-wf.get();
-wf.isEnQueued();
```
-#### 1.3.4 虚引用
+**(四)虚引用**
-又称为幽灵引用或者幻影引用,一个对象是否有虚引用的存在,完全不会对其生存时间构成影响,也无法通过虚引用取得一个对象实例。为一个对象设置虚引用关联的唯一目的就是能在这个对象被收集器回收时收到一个系统通知。
+又称为幽灵引用或者幻影引用。一个对象是否有虚引用的存在,完全不会对其生存时间构成影响,也无法通过虚引用取得一个对象实例。
+
+为一个对象设置虚引用关联的唯一目的就是能在这个对象被收集器回收时收到一个系统通知。
+
+使用 PhantomReference 来实现虚引用。
```java
Object obj = new Object();
PhantomReference pf = new PhantomReference(obj);
-obj=null;
-pf.get();
-pf.isEnQueued();
```
-### 1.3 方法区的回收
+### 4. 方法区的回收
-在方法区主要是对常量池的回收和对类的卸载。
+因为方法区主要存放永久代对象,而永久代对象的回收率比新生代差很多,因此在方法区上进行回收性价比不高。
-常量池的回收和堆中对象回收类似。
+主要是对常量池的回收和对类的卸载。
类的卸载条件很多,需要满足以下三个条件,并且满足了也不一定会被卸载:
@@ -215,61 +205,59 @@ pf.isEnQueued();
在大量使用反射、动态代理、CGLib 等 ByteCode 框架、动态生成 JSP 以及 OSGo 这类频繁自定义 ClassLoader 的场景都需要虚拟机具备类卸载功能,以保证不会出现内存溢出。
-### 1.4 finalize()
-
-当一个对象可被回收时,如果该对象有必要执行 finalize() 方法,那么就有可能可能通过在该方法中让对象重新被引用,从而实现自救。
+### 5. finalize()
finalize() 类似 C++ 的析构函数,用来做关闭外部资源等工作。但是 try-finally 等方式可以做的更好,并且该方法运行代价高昂,不确定性大,无法保证各个对象的调用顺序,因此最好不要使用。
-## 2. 垃圾收集算法
+当一个对象可被回收时,如果需要执行该对象的 finalize() 方法,那么就有可能通过在该方法中让对象重新被引用,从而实现自救。
-### 2.1 标记-清除算法
+## 垃圾收集算法
-
+### 1. 标记 - 清除
+
+
将需要回收的对象进行标记,然后清除。
不足:
-1. 标记和清除过程效率都不高
-2. 会产生大量碎片
+1. 标记和清除过程效率都不高;
+2. 会产生大量碎片,内存碎片过多可能导致无法给大对象分配内存。
-之后的算法都是基于该算法进行改进。
+### 2. 复制
-### 2.2 复制算法
-
-
+
将内存划分为大小相等的两块,每次只使用其中一块,当这一块内存用完了就将还存活的对象复制到另一块上面,然后再把使用过的内存空间进行一次清理。
主要不足是只使用了内存的一半。
-现在的商业虚拟机都采用这种收集算法来回收新生代,但是并不是将内存划分为大小相等的两块,而是分为一块较大的 Eden 空间和两块较小的 Survior 空间,每次使用 Eden 空间和其中一块 Survivor。在回收时,将 Eden 和 Survivor 中还存活着的对象一次性复制到另一块 Survivor 空间上,最后清理 Eden 和 Survivor。HotSpot 虚拟机的 Eden 和 Survivor 的大小比例默认为 8:1,保证了内存的利用率达到 90 %。如果每次回收有多于 10% 的对象存活,那么一块 Survivor 空间就不够用了,需要依赖于老年代进行分配担保,也就是借用老年代的空间。
+现在的商业虚拟机都采用这种收集算法来回收新生代,但是并不是将内存划分为大小相等的两块,而是分为一块较大的 Eden 空间和两块较小的 Survior 空间,每次使用 Eden 空间和其中一块 Survivor。在回收时,将 Eden 和 Survivor 中还存活着的对象一次性复制到另一块 Survivor 空间上,最后清理 Eden 和 使用过的那一块 Survivor。HotSpot 虚拟机的 Eden 和 Survivor 的大小比例默认为 8:1,保证了内存的利用率达到 90 %。如果每次回收有多于 10% 的对象存活,那么一块 Survivor 空间就不够用了,此时需要依赖于老年代进行分配担保,也就是借用老年代的空间。
-### 2.3 标记-整理算法
+### 3. 标记 - 整理
-
+
让所有存活的对象都向一端移动,然后直接清理掉端边界以外的内存。
-### 2.4 分代收集算法
+### 4. 分代收集
-现在的商业虚拟机采用分代收集算法,它使用了前面介绍的几种收集算法,根据对象存活周期将内存划分为几块,不同块采用适当的收集算法。
+现在的商业虚拟机采用分代收集算法,它根据对象存活周期将内存划分为几块,不同块采用适当的收集算法。
一般将 Java 堆分为新生代和老年代。
1. 新生代使用:复制算法
-2. 老年代使用:标记 - 清理 或者 标记 - 整理 算法。
+2. 老年代使用:标记 - 清理 或者 标记 - 整理 算法
-## 3. 垃圾收集器
+## 垃圾收集器
-
+
以上是 HotSpot 虚拟机中的 7 个垃圾收集器,连线表示垃圾收集器可以配合使用。
-### 3.1 Serial 收集器
+### 1. Serial 收集器
-
+
它是单线程的收集器,不仅意味着只会使用一个线程进行垃圾收集工作,更重要的是它在进行垃圾收集时,必须暂停所有其他工作线程,往往造成过长的等待时间。
@@ -277,9 +265,9 @@ finalize() 类似 C++ 的析构函数,用来做关闭外部资源等工作。
在 Client 应用场景中,分配给虚拟机管理的内存一般来说不会很大,该收集器收集几十兆甚至一两百兆的新生代停顿时间可以控制在一百多毫秒以内,只要不是太频繁,这点停顿是可以接受的。
-### 3.2 ParNew 收集器
+### 2. ParNew 收集器
-
+
它是 Serial 收集器的多线程版本。
@@ -287,7 +275,7 @@ finalize() 类似 C++ 的析构函数,用来做关闭外部资源等工作。
默认开始的线程数量与 CPU 数量相同,可以使用 -XX:ParallelGCThreads 参数来设置线程数。
-### 3.3 Parallel Scavenge 收集器
+### 3. Parallel Scavenge 收集器
是并行的多线程收集器。
@@ -299,28 +287,28 @@ finalize() 类似 C++ 的析构函数,用来做关闭外部资源等工作。
还提供了一个参数 -XX:+UseAdaptiveSizePolicy,这是一个开关参数,打开参数后,就不需要手工指定新生代的大小(-Xmn)、Eden 和 Survivor 区的比例(-XX:SurvivorRatio)、晋升老年代对象年龄(-XX:PretenureSizeThreshold)等细节参数了,虚拟机会根据当前系统的运行情况收集性能监控信息,动态调整这些参数以提供最合适的停顿时间或者最大的吞吐量,这种方式称为 GC 自适应的调节策略(GC Ergonomics)。自适应调节策略也是它与 ParNew 收集器的一个重要区别。
-### 3.4 Serial Old 收集器
+### 4. Serial Old 收集器
-
+
Serial Old 是 Serial 收集器的老年代版本,也是给 Client 模式下的虚拟机使用。如果用在 Server 模式下,它有两大用途:
1. 在 JDK 1.5 以及之前版本(Parallel Old 诞生以前)中与 Parallel Scavenge 收集器搭配使用。
2. 作为 CMS 收集器的后备预案,在并发收集发生 Concurrent Mode Failure 时使用。
-### 3.5 Parallel Old 收集器
+### 5. Parallel Old 收集器
-
+
是 Parallel Scavenge 收集器的老年代版本。
在注重吞吐量以及 CPU 资源敏感的场合,都可以优先考虑 Parallel Scavenge 加 Parallel Old 收集器。
-### 3.6 CMS 收集器
+### 6. CMS 收集器
-
+
-CMS(Concurrent Mark Sweep),从 Mark Sweep 可以知道它是基于 标记 - 清除 算法实现的。
+CMS(Concurrent Mark Sweep),从 Mark Sweep 可以知道它是基于标记 - 清除算法实现的。
特点:并发收集、低停顿。
@@ -339,17 +327,17 @@ CMS(Concurrent Mark Sweep),从 Mark Sweep 可以知道它是基于 标记
2. 无法处理浮动垃圾。由于并发清理阶段用户线程还在运行着,伴随程序运行自然就还会有新的垃圾不断产生。这一部分垃圾出现在标记过程之后,CMS 无法在当次收集中处理掉它们,只好留到下一次 GC 时再清理掉,这一部分垃圾就被称为“浮动垃圾”。也是由于在垃圾收集阶段用户线程还需要运行,那也就还需要预留有足够的内存空间给用户线程使用,因此它不能像其他收集器那样等到老年代几乎完全被填满了再进行收集,需要预留一部分空间提供并发收集时的程序运作使用。可以使用 -XX:CMSInitiatingOccupancyFraction 的值来改变触发收集器工作的内存占用百分比,JDK 1.5 默认设置下该值为 68,也就是当老年代使用了 68% 的空间之后会触发收集器工作。如果该值设置的太高,导致浮动垃圾无法保存,那么就会出现 Concurrent Mode Failure,此时虚拟机将启动后备预案:临时启用 Serial Old 收集器来重新进行老年代的垃圾收集。
-3. 标记 - 清除算法导致的空间碎片,给大对象分配带来很大麻烦,往往出现老年代空间剩余,但无法找到足够大连续空间来分配当前对象,不得不提前出发一次 Full GC。
+3. 标记 - 清除算法导致的空间碎片,给大对象分配带来很大麻烦,往往出现老年代空间剩余,但无法找到足够大连续空间来分配当前对象,不得不提前触发一次 Full GC。
-### 3.7 G1 收集器
+### 7. G1 收集器
-
+
G1(Garbage-First)收集器是当今收集器技术发展最前沿的成果之一,它是一款面向服务端应用的垃圾收集器,HotSpot 开发团队赋予它的使命是(在比较长期的)未来可以替换掉 JDK 1.5 中发布的 CMS 收集器。
具备如下特点:
-- 并行与并发:能充分利用多 CPU 环境下的硬件优势,使用多个 CPU 来缩短停顿时间;
+- 并行与并发:能充分利用多 CPU 环境下的硬件优势,使用多个 CPU 来缩短停顿时间。
- 分代收集:分代概念依然得以保留,虽然它不需要其它收集器配合就能独立管理整个 GC 堆,但它能够采用不同方式去处理新创建的对象和已存活一段时间、熬过多次 GC 的旧对象来获取更好的收集效果。
- 空间整合:整体来看是基于“标记 - 整理”算法实现的收集器,从局部(两个 Region 之间)上来看是基于“复制”算法实现的,这意味着运行期间不会产生内存空间碎片。
- 可预测的停顿:这是它相对 CMS 的一大优势,降低停顿时间是 G1 和 CMS 共同的关注点,但 G1 除了降低停顿外,还能建立可预测的停顿时间模型,能让使用者明确指定在一个长度为 M 毫秒的时间片段内,消耗在 GC 上的时间不得超过 N 毫秒,这几乎已经是实时 Java(RTSJ)的垃圾收集器的特征了。
@@ -367,73 +355,80 @@ Region 不可能是孤立的,一个对象分配在某个 Region 中,可以
3. 最终标记:为了修正在并发标记期间因用户程序继续运作而导致标记产生变动的那一部分标记记录,虚拟机将这段时间对象变化记录在线程的 Remembered Set Logs 里面,最终标记阶段需要把 Remembered Set Logs 的数据合并到 Remembered Set 中。这阶段需要停顿线程,但是可并行执行。
4. 筛选回收:首先对各个 Region 中的回收价值和成本进行排序,根据用户所期望的 GC 停顿是时间来制定回收计划。此阶段其实也可以做到与用户程序一起并发执行,但是因为只回收一部分 Region,时间是用户可控制的,而且停顿用户线程将大幅度提高收集效率。
-### 3.8 七种垃圾收集器的比较
+### 8. 七种垃圾收集器的比较
| 收集器 | 串行、并行 or 并发 | 新生代 / 老年代 | 算法 | 目标 | 适用场景 |
-| --- | --- | --- | --- | --- | --- |
+| :---: | :---: | :---: | :---: | :---: | :---: |
| **Serial** | 串行 | 新生代 | 复制算法 | 响应速度优先 | 单 CPU 环境下的 Client 模式 |
-| **Serial Old** | 串行 | 老年代 | 标记 - 整理 | 响应速度优先 | 单 CPU 环境下的 Client 模式、CMS 的后备预案 |
+| **Serial Old** | 串行 | 老年代 | 标记-整理 | 响应速度优先 | 单 CPU 环境下的 Client 模式、CMS 的后备预案 |
| **ParNew** | 并行 | 新生代 | 复制算法 | 响应速度优先 | 多 CPU 环境时在 Server 模式下与 CMS 配合 |
| **Parallel Scavenge** | 并行 | 新生代 | 复制算法 | 吞吐量优先 | 在后台运算而不需要太多交互的任务 |
-| **Parallel Old** | 并行 | 老年代 | 标记 - 整理 | 吞吐量优先 | 在后台运算而不需要太多交互的任务 |
-| **CMS** | 并发 | 老年代 | 标记 - 清除 | 响应速度优先 | 集中在互联网站或 B/S 系统服务端上的 Java 应用 |
-| **G1** | 并发 | both | 标记 - 整理 + 复制算法 | 响应速度优先 | 面向服务端应用,将来替换 CMS |
+| **Parallel Old** | 并行 | 老年代 | 标记-整理 | 吞吐量优先 | 在后台运算而不需要太多交互的任务 |
+| **CMS** | 并发 | 老年代 | 标记-清除 | 响应速度优先 | 集中在互联网站或 B/S 系统服务端上的 Java 应用 |
+| **G1** | 并发 | both | 标记-整理 + 复制算法 | 响应速度优先 | 面向服务端应用,将来替换 CMS |
-## 4. 内存分配与回收策略
+## 内存分配与回收策略
-### 4.1 优先在 Eden 分配
+对象的内存分配,也就是在堆上分配。主要分配在新生代的 Eden 区上,少数情况下也可能直接分配在老年代中。
-大多数情况下,对象在新生代 Eden 区分配,当 Eden 区空间不够时,发起 Minor GC;
+### 1. 优先在 Eden 分配
-### 4.2 大对象直接进入老年代
+大多数情况下,对象在新生代 Eden 区分配,当 Eden 区空间不够时,发起 Minor GC。
-提供 -XX:PretenureSizeThreshold 参数,大于此值的对象直接在老年代分配,避免在 Eden 区和 Survivor 区之间的大量内存复制;
-### 4.3 长期存活的对象进入老年代
+关于 Minor GC 和 Full GC:
-JVM 为对象定义年龄计数器,经过 Minor GC 依然存活且被 Survivor 区容纳的,移动到 Survivor 区,年龄加 1,每经历一次 Minor GC 不被清理则年龄加 1,增加到一定年龄则移动到老年区(默认 15 岁,通过 -XX:MaxTenuringThreshold 设置);
+- Minor GC:发生在新生代上,因为新生代对象存活时间很短,因此 Minor GC 会频繁执行,执行的速度一般也会比较快。
+- Full GC:发生在老年代上,老年代对象和新生代的相反,其存活时间长,因此 Full GC 很少执行,而且执行速度会比 Minor GC 慢很多。
+### 2. 大对象直接进入老年代
-### 4.4 动态对象年龄判定
+大对象是指需要连续内存空间的对象,最典型的大对象是那种很长的字符串以及数组。经常出现大对象会提前触发垃圾收集以获取足够的连续空间分配给大对象。
-若 Survivor 区中同年龄所有对象大小总和大于 Survivor 空间一半,则年龄大于等于该年龄的对象可以直接进入老年代;
+提供 -XX:PretenureSizeThreshold 参数,大于此值的对象直接在老年代分配,避免在 Eden 区和 Survivor 区之间的大量内存复制。
-### 4.5 空间分配担保
+### 3. 长期存活的对象进入老年代
-在发生 Minor GC 之前,JVM 先检查老年代最大可用连续空间是否大于新生代所有对象总空间,成立的话 Minor GC 确认是安全的;否则继续检查老年代最大可用连续空间是否大于历次晋升到老年代对象的平均大小,大于的话进行 Minor GC,小于的话进行 Full GC。
+JVM 为对象定义年龄计数器,经过 Minor GC 依然存活,并且能被 Survivor 区容纳的,移被移到 Survivor 区,年龄就增加 1 岁,增加到一定年龄则移动到老年代中(默认 15 岁,通过 -XX:MaxTenuringThreshold 设置)。
-## 4.6 Full GC 的触发条件
+### 4. 动态对象年龄判定
+
+JVM 并不是永远地要求对象的年龄必须达到 MaxTenuringThreshold 才能晋升老年代,如果在 Survivor 区中相同年龄所有对象大小的总和大于 Survivor 空间的一半,则年龄大于或等于该年龄的对象可以直接进入老年代,无需等待 MaxTenuringThreshold 中要求的年龄。
+
+### 5. 空间分配担保
+
+在发生 Minor GC 之前,JVM 先检查老年代最大可用的连续空间是否大于新生代所有对象总空间,如果条件成立的话,那么 Minor GC 可以确认是安全的;如果不成立的话 JVM 会查看 HandlePromotionFailure 设置值是否允许担保失败,如果允许那么就会继续检查老年代最大可用的连续空间是否大于历次晋升到老年代对象的平均大小,如果大于,将尝试着进行一次 Minor GC,尽管这次 Minor GC 是有风险的;如果小于,或者 HandlePromotionFailure 设置不允许冒险,那这时也要改为进行一次 Full GC。
+
+## Full GC 的触发条件
对于 Minor GC,其触发条件非常简单,当 Eden 区空间满时,就将触发一次 Minor GC。而 Full GC 则相对复杂,有以下条件:
-### 4.6.1 调用 System.gc()
+### 1. 调用 System.gc()
-此方法的调用是建议 JVM 进行 Full GC,虽然只是建议而非一定,但很多情况下它会触发 Full GC,从而增加 Full GC 的频率,也即增加了间歇性停顿的次数。因此强烈建议能不使用此方法就不要使用,让虚拟机自己去管理它的内存,可通过 -XX:+ DisableExplicitGC 来禁止 RMI 调用 System.gc()。
+此方法的调用是建议 JVM 进行 Full GC,虽然只是建议而非一定,但很多情况下它会触发 Full GC,从而增加 Full GC 的频率,也即增加了间歇性停顿的次数。因此强烈建议能不使用此方法就不要使用,让虚拟机自己去管理它的内存。可通过 -XX:+ DisableExplicitGC 来禁止 RMI 调用 System.gc()。
-### 4.6.2 老年代空间不足
+### 2. 老年代空间不足
-老年代空间不足的常见场景为前文所讲的大对象直接进入老年代、长期存活的对象进入老年代等,当执行 Full GC 后空间仍然不足,则抛出如下错误: Java.lang.OutOfMemoryError: Java heap space 为避免以上两种状况引起的 Full GC,调优时应尽量做到让对象在 Minor GC 阶段被回收、让对象在新生代多存活一段时间及不要创建过大的对象及数组。
+老年代空间不足的常见场景为前文所讲的大对象直接进入老年代、长期存活的对象进入老年代等,当执行 Full GC 后空间仍然不足,则抛出 Java.lang.OutOfMemoryError。为避免以上原因引起的 Full GC,调优时应尽量做到让对象在 Minor GC 阶段被回收、让对象在新生代多存活一段时间以及不要创建过大的对象及数组。
-### 4.6.3 空间分配担保失败
+### 3. 空间分配担保失败
使用复制算法的 Minor GC 需要老年代的内存空间作担保,如果出现了 HandlePromotionFailure 担保失败,则会触发 Full GC。
-### 4.6.4 JDK 1.7 及以前的永久代空间不足
+### 4. JDK 1.7 及以前的永久代空间不足
-在 JDK 1.7 及以前,HotSpot 虚拟机中的方法区是用永久代实现的,永久代中存放的为一些 class 的信息、常量、静态变量等数据,当系统中要加载的类、反射的类和调用的方法较多时,Permanet Generation 可能会被占满,在未配置为采用 CMS GC 的情况下也会执行 Full GC。如果经过 Full GC 仍然回收不了,那么 JVM 会抛出如下错误信息:java.lang.OutOfMemoryError: PermGen space 为避免 PermGen 占满造成 Full GC 现象,可采用的方法为增大 PermGen 空间或转为使用 CMS GC。
+在 JDK 1.7 及以前,HotSpot 虚拟机中的方法区是用永久代实现的,永久代中存放的为一些 Class 的信息、常量、静态变量等数据,当系统中要加载的类、反射的类和调用的方法较多时,永久代可能会被占满,在未配置为采用 CMS GC 的情况下也会执行 Full GC。如果经过 Full GC 仍然回收不了,那么 JVM 会抛出 java.lang.OutOfMemoryError,为避免以上原因引起的 Full GC,可采用的方法为增大永久代空间或转为使用 CMS GC。
-在 JDK 1.8 中用元空间替换了永久代作为方法区的实现,元空间是本地内存,因此减少了一种 Full GC 触发的可能性。
-
-### 4.6.5 Concurrent Mode Failure
+### 5. Concurrent Mode Failure
执行 CMS GC 的过程中同时有对象要放入老年代,而此时老年代空间不足(有时候“空间不足”是 CMS GC 时当前的浮动垃圾过多导致暂时性的空间不足触发 Full GC),便会报 Concurrent Mode Failure 错误,并触发 Full GC。
-# 类加载机制
+# 三、类加载机制
类是在运行期间动态加载的。
-## 1 类的生命周期
+## 类的生命周期
-
+
包括以下 7 个阶段:
@@ -447,11 +442,11 @@ JVM 为对象定义年龄计数器,经过 Minor GC 依然存活且被 Survivor
其中解析过程在某些情况下可以在初始化阶段之后再开始,这是为了支持 Java 的动态绑定。
-## 2. 类初始化时机
+## 类初始化时机
-虚拟机规范中并没有强制约束何时进行加载,但是规范严格规定了有且只有下列五种情况必须对类进行初始化:( 加载、验证、准备都会随着发生 )
+虚拟机规范中并没有强制约束何时进行加载,但是规范严格规定了有且只有下列五种情况必须对类进行初始化(加载、验证、准备都会随着发生):
-1. 遇到 new、getstatic、putstatic、invokestatic 这四条字节码指令时,如果类没有进行过初始化,则必须先触发其初始化。最常见的生成这 4 条指令的场景是:使用 new 关键字实例化对象的时候;读取或设置一个类的静态字段(被 final 修饰、已在编译器把结果放入常量池的静态字段除外)的时候;以及调用一个类的静态方法的时候。
+1. 遇到 new、getstatic、putstatic、invokestatic 这四条字节码指令时,如果类没有进行过初始化,则必须先触发其初始化。最常见的生成这 4 条指令的场景是:使用 new 关键字实例化对象的时候;读取或设置一个类的静态字段(被 final 修饰、已在编译期把结果放入常量池的静态字段除外)的时候;以及调用一个类的静态方法的时候。
2. 使用 java.lang.reflect 包的方法对类进行反射调用的时候,如果类没有进行初始化,则需要先触发其初始化。
@@ -459,33 +454,33 @@ JVM 为对象定义年龄计数器,经过 Minor GC 依然存活且被 Survivor
4. 当虚拟机启动时,用户需要指定一个要执行的主类(包含 main() 方法的那个类),虚拟机会先初始化这个主类;
-5. 当使用 jdk1.7 的动态语言支持时,如果一个 java.lang.invoke.MethodHandle 实例最后的解析结果为 REF_getStatic, REF_putStatic, REF_invokeStatic 的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化;
+5. 当使用 JDK.7 的动态语言支持时,如果一个 java.lang.invoke.MethodHandle 实例最后的解析结果为 REF_getStatic, REF_putStatic, REF_invokeStatic 的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化;
以上 5 种场景中的行为称为对一个类进行主动引用。除此之外,所有引用类的方式都不会触发初始化,称为被动引用。被动引用的常见例子包括:
-1\. 通过子类引用父类的静态字段,不会导致子类初始化。
+- 通过子类引用父类的静态字段,不会导致子类初始化。
```java
System.out.println(SubClass.value); // value 字段在 SuperClass 中定义
```
-2\. 通过数组定义来引用类,不会触发此类的初始化。该过程会对数组类进行初始化,数组类是一个由虚拟机自动生成的、直接继承自 Object 的子类,其中包含了数组的属性和方法。
+- 通过数组定义来引用类,不会触发此类的初始化。该过程会对数组类进行初始化,数组类是一个由虚拟机自动生成的、直接继承自 Object 的子类,其中包含了数组的属性和方法。
```java
SuperClass[] sca = new SuperClass[10];
```
-3\. 常量在编译阶段会存入调用类的常量池中,本质上并没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化。
+- 常量在编译阶段会存入调用类的常量池中,本质上并没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化。
```java
System.out.println(ConstClass.HELLOWORLD);
```
-## 3. 类加载过程
+## 类加载过程
包含了加载、验证、准备、解析和初始化这 5 个阶段。
-### 3.1 加载
+### 1. 加载
加载是类加载的一个阶段,注意不要混淆。
@@ -504,18 +499,29 @@ System.out.println(ConstClass.HELLOWORLD);
- 从数据库读取,这种场景相对少见,例如有些中间件服务器(如 SAP Netweaver)可以选择把程序安装到数据库中来完成程序代码在集群间的分发。
...
-### 3.2 验证
+### 2. 验证
确保 Class 文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。
主要有以下 4 个阶段:
-1. 文件格式验证
-2. 元数据验证(对字节码描述的信息进行语义分析)
-3. 字节码验证(通过数据流和控制流分析,确保程序语义是合法、符合逻辑的,将对类的方法体进行校验分析)
-4. 符号引用验证
+(一)文件格式验证
-### 3.3 准备
+验证字节流是否符合 Class 文件格式的规范,并且能被当前版本的虚拟机处理。
+
+(二)元数据验证
+
+对字节码描述的信息进行语义分析,以保证其描述的信息符合 Java 语言规范的要求。
+
+(三)字节码验证
+
+通过数据流和控制流分析,确保程序语义是合法、符合逻辑的。
+
+(四)符号引用验证
+
+发生在虚拟机将符号引用转换为直接引用的时候,对类自身以外(常量池中的各种符号引用)的信息进行匹配性校验。
+
+### 3. 准备
类变量是被 static 修饰的变量,准备阶段为类变量分配内存并设置初始值,使用的是方法区的内存。
@@ -533,13 +539,13 @@ public static int value = 123;
public static final int value = 123;
```
-### 3.4 解析
+### 4. 解析
将常量池的符号引用替换为直接引用的过程。
-### 3.5 初始化
+### 5. 初始化
-初始化阶段即虚拟机执行类构造器 <clinit>() 方法的过程。
+初始化阶段才真正开始执行类中的定义的 Java 程序代码。初始化阶段即虚拟机执行类构造器 <clinit>() 方法的过程。
在准备阶段,类变量已经赋过一次系统要求的初始值,而在初始化阶段,根据程序员通过程序制定的主观计划去初始化类变量和其它资源。
@@ -563,18 +569,18 @@ public class Test {
```java
static class Parent {
- public static int A = 1;
- static {
- A = 2;
- }
+ public static int A = 1;
+ static {
+ A = 2;
+ }
}
static class Sub extends Parent {
- public static int B = A;
+ public static int B = A;
}
public static void main(String[] args) {
- System.out.println(Sub.B); // 输出结果是父类中的静态变量值 A,也就是 2
+ System.out.println(Sub.B); // 输出结果是父类中的静态变量 A 的值 ,也就是 2。
}
```
@@ -584,47 +590,49 @@ public static void main(String[] args) {
- 虚拟机会保证一个类的 <clinit>() 方法在多线程环境下被正确的加锁和同步,如果多个线程同时初始化一个类,只会有一个线程执行这个类的 <clinit>() 方法,其它线程都会阻塞等待,直到活动线程执行 <clinit>() 方法完毕。如果在一个类的 <clinit>() 方法中有耗时的操作,就可能造成多个进程阻塞,在实际过程中此种阻塞很隐蔽。
-## 4. 类加载器
+## 类加载器
虚拟机设计团队把类加载阶段中的“通过一个类的全限定名来获取描述此类的二进制字节流 ( 即字节码 )”这个动作放到 Java 虚拟机外部去实现,以便让应用程序自己决定如何去获取所需要的类。实现这个动作的代码模块称为“类加载器”。
-### 4.1 类与类加载器
+### 1. 类与类加载器
-对于任意一个类,都需要由加载它的类加载器和这个类本身一同确立其在 Java 虚拟机中的唯一性,每一个类加载器,都拥有一个独立的类名称空间。通俗而言:比较两个类是否“相等”(这里所指的“相等”,包括类的 Class 对象的 equals() 方法、isAssignableFrom() 方法、isInstance() 方法的返回结果,也包括使用 instanceof() 关键字对做对象所属关系判定等情况),只有在这两个类时由同一个类加载器加载的前提下才有意义,否则,即使这两个类来源于同一个 Class 文件,被同一个虚拟机加载,只要加载它们的类加载器不同,那这两个类就必定不相等。
+对于任意一个类,都需要由加载它的类加载器和这个类本身一同确立其在 Java 虚拟机中的唯一性,每一个类加载器,都拥有一个独立的类名称空间。通俗而言:比较两个类是否“相等”(这里所指的“相等”,包括类的 Class 对象的 equals() 方法、isAssignableFrom() 方法、isInstance() 方法的返回结果,也包括使用 instanceof 关键字做对象所属关系判定等情况),只有在这两个类是由同一个类加载器加载的前提下才有意义,否则,即使这两个类来源于同一个 Class 文件,被同一个虚拟机加载,只要加载它们的类加载器不同,那这两个类就必定不相等。
-### 4.2 类加载器分类
+### 2. 类加载器分类
从 Java 虚拟机的角度来讲,只存在以下两种不同的类加载器:
-一种是启动类加载器(Bootstrap ClassLoader),这个类加载器用 C++ 实现,是虚拟机自身的一部分;另一种就是所有其他类的加载器,这些类由 Java 实现,独立于虚拟机外部,并且全都继承自抽象类 java.lang.ClassLoader。
+- 启动类加载器(Bootstrap ClassLoader),这个类加载器用 C++ 实现,是虚拟机自身的一部分;
+
+- 所有其他类的加载器,这些类由 Java 实现,独立于虚拟机外部,并且全都继承自抽象类 java.lang.ClassLoader。
从 Java 开发人员的角度看,类加载器可以划分得更细致一些:
-- 启动类加载器(Bootstrap ClassLoader) 此类加载器负责将存放在 \lib 目录中的,或者被 -Xbootclasspath 参数所指定的路径中的,并且是虚拟机识别的(仅按照文件名识别,如 rt.jar,名字不符合的类库即使放在 lib 目录中也不会被加载)类库加载到虚拟机内存中。 启动类加载器无法被 Java 程序直接引用,用户在编写自定义类加载器时,如果需要把加载请求委派给引导类加载器,直接使用 null 代替即可。
+- 启动类加载器(Bootstrap ClassLoader)此类加载器负责将存放在 <JAVA_HOME>\lib 目录中的,或者被 -Xbootclasspath 参数所指定的路径中的,并且是虚拟机识别的(仅按照文件名识别,如 rt.jar,名字不符合的类库即使放在 lib 目录中也不会被加载)类库加载到虚拟机内存中。 启动类加载器无法被 Java 程序直接引用,用户在编写自定义类加载器时,如果需要把加载请求委派给启动类加载器,直接使用 null 代替即可。
-- 扩展类加载器(Extension ClassLoader) 这个类加载器是由 ExtClassLoader(sun.misc.Launcher$ExtClassLoader)实现的。它负责将 /lib/ext 或者被 java.ext.dir 系统变量所指定路径中的所有类库加载到内存中,开发者可以直接使用扩展类加载器。
+- 扩展类加载器(Extension ClassLoader)这个类加载器是由 ExtClassLoader(sun.misc.Launcher$ExtClassLoader)实现的。它负责将 <JAVA_HOME>/lib/ext 或者被 java.ext.dir 系统变量所指定路径中的所有类库加载到内存中,开发者可以直接使用扩展类加载器。
-- 应用程序类加载器(Application ClassLoader) 这个类加载器是由 AppClassLoader(sun.misc.Launcher$AppClassLoader)实现的。由于这个类加载器是 ClassLoader 中的 getSystemClassLoader() 方法的返回值,因此一般称为系统类加载器。它负责加载用户类路径(ClassPath)上所指定的类库,开发者可以直接使用这个类加载器,如果应用程序中没有自定义过自己的类加载器,一般情况下这个就是程序中默认的类加载器。
+- 应用程序类加载器(Application ClassLoader)这个类加载器是由 AppClassLoader(sun.misc.Launcher$AppClassLoader)实现的。由于这个类加载器是 ClassLoader 中的 getSystemClassLoader() 方法的返回值,因此一般称为系统类加载器。它负责加载用户类路径(ClassPath)上所指定的类库,开发者可以直接使用这个类加载器,如果应用程序中没有自定义过自己的类加载器,一般情况下这个就是程序中默认的类加载器。
-### 4.3 双亲委派模型
+### 3. 双亲委派模型
应用程序都是由三种类加载器相互配合进行加载的,如果有必要,还可以加入自己定义的类加载器。下图展示的类加载器之间的层次关系,称为类加载器的双亲委派模型(Parents Delegation Model)。该模型要求除了顶层的启动类加载器外,其余的类加载器都应有自己的父类加载器,这里类加载器之间的父子关系一般通过组合(Composition)关系来实现,而不是通过继承(Inheritance)的关系实现。
-
+
-**工作过程**
+**(一)工作过程**
-如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载,而是把这个请求委派给父类加载器,每一个层次的加载器都是如此,依次递归,因此所有的加载请求最终都应该传送到顶层的启动类加载器中,只有当父加载器反馈自己无法完成此加载请求(它搜索范围中没有找到所需类)时,子加载器才会尝试自己加载。
+如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载,而是把这个请求委派给父类加载器,每一个层次的加载器都是如此,依次递归。因此所有的加载请求最终都应该传送到顶层的启动类加载器中,只有当父加载器反馈自己无法完成此加载请求(它搜索范围中没有找到所需类)时,子加载器才会尝试自己加载。
-**好处**
+**(二)好处**
-使用双亲委派模型来组织类加载器之间的关系,使得 Java 类随着它的类加载器一起具备了一种带有优先级的层次关系。例如类 java.lang.Object,它存放再 rt.jar 中,无论哪个类加载器要加载这个类,最终都是委派给处于模型最顶端的启动类加载器进行加载,因此 Object 类在程序的各种类加载器环境中都是同一个类。相反,如果没有双亲委派模型,由各个类加载器自行加载的话,如果用户编写了一个称为`java.lang.Object 的类,并放在程序的 ClassPath 中,那系统中将会出现多个不同的 Object 类,程序将变得一片混乱。如果开发者尝试编写一个与 rt.jar 类库中已有类重名的 Java 类,将会发现可以正常编译,但是永远无法被加载运行。
+使用双亲委派模型来组织类加载器之间的关系,使得 Java 类随着它的类加载器一起具备了一种带有优先级的层次关系。例如类 java.lang.Object,它存放在 rt.jar 中,无论哪个类加载器要加载这个类,最终都是委派给处于模型最顶端的启动类加载器进行加载,因此 Object 类在程序的各种类加载器环境中都是同一个类。相反,如果没有双亲委派模型,由各个类加载器自行加载的话,如果用户编写了一个称为java.lang.Object 的类,并放在程序的 ClassPath 中,那系统中将会出现多个不同的 Object 类,程序将变得一片混乱。如果开发者尝试编写一个与 rt.jar 类库中已有类重名的 Java 类,将会发现可以正常编译,但是永远无法被加载运行。
-**实现**
+**(三)实现**
```java
protected synchronized Class> loadClass(String name, boolean resolve) throws ClassNotFoundException{
- //check the class has been loaded or not
+ // 先检查请求的类是否已经被加载过了
Class c = findLoadedClass(name);
if(c == null) {
try{
@@ -634,9 +642,10 @@ protected synchronized Class> loadClass(String name, boolean resolve) throws C
c = findBootstrapClassOrNull(name);
}
} catch(ClassNotFoundException e) {
- //if throws the exception , the father can not complete the load
+ // 如果父类加载器抛出 ClassNotFoundException,说明父类加载器无法完成加载请求
}
if(c == null) {
+ // 如果父类加载器无法完成加载请求,再调用自身的 findClass() 来进行加载
c = findClass(name);
}
}
@@ -647,7 +656,7 @@ protected synchronized Class> loadClass(String name, boolean resolve) throws C
}
```
-# JVM 参数
+# 四、JVM 参数
## GC 优化配置
@@ -672,3 +681,14 @@ protected synchronized Class> loadClass(String name, boolean resolve) throws C
```java
java -Xmx12m -Xms3m -Xmn1m -XX:PermSize=20m -XX:MaxPermSize=20m -XX:+UseSerialGC -jar java-application.jar
```
+
+# 参考资料
+
+- 周志明. 深入理解 Java 虚拟机 [M]. 机械工业出版社, 2011.
+- [Jvm memory](https://www.slideshare.net/benewu/jvm-memory)
+- [Memory Architecture Of JVM(Runtime Data Areas)](https://hackthejava.wordpress.com/2015/01/09/memory-architecture-by-jvmruntime-data-areas/)
+- [JVM Run-Time Data Areas](https://www.programcreek.com/2013/04/jvm-run-time-data-areas/)
+- [Android on x86: Java Native Interface and the Android Native Development Kit](http://www.drdobbs.com/architecture-and-design/android-on-x86-java-native-interface-and/240166271)
+- [深入理解 JVM(2)——GC 算法与内存分配策略](https://crowhawk.github.io/2017/08/10/jvm_2/)
+- [深入理解 JVM(3)——7 种垃圾收集器](https://crowhawk.github.io/2017/08/15/jvm_3/)
+- [JVM Internals](http://blog.jamesdbloom.com/JVMInternals.html)
diff --git a/notes/Leetcode 题解.md b/notes/Leetcode 题解.md
index cf67cdca..f3f229d8 100644
--- a/notes/Leetcode 题解.md
+++ b/notes/Leetcode 题解.md
@@ -13,14 +13,14 @@
* [Backtracking](#backtracking)
* [分治](#分治)
* [动态规划](#动态规划)
- * [分割整数](#分割整数)
- * [矩阵路径](#矩阵路径)
* [斐波那契数列](#斐波那契数列)
* [最长递增子序列](#最长递增子序列)
- * [最长公共子系列](#最长公共子系列)
+ * [最长公共子序列](#最长公共子序列)
* [0-1 背包](#0-1-背包)
* [数组区间](#数组区间)
* [字符串编辑](#字符串编辑)
+ * [分割整数](#分割整数)
+ * [矩阵路径](#矩阵路径)
* [其它问题](#其它问题)
* [数学](#数学)
* [素数](#素数)
@@ -36,8 +36,6 @@
* [哈希表](#哈希表)
* [字符串](#字符串)
* [数组与矩阵](#数组与矩阵)
- * [1-n 分布](#1-n-分布)
- * [有序矩阵](#有序矩阵)
* [链表](#链表)
* [树](#树)
* [递归](#递归)
@@ -70,11 +68,11 @@ public int search(int key, int[] array) {
实现时需要注意以下细节:
-1. 在计算 mid 时不能使用 mid = (l + h) / 2 这种方式,因为 l + h 可能会导致加法溢出,应该使用 mid = l + (h - l) / 2。
+- 在计算 mid 时不能使用 mid = (l + h) / 2 这种方式,因为 l + h 可能会导致加法溢出,应该使用 mid = l + (h - l) / 2。
-2. 对 h 的赋值和循环条件有关,当循环条件为 l <= h 时,h = mid - 1;当循环条件为 l < h 时,h = mid。解释如下:在循环条件为 l <= h 时,如果 h = mid,会出现循环无法退出的情况,例如 l = 1,h = 1,此时 mid 也等于 1,如果此时继续执行 h = mid,那么就会无限循环;在循环条件为 l < h,如果 h = mid - 1,会错误跳过查找的数,例如对于数组 [1,2,3],要查找 1,最开始 l = 0,h = 2,mid = 1,判断 key < arr[mid] 执行 h = mid - 1 = 0,此时循环退出,直接把查找的数跳过了。
+- 对 h 的赋值和循环条件有关,当循环条件为 l <= h 时,h = mid - 1;当循环条件为 l < h 时,h = mid。解释如下:在循环条件为 l <= h 时,如果 h = mid,会出现循环无法退出的情况,例如 l = 1,h = 1,此时 mid 也等于 1,如果此时继续执行 h = mid,那么就会无限循环;在循环条件为 l < h,如果 h = mid - 1,会错误跳过查找的数,例如对于数组 [1,2,3],要查找 1,最开始 l = 0,h = 2,mid = 1,判断 key < arr[mid] 执行 h = mid - 1 = 0,此时循环退出,直接把查找的数跳过了。
-3. l 的赋值一般都为 l = mid + 1。
+- l 的赋值一般都为 l = mid + 1。
**求开方**
@@ -188,7 +186,7 @@ Input: [1,2], [1,2,3]
Output: 2
Explanation: You have 2 children and 3 cookies. The greed factors of 2 children are 1, 2.
-You have 3 cookies and their sizes are big enough to gratify all of the children,
+You have 3 cookies and their sizes are big enough to gratify all of the children,
You need to output 2.
```
@@ -253,7 +251,7 @@ public int findMinArrowShots(int[][] points) {
题目描述:一次交易包含买入和卖出,多个交易之间不能交叉进行。
-对于 [a, b, c, d],如果有 a <= b <= c <= d ,那么最大收益为 d - a。而 d - a = (d - c) + (c - b) + (b - a) ,因此当访问到一个 prices[i] 且 prices[i] - prices[i-1] > 0,那么就把 prices[i] - prices[i-1] 添加加到收益中,从而在局部最优的情况下也保证全局最优。
+对于 [a, b, c, d],如果有 a <= b <= c <= d ,那么最大收益为 d - a。而 d - a = (d - c) + (c - b) + (b - a) ,因此当访问到一个 prices[i] 且 prices[i] - prices[i-1] > 0,那么就把 prices[i] - prices[i-1] 添加到收益中,从而在局部最优的情况下也保证全局最优。
```java
public int maxProfit(int[] prices) {
@@ -331,9 +329,10 @@ Return true.
```java
public boolean isSubsequence(String s, String t) {
- for (int i = 0, pos = 0; i < s.length(); i++, pos++) {
- pos = t.indexOf(s.charAt(i), pos);
- if(pos == -1) return false;
+ int index = 0;
+ for (char c : s.toCharArray()) {
+ index = t.indexOf(c, index);
+ if (index == -1) return false;
}
return true;
}
@@ -355,18 +354,20 @@ A partition like "ababcbacadefegde", "hijhklij" is incorrect, because it splits
```java
public List partitionLabels(String S) {
List ret = new ArrayList<>();
- int[] lastIdxs = new int[26];
- for(int i = 0; i < S.length(); i++) lastIdxs[S.charAt(i) - 'a'] = i;
- int startIdx = 0;
- while(startIdx < S.length()) {
- int endIdx = startIdx;
- for(int i = startIdx; i < S.length() && i <= endIdx; i++) {
- int lastIdx = lastIdxs[S.charAt(i) - 'a'];
- if(lastIdx == i) continue;
- if(lastIdx > endIdx) endIdx = lastIdx;
+ int[] lastIndexs = new int[26];
+ for (int i = 0; i < S.length(); i++) {
+ lastIndexs[S.charAt(i) - 'a'] = i;
+ }
+ int firstIndex = 0;
+ while (firstIndex < S.length()) {
+ int lastIndex = firstIndex;
+ for (int i = firstIndex; i < S.length() && i <= lastIndex; i++) {
+ int index = lastIndexs[S.charAt(i) - 'a'];
+ if (index == i) continue;
+ if (index > lastIndex) lastIndex = index;
}
- ret.add(endIdx - startIdx + 1);
- startIdx = endIdx + 1;
+ ret.add(lastIndex - firstIndex + 1);
+ firstIndex = lastIndex + 1;
}
return ret;
}
@@ -392,23 +393,21 @@ Output:
```java
public int[][] reconstructQueue(int[][] people) {
- if(people == null || people.length == 0 || people[0].length == 0) return new int[0][0];
-
- Arrays.sort(people, new Comparator() {
- public int compare(int[] a, int[] b) {
- if(a[0] == b[0]) return a[1] - b[1];
- return b[0] - a[0];
- }
+ if (people == null || people.length == 0 || people[0].length == 0) return new int[0][0];
+ Arrays.sort(people, (a, b) -> {
+ if (a[0] == b[0]) return a[1] - b[1];
+ return b[0] - a[0];
});
-
- int n = people.length;
+ int N = people.length;
List tmp = new ArrayList<>();
- for(int i = 0; i < n; i++) {
- tmp.add(people[i][1], new int[]{people[i][0], people[i][1]});
+ for (int i = 0; i < N; i++) {
+ int index = people[i][1];
+ int[] p = new int[]{people[i][0], people[i][1]};
+ tmp.add(index, p);
}
- int[][] ret = new int[n][2];
- for(int i = 0; i < n; i++) {
+ int[][] ret = new int[N][2];
+ for (int i = 0; i < N; i++) {
ret[i][0] = tmp.get(i)[0];
ret[i][1] = tmp.get(i)[1];
}
@@ -420,7 +419,7 @@ public int[][] reconstructQueue(int[][] people) {
双指针主要用于遍历数组,两个指针指向不同的元素,从而协同完成任务。
-**从一个已经排序的数组中查找出两个数,使它们的和为 0**
+**有序数组的 Tow Sum**
[Leetcode :167. Two Sum II - Input array is sorted (Easy)](https://leetcode.com/problems/two-sum-ii-input-array-is-sorted/description/)
@@ -429,6 +428,8 @@ Input: numbers={2, 7, 11, 15}, target=9
Output: index1=1, index2=2
```
+题目描述:从一个已经排序的数组中找出两个数,使它们的和为 0。
+
使用双指针,一个指针指向元素较小的值,一个指针指向元素较大的值。指向较小元素的指针从头向尾遍历,指向较大元素的指针从尾向头遍历。
如果两个指针指向元素的和 sum == target,那么得到要求的结果;如果 sum > target,移动较大的元素,使 sum 变小一些;如果 sum < target,移动较小的元素,使 sum 变大一些。
@@ -457,22 +458,22 @@ Given s = "leetcode", return "leotcede".
使用双指针,指向待反转的两个元音字符,一个指针从头向尾遍历,一个指针从尾到头遍历。
```java
-private HashSet vowels = new HashSet<>(Arrays.asList('a','e','i','o','u','A','E','I','O','U'));
+private HashSet vowels = new HashSet<>(Arrays.asList('a', 'e', 'i', 'o', 'u', 'A', 'E', 'I', 'O', 'U'));
public String reverseVowels(String s) {
- if(s.length() == 0) return s;
+ if (s.length() == 0) return s;
int i = 0, j = s.length() - 1;
char[] result = new char[s.length()];
- while(i <= j){
+ while (i <= j) {
char ci = s.charAt(i);
char cj = s.charAt(j);
- if(!vowels.contains(ci)){
+ if (!vowels.contains(ci)) {
result[i] = ci;
i++;
- } else if(!vowels.contains(cj)){
+ } else if (!vowels.contains(cj)) {
result[j] = cj;
j--;
- } else{
+ } else {
result[i] = cj;
result[j] = ci;
i++;
@@ -497,12 +498,12 @@ Explanation: 1 * 1 + 2 * 2 = 5
```java
public boolean judgeSquareSum(int c) {
- int left = 0, right = (int) Math.sqrt(c);
- while(left <= right){
- int powSum = left * left + right * right;
- if(powSum == c) return true;
- else if(powSum > c) right--;
- else left++;
+ int i = 0, j = (int) Math.sqrt(c);
+ while (i <= j) {
+ int powSum = i * i + j * j;
+ if (powSum == c) return true;
+ if (powSum > c) j--;
+ else i++;
}
return false;
}
@@ -518,13 +519,13 @@ Output: True
Explanation: You could delete the character 'c'.
```
-题目描述:字符串可以删除一个字符,判断是否能构成回文字符串。
+题目描述:可以删除一个字符,判断是否能构成回文字符串。
```java
public boolean validPalindrome(String s) {
- int i = 0, j = s.length() -1;
- while(i < j){
- if(s.charAt(i) != s.charAt(j)){
+ int i = 0, j = s.length() - 1;
+ while (i < j) {
+ if (s.charAt(i) != s.charAt(j)) {
return isPalindrome(s, i, j - 1) || isPalindrome(s, i + 1, j);
}
i++;
@@ -533,12 +534,9 @@ public boolean validPalindrome(String s) {
return true;
}
-private boolean isPalindrome(String s, int l, int r){
- while(l < r){
- if(s.charAt(l) != s.charAt(r))
- return false;
- l++;
- r--;
+private boolean isPalindrome(String s, int l, int r) {
+ while (l < r) {
+ if (s.charAt(l++) != s.charAt(r--)) return false;
}
return true;
}
@@ -548,18 +546,18 @@ private boolean isPalindrome(String s, int l, int r){
[Leetcode : 88. Merge Sorted Array (Easy)](https://leetcode.com/problems/merge-sorted-array/description/)
-题目描述:把归并结果存到第一个数组上
+题目描述:把归并结果存到第一个数组上。
```java
public void merge(int[] nums1, int m, int[] nums2, int n) {
int i = m - 1, j = n - 1; // 需要从尾开始遍历,否则在 nums1 上归并得到的值会覆盖还未进行归并比较的值
- int idx = m + n - 1;
- while(i >= 0 || j >= 0){
- if(i < 0) nums1[idx] = nums2[j--];
- else if(j < 0) nums1[idx] = nums1[i--];
- else if(nums1[i] > nums2[j]) nums1[idx] = nums1[i--];
- else nums1[idx] = nums2[j--];
- idx--;
+ int index = m + n - 1;
+ while (i >= 0 || j >= 0) {
+ if (i < 0) nums1[index] = nums2[j--];
+ else if (j < 0) nums1[index] = nums1[i--];
+ else if (nums1[i] > nums2[j]) nums1[index] = nums1[i--];
+ else nums1[index] = nums2[j--];
+ index--;
}
}
```
@@ -572,12 +570,12 @@ public void merge(int[] nums1, int m, int[] nums2, int n) {
```java
public boolean hasCycle(ListNode head) {
- if(head == null) return false;
+ if (head == null) return false;
ListNode l1 = head, l2 = head.next;
- while(l1 != null && l2 != null){
- if(l1 == l2) return true;
+ while (l1 != null && l2 != null) {
+ if (l1 == l2) return true;
l1 = l1.next;
- if(l2.next == null) break;
+ if (l2.next == null) break;
l2 = l2.next.next;
}
return false;
@@ -596,7 +594,7 @@ Output:
"apple"
```
-题目描述:可以删除 s 中的一些字符,使得它成为字符串列表 d 中的一个字符串。要求在 d 中找到满足条件的最长字符串。
+题目描述:删除 s 中的一些字符,使得它构成字符串列表 d 中的一个字符串,找出能构成的最长字符串。如果有多个相同长度的结果,返回按字典序排序的最大字符串。
```java
public String findLongestWord(String s, List d) {
@@ -605,8 +603,7 @@ public String findLongestWord(String s, List d) {
for (int i = 0, j = 0; i < s.length() && j < str.length(); i++) {
if (s.charAt(i) == str.charAt(j)) j++;
if (j == str.length()) {
- if (ret.length() < str.length()
- || (ret.length() == str.length() && ret.compareTo(str) > 0)) {
+ if (ret.length() < str.length() || (ret.length() == str.length() && ret.compareTo(str) > 0)) {
ret = str;
}
}
@@ -620,9 +617,9 @@ public String findLongestWord(String s, List d) {
### 快速选择
-一般用于求解 **Kth Element** 问题,可以在 O(n) 时间复杂度,O(1) 空间复杂度完成求解工作。
+一般用于求解 **Kth Element** 问题,可以在 O(N) 时间复杂度,O(1) 空间复杂度完成求解工作。
-与快速排序一样,快速选择一般需要先打乱数组,否则最坏情况下时间复杂度为 O(n2 )。
+与快速排序一样,快速选择一般需要先打乱数组,否则最坏情况下时间复杂度为 O(N2 )。
### 堆排序
@@ -632,24 +629,23 @@ public String findLongestWord(String s, List d) {
[Leetocde : 215. Kth Largest Element in an Array (Medium)](https://leetcode.com/problems/kth-largest-element-in-an-array/description/)
-**排序** :时间复杂度 O(nlgn),空间复杂度 O(1) 解法
+**排序** :时间复杂度 O(NlogN),空间复杂度 O(1)
```java
public int findKthLargest(int[] nums, int k) {
- int N = nums.length;
- Arrays.sort(nums);
- return nums[N - k];
+ Arrays.sort(nums);
+ return nums[nums.length - k];
}
```
-**堆排序** :时间复杂度 O(nlgk),空间复杂度 O(k)
+**堆排序** :时间复杂度 O(OlogK),空间复杂度 O(K)。
```java
public int findKthLargest(int[] nums, int k) {
PriorityQueue pq = new PriorityQueue<>();
- for(int val : nums) {
- pq.offer(val);
- if(pq.size() > k) {
+ for (int val : nums) {
+ pq.add(val);
+ if (pq.size() > k) {
pq.poll();
}
}
@@ -657,56 +653,47 @@ public int findKthLargest(int[] nums, int k) {
}
```
-**快速选择** :时间复杂度 O(n),空间复杂度 O(1)
+**快速选择** :时间复杂度 O(N),空间复杂度 O(1)
```java
public int findKthLargest(int[] nums, int k) {
- k = nums.length - k;
- int lo = 0;
- int hi = nums.length - 1;
- while (lo < hi) {
- final int j = partition(nums, lo, hi);
- if(j < k) {
- lo = j + 1;
- } else if (j > k) {
- hi = j - 1;
- } else {
- break;
- }
- }
- return nums[k];
+ k = nums.length - k;
+ int l = 0, h = nums.length - 1;
+ while (l < h) {
+ int j = partition(nums, l, h);
+ if (j == k) break;
+ if (j < k) l = j + 1;
+ else h = j - 1;
}
+ return nums[k];
+}
- private int partition(int[] a, int lo, int hi) {
- int i = lo;
- int j = hi + 1;
- while(true) {
- while(i < hi && less(a[++i], a[lo]));
- while(j > lo && less(a[lo], a[--j]));
- if(i >= j) {
- break;
- }
- exch(a, i, j);
- }
- exch(a, lo, j);
- return j;
+private int partition(int[] a, int l, int h) {
+ int i = l, j = h + 1;
+ while (true) {
+ while (i < h && less(a[++i], a[l])) ;
+ while (j > l && less(a[l], a[--j])) ;
+ if (i >= j) break;
+ swap(a, i, j);
}
+ swap(a, l, j);
+ return j;
+}
- private void exch(int[] a, int i, int j) {
- final int tmp = a[i];
- a[i] = a[j];
- a[j] = tmp;
- }
+private void swap(int[] a, int i, int j) {
+ int tmp = a[i];
+ a[i] = a[j];
+ a[j] = tmp;
+}
- private boolean less(int v, int w) {
- return v < w;
- }
+private boolean less(int v, int w) {
+ return v < w;
}
```
### 桶排序
-**找出出现频率最多的 k 个数**
+**出现频率最多的 k 个数**
[Leetcode : 347. Top K Frequent Elements (Medium)](https://leetcode.com/problems/top-k-frequent-elements/description/)
@@ -714,24 +701,26 @@ public int findKthLargest(int[] nums, int k) {
Given [1,1,1,2,2,3] and k = 2, return [1,2].
```
+设置若干个桶,每个桶存储出现频率相同的数,并且桶的下标代表桶中数出现的频率,即第 i 个桶中存储的数出现的频率为 i。把数都放到桶之后,从后向前遍历桶,最先得到的 k 个数就是出现频率最多的的 k 个数。
+
```java
public List topKFrequent(int[] nums, int k) {
List ret = new ArrayList<>();
- Map map = new HashMap<>();
- for(int num : nums) {
- map.put(num, map.getOrDefault(num, 0) + 1);
+ Map frequencyMap = new HashMap<>();
+ for (int num : nums) {
+ frequencyMap.put(num, frequencyMap.getOrDefault(num, 0) + 1);
}
List[] bucket = new List[nums.length + 1];
- for(int key : map.keySet()) {
- int frequency = map.get(key);
- if(bucket[frequency] == null) {
+ for (int key : frequencyMap.keySet()) {
+ int frequency = frequencyMap.get(key);
+ if (bucket[frequency] == null) {
bucket[frequency] = new ArrayList<>();
}
bucket[frequency].add(key);
}
- for(int i = bucket.length - 1; i >= 0 && ret.size() < k; i--) {
- if(bucket[i] != null) {
+ for (int i = bucket.length - 1; i >= 0 && ret.size() < k; i--) {
+ if (bucket[i] != null) {
ret.addAll(bucket[i]);
}
}
@@ -739,6 +728,51 @@ public List topKFrequent(int[] nums, int k) {
}
```
+**按照字符出现次数对字符串排序**
+
+[Leetcode : 451. Sort Characters By Frequency (Medium)](https://leetcode.com/problems/sort-characters-by-frequency/description/)
+
+```html
+Input:
+"tree"
+
+Output:
+"eert"
+
+Explanation:
+'e' appears twice while 'r' and 't' both appear once.
+So 'e' must appear before both 'r' and 't'. Therefore "eetr" is also a valid answer.
+```
+
+```java
+public String frequencySort(String s) {
+ Map map = new HashMap<>();
+ for (char c : s.toCharArray()) {
+ map.put(c, map.getOrDefault(c, 0) + 1);
+ }
+ List[] frequencyBucket = new List[s.length() + 1];
+ for(char c : map.keySet()){
+ int f = map.get(c);
+ if (frequencyBucket[f] == null) {
+ frequencyBucket[f] = new ArrayList<>();
+ }
+ frequencyBucket[f].add(c);
+ }
+ StringBuilder str = new StringBuilder();
+ for (int i = frequencyBucket.length - 1; i >= 0; i--) {
+ if (frequencyBucket[i] == null) {
+ continue;
+ }
+ for (char c : frequencyBucket[i]) {
+ for (int j = 0; j < i; j++) {
+ str.append(c);
+ }
+ }
+ }
+ return str.toString();
+}
+```
+
## 搜索
深度优先搜索和广度优先搜索广泛运用于树和图中,但是它们的应用远远不止如此。
@@ -747,28 +781,42 @@ public List topKFrequent(int[] nums, int k) {
-广度优先搜索的搜索过程有点像一层一层地进行遍历:从节点 0 出发,遍历到 6、2、1 和 5 这四个新节点。
+广度优先搜索的搜索过程有点像一层一层地进行遍历,每层遍历都以上一层遍历的结果作为起点,遍历一个长度。需要注意的是,遍历过的节点不能再次被遍历。
-继续从 6 开始遍历,得到节点 4 ;从 2 开始遍历,没有下一个节点;从 1 开始遍历,没有下一个节点;从 5 开始遍历,得到 3 和 4 节点。这一轮总共得到两个新节点:4 和 3 。
+第一层:
-反复从新节点出发进行上述的遍历操作。
+- 0 -> {6,2,1,5};
-可以看到,每一轮遍历的节点都与根节点路径长度相同。设 di 表示第 i 个节点与根节点的路径长度,推导出一个结论:对于先遍历的节点 i 与后遍历的节点 j,有 di <=dj 。利用这个结论,可以求解最短路径 **最优解** 问题:第一次遍历到目的节点,其所经过的路径为最短路径,如果继续遍历,之后再遍历到目的节点,所经过的路径就不是最短路径。
+第二层:
+
+- 6 -> {4}
+- 2 -> {}
+- 1 -> {}
+- 5 -> {3}
+
+第三层:
+
+- 4 -> {}
+- 3 -> {}
+
+可以看到,每一轮遍历的节点都与根节点路径长度相同。设 di 表示第 i 个节点与根节点的路径长度,推导出一个结论:对于先遍历的节点 i 与后遍历的节点 j,有 di <=dj 。利用这个结论,可以求解最短路径等 **最优解** 问题:第一次遍历到目的节点,其所经过的路径为最短路径,如果继续遍历,之后再遍历到目的节点,所经过的路径就不是最短路径。
在程序实现 BFS 时需要考虑以下问题:
-- 队列:用来存储每一轮遍历的节点
-- 标记:对于遍历过得节点,应该将它标记,防止重复遍历;
+- 队列:用来存储每一轮遍历的节点;
+- 标记:对于遍历过的节点,应该将它标记,防止重复遍历。
**计算在网格中从原点到特定点的最短路径长度**
```html
[[1,1,0,1],
-[1,0,1,0],
-[1,1,1,1],
-[1,0,1,1]]
+ [1,0,1,0],
+ [1,1,1,1],
+ [1,0,1,1]]
```
+1 表示可以经过某个位置。
+
```java
public int minPathLength(int[][] grids, int tr, int tc) {
int[][] next = {{1, 0}, {-1, 0}, {0, 1}, {0, -1}};
@@ -803,14 +851,14 @@ private class Position {
-广度优先搜索一层一层遍历,每一层遍历到的所有新节点,要用队列先存储起来以备下一层遍历的时候再遍历;而深度优先搜索在遍历到一个新节点时立马对新节点进行遍历:从节点 0 出发开始遍历,得到到新节点 6 时,立马对新节点 6 进行遍历,得到新节点 4;如此反复以这种方式遍历新节点,直到没有新节点了,此时返回。返回到根节点 0 的情况是,继续对根节点 0 进行遍历,得到新节点 2,然后继续以上步骤。
+广度优先搜索一层一层遍历,每一层得到的所有新节点,要用队列先存储起来以备下一层遍历的时候再遍历;而深度优先搜索在得到到一个新节点时立马对新节点进行遍历:从节点 0 出发开始遍历,得到到新节点 6 时,立马对新节点 6 进行遍历,得到新节点 4;如此反复以这种方式遍历新节点,直到没有新节点了,此时返回。返回到根节点 0 的情况是,继续对根节点 0 进行遍历,得到新节点 2,然后继续以上步骤。
从一个节点出发,使用 DFS 对一个图进行遍历时,能够遍历到的节点都是从初始节点可达的,DFS 常用来求解这种 **可达性** 问题。
在程序实现 DFS 时需要考虑以下问题:
-- 栈:用栈来保存当前节点信息,当遍历新节点返回时能够继续遍历当前节点。也可以使用递归栈。
-- 标记:和 BFS 一样同样需要对已经遍历过得节点进行标记。
+- 栈:用栈来保存当前节点信息,当遍历新节点返回时能够继续遍历当前节点。可以使用递归栈。
+- 标记:和 BFS 一样同样需要对已经遍历过的节点进行标记。
**查找最大的连通面积**
@@ -818,31 +866,32 @@ private class Position {
```html
[[0,0,1,0,0,0,0,1,0,0,0,0,0],
-[0,0,0,0,0,0,0,1,1,1,0,0,0],
-[0,1,1,0,1,0,0,0,0,0,0,0,0],
-[0,1,0,0,1,1,0,0,1,0,1,0,0],
-[0,1,0,0,1,1,0,0,1,1,1,0,0],
-[0,0,0,0,0,0,0,0,0,0,1,0,0],
-[0,0,0,0,0,0,0,1,1,1,0,0,0],
-[0,0,0,0,0,0,0,1,1,0,0,0,0]]
+ [0,0,0,0,0,0,0,1,1,1,0,0,0],
+ [0,1,1,0,1,0,0,0,0,0,0,0,0],
+ [0,1,0,0,1,1,0,0,1,0,1,0,0],
+ [0,1,0,0,1,1,0,0,1,1,1,0,0],
+ [0,0,0,0,0,0,0,0,0,0,1,0,0],
+ [0,0,0,0,0,0,0,1,1,1,0,0,0],
+ [0,0,0,0,0,0,0,1,1,0,0,0,0]]
```
```java
public int maxAreaOfIsland(int[][] grid) {
- int m = grid.length, n = grid[0].length;
int max = 0;
- for(int i = 0; i < m; i++){
- for(int j = 0; j < n; j++){
- if(grid[i][j] == 1) max = Math.max(max, dfs(grid, i, j));
+ for (int i = 0; i < grid.length; i++) {
+ for (int j = 0; j < grid[i].length; j++) {
+ if (grid[i][j] == 1) {
+ max = Math.max(max, dfs(grid, i, j));
+ }
}
}
return max;
}
-private int dfs(int[][] grid, int i, int j){
- int m = grid.length, n = grid[0].length;
- if(i < 0 || i >= m || j < 0 || j >= n) return 0;
- if(grid[i][j] == 0) return 0;
+private int dfs(int[][] grid, int i, int j) {
+ if (i < 0 || i >= grid.length || j < 0 || j >= grid[i].length || grid[i][j] == 0) {
+ return 0;
+ }
grid[i][j] = 0;
return dfs(grid, i + 1, j) + dfs(grid, i - 1, j) + dfs(grid, i, j + 1) + dfs(grid, i, j - 1) + 1;
}
@@ -866,22 +915,22 @@ The 2nd student himself is in a friend circle. So return 2.
public int findCircleNum(int[][] M) {
int n = M.length;
int ret = 0;
- boolean[] hasFind = new boolean[n];
- for(int i = 0; i < n; i++) {
- if(!hasFind[i]) {
- dfs(M, i, hasFind);
+ boolean[] hasVisited = new boolean[n];
+ for (int i = 0; i < n; i++) {
+ if (!hasVisited[i]) {
+ dfs(M, i, hasVisited);
ret++;
}
+
}
return ret;
}
-private void dfs(int[][] M, int i, boolean[] hasFind) {
- hasFind[i] = true;
- int n = M.length;
- for(int k = 0; k < n; k++) {
- if(M[i][k] == 1 && !hasFind[k]) {
- dfs(M, k, hasFind);
+private void dfs(int[][] M, int i, boolean[] hasVisited) {
+ hasVisited[i] = true;
+ for (int k = 0; k < M.length; k++) {
+ if (M[i][k] == 1 && !hasVisited[k]) {
+ dfs(M, k, hasVisited);
}
}
}
@@ -933,11 +982,11 @@ private void dfs(char[][] grid, int i, int j) {
[Leetcode : 257. Binary Tree Paths (Easy)](https://leetcode.com/problems/binary-tree-paths/description/)
```html
- 1
-/ \
+ 1
+ / \
2 3
\
- 5
+ 5
```
```html
["1->2->5", "1->3"]
@@ -963,6 +1012,41 @@ private void dfs(TreeNode root, String prefix, List ret){
}
```
+**IP 地址划分**
+
+[Leetcode : 93. Restore IP Addresses(Medium)](https://leetcode.com/problems/restore-ip-addresses/description/)
+
+```html
+Given "25525511135",
+return ["255.255.11.135", "255.255.111.35"].
+```
+
+```java
+private List ret;
+
+public List restoreIpAddresses(String s) {
+ ret = new ArrayList<>();
+ doRestore(0, "", s);
+ return ret;
+}
+
+private void doRestore(int k, String path, String s) {
+ if (k == 4 || s.length() == 0) {
+ if (k == 4 && s.length() == 0) {
+ ret.add(path);
+ }
+ return;
+ }
+ for (int i = 0; i < s.length() && i <= 2; i++) {
+ if (i != 0 && s.charAt(0) == '0') break;
+ String part = s.substring(0, i + 1);
+ if (Integer.valueOf(part) <= 255) {
+ doRestore(k + 1, path.length() != 0 ? path + "." + part : part, s.substring(i + 1));
+ }
+ }
+}
+```
+
**填充封闭区域**
[Leetcode : 130. Surrounded Regions (Medium)](https://leetcode.com/problems/surrounded-regions/description/)
@@ -1083,74 +1167,11 @@ private void dfs(int r, int c, boolean[][] canReach) {
}
```
-**N 皇后**
-
-[Leetcode : 51. N-Queens (Hard)](https://leetcode.com/problems/n-queens/description/)
-
-
-
-题目描述:在 n\*n 的矩阵中摆放 n 个皇后,并且每个皇后不能在同一行,同一列,同一对角线上,要求解所有的 n 皇后解。
-
-一行一行地摆放,在确定一行中的那个皇后应该摆在哪一列时,需要用三个标记数组来确定某一列是否合法,这三个标记数组分别为:列标记数组、45 度对角线标记数组和 135 度对角线标记数组。
-
-45 度对角线标记数组的维度为 2\*n - 1,通过下图可以明确 (r,c) 的位置所在的数组下标为 r + c。
-
-
-
-135 度对角线标记数组的维度也是 2\*n - 1,(r,c) 的位置所在的数组下标为 n - 1 - (r - c)。
-
-
-
-```java
-private List> ret;
-private char[][] nQueens;
-private boolean[] colUsed;
-private boolean[] diagonals45Used;
-private boolean[] diagonals135Used;
-private int n;
-
-public List> solveNQueens(int n) {
- ret = new ArrayList<>();
- nQueens = new char[n][n];
- Arrays.fill(nQueens, '.');
- colUsed = new boolean[n];
- diagonals45Used = new boolean[2 * n - 1];
- diagonals135Used = new boolean[2 * n - 1];
- this.n = n;
- backstracking(0);
- return ret;
-}
-
-private void backstracking(int row) {
- if (row == n) {
- List list = new ArrayList<>();
- for (char[] chars : nQueens) {
- list.add(new String(chars));
- }
- ret.add(list);
- return;
- }
-
- for (int col = 0; col < n; col++) {
- int diagonals45Idx = row + col;
- int diagonals135Idx = n - 1 - (row - col);
- if (colUsed[col] || diagonals45Used[diagonals45Idx] || diagonals135Used[diagonals135Idx]) {
- continue;
- }
- nQueens[row][col] = 'Q';
- colUsed[col] = diagonals45Used[diagonals45Idx] = diagonals135Used[diagonals135Idx] = true;
- backstracking(row + 1);
- colUsed[col] = diagonals45Used[diagonals45Idx] = diagonals135Used[diagonals135Idx] = false;
- nQueens[row][col] = '.';
- }
-}
-```
-
### Backtracking
-回溯是 DFS 的一种,它不是用在遍历图的节点上,而是用于求解 **排列组合** 问题,例如有 { 'a','b','c' } 三个字符,求解所有由这三个字符排列得到的字符串。
+回溯属于 DF,它不是用在遍历图的节点上,而是用于求解 **排列组合** 问题,例如有 { 'a','b','c' } 三个字符,求解所有由这三个字符排列得到的字符串。
-在程序实现时,回溯需要注意对元素进行标记的问题。使用递归实现的回溯,在访问一个新元素进入新的递归调用,此时需要将新元素标记为已经访问,这样才能在继续递归调用时不用重复访问该元素;但是在递归返回时,需要将该元素标记为未访问,因为只需要保证在一个递归链中不同时访问一个元素,而在不同的递归链是可以访问已经访问过但是不在当前递归链中的元素。
+在程序实现时,回溯需要注意对元素进行标记的问题。使用递归实现的回溯,在访问一个新元素进入新的递归调用时,需要将新元素标记为已经访问,这样才能在继续递归调用时不用重复访问该元素;但是在递归返回时,需要将该元素标记为未访问,因为只需要保证在一个递归链中不同时访问一个元素,可以访问已经访问过但是不在当前递归链中的元素。
**数字键盘组合**
@@ -1168,20 +1189,21 @@ private static final String[] KEYS = {"", "", "abc", "def", "ghi", "jkl", "mno",
public List letterCombinations(String digits) {
List ret = new ArrayList<>();
- if (digits != null && digits.length() != 0) {
- combination("", digits, 0, ret);
- }
+ if (digits == null || digits.length() == 0) return ret;
+ combination(new StringBuilder(), digits, ret);
return ret;
}
-private void combination(String prefix, String digits, int offset, List ret) {
- if (offset == digits.length()) {
- ret.add(prefix);
+private void combination(StringBuilder prefix, String digits, List ret) {
+ if (prefix.length() == digits.length()) {
+ ret.add(prefix.toString());
return;
}
- String letters = KEYS[digits.charAt(offset) - '0'];
+ String letters = KEYS[digits.charAt(prefix.length()) - '0'];
for (char c : letters.toCharArray()) {
- combination(prefix + c, digits, offset + 1, ret);
+ prefix.append(c);
+ combination(prefix, digits, ret);
+ prefix.deleteCharAt(prefix.length() - 1); // 删除
}
}
```
@@ -1227,55 +1249,22 @@ private boolean dfs(char[][] board, String word, int start, int r, int c) {
if (start == word.length()) {
return true;
}
- if (r < 0 || r >= m || c < 0 || c >= n || board[r][c] != word.charAt(start) || visited[r][c] ) {
+ if (r < 0 || r >= m || c < 0 || c >= n || board[r][c] != word.charAt(start) || visited[r][c]) {
return false;
}
visited[r][c] = true;
for (int i = 0; i < shift.length; i++) {
int nextR = r + shift[i][0];
int nextC = c + shift[i][1];
- if (dfs(board, word, start + 1, nextR, nextC)) return true;
+ if (dfs(board, word, start + 1, nextR, nextC)) {
+ return true;
+ }
}
visited[r][c] = false;
return false;
}
```
-**IP 地址划分**
-
-[Leetcode : 93. Restore IP Addresses(Medium)](https://leetcode.com/problems/restore-ip-addresses/description/)
-
-```html
-Given "25525511135",
-return ["255.255.11.135", "255.255.111.35"].
-```
-
-```java
-private List ret;
-
-public List restoreIpAddresses(String s) {
- ret = new ArrayList<>();
- doRestore(0, "", s);
- return ret;
-}
-
-private void doRestore(int k, String path, String s) {
- if (k == 4 || s.length() == 0) {
- if (k == 4 && s.length() == 0) {
- ret.add(path);
- }
- return;
- }
- for (int i = 0; i < s.length() && i <= 2; i++) {
- if (i != 0 && s.charAt(0) == '0') break;
- String part = s.substring(0, i + 1);
- if (Integer.valueOf(part) <= 255) {
- doRestore(k + 1, path.length() != 0 ? path + "." + part : part, s.substring(i + 1));
- }
- }
-}
-```
-
**排列**
[Leetcode : 46. Permutations (Medium)](https://leetcode.com/problems/permutations/description/)
@@ -1327,7 +1316,7 @@ private void backtracking(List permuteList, boolean[] visited, int[] nu
[[1,1,2], [1,2,1], [2,1,1]]
```
-题目描述:数组元素可能含有相同的元素,进行排列时就有可能出先重复的排列,要求重复的排列只返回一个。
+题目描述:数组元素可能含有相同的元素,进行排列时就有可能出现 重复的排列,要求重复的排列只返回一个。
在实现上,和 Permutations 不同的是要先排序,然后在添加一个元素时,判断这个元素是否等于前一个元素,如果等于,并且前一个元素还未访问,那么就跳过这个元素。
@@ -1343,12 +1332,12 @@ public List> permuteUnique(int[] nums) {
private void backtracking(List permuteList, boolean[] visited, int[] nums, List> ret) {
if (permuteList.size() == nums.length) {
- ret.add(new ArrayList(permuteList));
+ ret.add(new ArrayList(permuteList)); // 重新构造一个 List
return;
}
for (int i = 0; i < visited.length; i++) {
- if (i != 0 && nums[i] == nums[i - 1] && !visited[i - 1]) continue;
+ if (i != 0 && nums[i] == nums[i - 1] && !visited[i - 1]) continue; // 防止重复
if (visited[i]) continue;
visited[i] = true;
permuteList.add(nums[i]);
@@ -1385,12 +1374,11 @@ public List> combine(int n, int k) {
private void backtracking(int start, int n, int k, List combineList, List> ret){
if(k == 0){
- ret.add(new ArrayList(combineList)); // 这里要重新构造一个 List
+ ret.add(new ArrayList(combineList));
return;
}
- for(int i = start; i <= n - k + 1; i++){ // 剪枝
-
+ for(int i = start; i <= n - k + 1; i++) { // 剪枝
combineList.add(i); // 把 i 标记为已访问
backtracking(i + 1, n, k - 1, combineList, ret);
combineList.remove(combineList.size() - 1); // 把 i 标记为未访问
@@ -1475,6 +1463,46 @@ private void doCombination(int[] candidates, int target, int start, List> combinationSum3(int k, int n) {
+ List> ret = new ArrayList<>();
+ List path = new ArrayList<>();
+ for (int i = 1; i <= 9; i++) {
+ path.add(i);
+ backtracking(k - 1, n - i, path, i, ret);
+ path.remove(0);
+ }
+ return ret;
+}
+
+private void backtracking(int k, int n, List path, int start, List> ret) {
+ if (k == 0 && n == 0) {
+ ret.add(new ArrayList<>(path));
+ return;
+ }
+ if (k == 0 || n == 0) return;
+ for (int i = start + 1; i <= 9; i++) { // 只能访问下一个元素,防止遍历的结果重复
+ path.add(i);
+ backtracking(k - 1, n - i, path, i, ret);
+ path.remove(path.size() - 1);
+ }
+}
+```
+
**子集**
[Leetcode : 78. Subsets (Medium)](https://leetcode.com/problems/subsets/description/)
@@ -1488,7 +1516,7 @@ private List subsetList;
public List> subsets(int[] nums) {
ret = new ArrayList<>();
subsetList = new ArrayList<>();
- for (int i = 0; i <= nums.length; i++) {
+ for (int i = 0; i <= nums.length; i++) { // 不同的子集大小
backtracking(0, i, nums);
}
return ret;
@@ -1502,7 +1530,7 @@ private void backtracking(int startIdx, int size, int[] nums) {
for (int i = startIdx; i < nums.length; i++) {
subsetList.add(nums[i]);
- backtracking(i + 1, size, nums); // startIdx 设为下一个元素,使 subset 中的元素都递增排序
+ backtracking(i + 1, size, nums);
subsetList.remove(subsetList.size() - 1);
}
}
@@ -1559,10 +1587,20 @@ private void backtracking(int startIdx, int size, int[] nums) {
}
```
-**分割字符串使得每部分都是回文数**
+**分割字符串使得每个部分都是回文数**
[Leetcode : 131. Palindrome Partitioning (Medium)](https://leetcode.com/problems/palindrome-partitioning/description/)
+```html
+For example, given s = "aab",
+Return
+
+[
+ ["aa","b"],
+ ["a","a","b"]
+]
+```
+
```java
private List> ret;
@@ -1650,6 +1688,69 @@ private int cubeNum(int i, int j) {
}
```
+**N 皇后**
+
+[Leetcode : 51. N-Queens (Hard)](https://leetcode.com/problems/n-queens/description/)
+
+
+
+题目描述:在 n\*n 的矩阵中摆放 n 个皇后,并且每个皇后不能在同一行,同一列,同一对角线上,要求解所有的 n 皇后解。
+
+一行一行地摆放,在确定一行中的那个皇后应该摆在哪一列时,需要用三个标记数组来确定某一列是否合法,这三个标记数组分别为:列标记数组、45 度对角线标记数组和 135 度对角线标记数组。
+
+45 度对角线标记数组的维度为 2\*n - 1,通过下图可以明确 (r,c) 的位置所在的数组下标为 r + c。
+
+
+
+135 度对角线标记数组的维度也是 2\*n - 1,(r,c) 的位置所在的数组下标为 n - 1 - (r - c)。
+
+
+
+```java
+private List> ret;
+private char[][] nQueens;
+private boolean[] colUsed;
+private boolean[] diagonals45Used;
+private boolean[] diagonals135Used;
+private int n;
+
+public List> solveNQueens(int n) {
+ ret = new ArrayList<>();
+ nQueens = new char[n][n];
+ Arrays.fill(nQueens, '.');
+ colUsed = new boolean[n];
+ diagonals45Used = new boolean[2 * n - 1];
+ diagonals135Used = new boolean[2 * n - 1];
+ this.n = n;
+ backstracking(0);
+ return ret;
+}
+
+private void backstracking(int row) {
+ if (row == n) {
+ List list = new ArrayList<>();
+ for (char[] chars : nQueens) {
+ list.add(new String(chars));
+ }
+ ret.add(list);
+ return;
+ }
+
+ for (int col = 0; col < n; col++) {
+ int diagonals45Idx = row + col;
+ int diagonals135Idx = n - 1 - (row - col);
+ if (colUsed[col] || diagonals45Used[diagonals45Idx] || diagonals135Used[diagonals135Idx]) {
+ continue;
+ }
+ nQueens[row][col] = 'Q';
+ colUsed[col] = diagonals45Used[diagonals45Idx] = diagonals135Used[diagonals135Idx] = true;
+ backstracking(row + 1);
+ colUsed[col] = diagonals45Used[diagonals45Idx] = diagonals135Used[diagonals135Idx] = false;
+ nQueens[row][col] = '.';
+ }
+}
+```
+
## 分治
**给表达式加括号**
@@ -1692,123 +1793,7 @@ public List diffWaysToCompute(String input) {
## 动态规划
-递归和动态规划都是将原问题拆成多个子问题然后求解,他们之间最本质的区别是,动态规划保存了子问题的解。
-
-### 分割整数
-
-**分割整数的最大乘积**
-
-[Leetcode : 343. Integer Break (Medim)](https://leetcode.com/problems/integer-break/description/)
-
-题目描述:For example, given n = 2, return 1 (2 = 1 + 1); given n = 10, return 36 (10 = 3 + 3 + 4).
-
-```java
-public int integerBreak(int n) {
- int[] dp = new int[n + 1];
- dp[1] = 1;
- for(int i = 2; i <= n; i++) {
- for(int j = 1; j <= i - 1; j++) {
- dp[i] = Math.max(dp[i], Math.max(j * dp[i - j], j * (i - j)));
- }
- }
- return dp[n];
-}
-```
-
-**按平方数来分割整数**
-
-[Leetcode : 279. Perfect Squares(Medium)](https://leetcode.com/problems/perfect-squares/description/)
-
-题目描述:For example, given n = 12, return 3 because 12 = 4 + 4 + 4; given n = 13, return 2 because 13 = 4 + 9.
-
-```java
-public int numSquares(int n) {
- List squares = new ArrayList<>(); // 存储小于 n 的平方数
- int diff = 3;
- while(square <= n) {
- squares.add(square);
- square += diff;
- diff += 2;
- }
- int[] dp = new int[n + 1];
- for(int i = 1; i <= n; i++) {
- int max = Integer.MAX_VALUE;
- for(int s : squares) {
- if(s > i) break;
- max = Math.min(max, dp[i - s] + 1);
- }
- dp[i] = max;
- }
- return dp[n];
-}
-```
-
-**分割整数构成字母字符串**
-
-[Leetcode : 91. Decode Ways (Medium)](https://leetcode.com/problems/decode-ways/description/)
-
-题目描述:Given encoded message "12", it could be decoded as "AB" (1 2) or "L" (12).
-
-```java
-public int numDecodings(String s) {
- if(s == null || s.length() == 0) return 0;
- int n = s.length();
- int[] dp = new int[n + 1];
- dp[0] = 1;
- dp[1] = s.charAt(0) == '0' ? 0 : 1;
- for(int i = 2; i <= n; i++) {
- int one = Integer.valueOf(s.substring(i - 1, i));
- if(one != 0) dp[i] += dp[i - 1];
- if(s.charAt(i - 2) == '0') continue;
- int two = Integer.valueOf(s.substring(i - 2, i));
- if(two <= 26) dp[i] += dp[i - 2];
- }
- return dp[n];
-}
-```
-
-### 矩阵路径
-
-**矩阵的总路径数**
-
-[Leetcode : 62. Unique Paths (Medium)](https://leetcode.com/problems/unique-paths/description/)
-
-题目描述:统计从矩阵左上角到右下角的路径总数,每次只能向左和向下移动。
-
-```java
-public int uniquePaths(int m, int n) {
- int[] dp = new int[n];
- for (int i = 0; i < m; i++) {
- for (int j = 0; j < n; j++) {
- if(i == 0) dp[j] = 1;
- else if(j != 0) dp[j] = dp[j] + dp[j - 1];
- }
- }
- return dp[n - 1];
-}
-```
-
-**矩阵的最小路径和**
-
-[Leetcode : 64. Minimum Path Sum (Medium)](https://leetcode.com/problems/minimum-path-sum/description/)
-
-题目描述:求从矩阵的左上角到右下角的最小路径和,每次只能向左和向下移动。
-
-```java
-public int minPathSum(int[][] grid) {
- if(grid.length == 0 || grid[0].length == 0) return 0;
- int m = grid.length, n = grid[0].length;
- int[] dp = new int[n];
- for(int i = 0; i < m; i++) {
- for(int j = 0; j < n; j++) {
- if(j == 0) dp[0] = dp[0] + grid[i][0];
- else if(i == 0) dp[j] = dp[j - 1] + grid[0][j];
- else dp[j] = Math.min(dp[j - 1], dp[j]) + grid[i][j];
- }
- }
- return dp[n - 1];
-}
-```
+递归和动态规划都是将原问题拆成多个子问题然后求解,他们之间最本质的区别是,动态规划保存了子问题的解,避免重复计算。
### 斐波那契数列
@@ -1824,13 +1809,12 @@ public int minPathSum(int[][] grid) {
dp[N] 即为所求。
-考虑到 dp[i] 只与 dp[i - 1] 和 dp[i - 2] 有关,因此可以只用两个变量来存储 dp[i - 1] 和 dp[i - 2] 即可,使得原来的 O(n) 空间复杂度优化为 O(1) 复杂度。
+考虑到 dp[i] 只与 dp[i - 1] 和 dp[i - 2] 有关,因此可以只用两个变量来存储 dp[i - 1] 和 dp[i - 2],使得原来的 O(N) 空间复杂度优化为 O(1) 复杂度。
```java
public int climbStairs(int n) {
if(n == 1) return 1;
if(n == 2) return 2;
- // 前一个楼梯、后一个楼梯
int pre1 = 2, pre2 = 1;
for(int i = 2; i < n; i++){
int cur = pre1 + pre2;
@@ -1904,37 +1888,37 @@ public int rob(int[] nums) {
[Leetcode : 213. House Robber II (Medium)](https://leetcode.com/problems/house-robber-ii/description/)
```java
+private int[] dp;
+
public int rob(int[] nums) {
- if(nums == null || nums.length == 0) return 0;
+ if (nums == null || nums.length == 0) return 0;
int n = nums.length;
- if(n == 1) return nums[0];
+ if (n == 1) return nums[0];
+ dp = new int[n];
return Math.max(rob(nums, 0, n - 2), rob(nums, 1, n - 1));
}
-private int rob(int[] nums, int s, int e) {
- int n = nums.length;
- if(e - s == 0) return nums[s];
- if(e - s == 1) return Math.max(nums[s], nums[s + 1]);
- int[] dp = new int[n];
- dp[s] = nums[s];
- dp[s + 1] = nums[s + 1];
- dp[s + 2] = nums[s] + nums[s + 2];
- for (int i = s + 3; i <= e; i++) {
+private int rob(int[] nums, int first, int last) {
+ if (last - first == 0) return nums[first];
+ if (last - first == 1) return Math.max(nums[first], nums[first + 1]);
+ dp[first] = nums[first];
+ dp[first + 1] = nums[first + 1];
+ dp[first + 2] = nums[first] + nums[first + 2];
+ for (int i = first + 3; i <= last; i++) {
dp[i] = Math.max(dp[i - 2], dp[i - 3]) + nums[i];
}
- return Math.max(dp[e], dp[e - 1]);
+ return Math.max(dp[last], dp[last - 1]);
}
```
**信件错排**
-题目描述:有 N 个 信 和 信封,它们被打乱,求错误装信的方式数量。
+题目描述:有 N 个 信 和 信封,它们被打乱,求错误装信方式的数量。
-定义一个数组 dp 存储错误方式数量,dp[i] 表示前 i 个信和信封的错误方式数量。假设第 i 个信装到第 j 个信封里面,而第 j 个信装到第 k 个信封里面。根据 i 和 k 是否相等,有两种情况:
+定义一个数组 dp 存储错误方式数量,dp[i] 表示 i 个信和信封的错误方式数量。假设第 i 个信装到第 j 个信封里面,而第 j 个信装到第 k 个信封里面。根据 i 和 k 是否相等,有两种情况:
-① i==k,交换 i 和 k 的信后,它们的信和信封在正确的位置,但是其余 i-2 封信有 dp[i-2] 种错误装信的方式。由于 j 有 i-1 种取值,因此共有 (i-1)\*dp[i-2] 种错误装信方式。
-
-② i != k,交换 i 和 j 的信后,第 i 个信和信封在正确的位置,其余 i-1 封信有 dp[i-1] 种错误装信方式。由于 j 有 i-1 种取值,因此共有 (n-1)\*dp[i-1] 种错误装信方式。
+- i==k,交换 i 和 k 的信后,它们的信和信封在正确的位置,但是其余 i-2 封信有 dp[i-2] 种错误装信的方式。由于 j 有 i-1 种取值,因此共有 (i-1)\*dp[i-2] 种错误装信方式。
+- i != k,交换 i 和 j 的信后,第 i 个信和信封在正确的位置,其余 i-1 封信有 dp[i-1] 种错误装信方式。由于 j 有 i-1 种取值,因此共有 (n-1)\*dp[i-1] 种错误装信方式。
综上所述,错误装信数量方式数量为:
@@ -1952,7 +1936,7 @@ dp[N] 即为所求。
定义一个数组 dp 存储最长递增子序列的长度,dp[n] 表示以 Sn 结尾的序列的最长递增子序列长度。对于一个递增子序列 {Si1 , Si2 ,...,Sim },如果 im < n 并且 Sim < Sn ,此时 {Si1 , Si2 ,..., Sim , Sn } 为一个递增子序列,递增子序列的长度增加 1。满足上述条件的递增子序列中,长度最长的那个递增子序列就是要找的,在长度最长的递增子序列上加上 Sn 就构成了以 Sn 为结尾的最长递增子序列。因此 dp[n] = max{ dp[i]+1 | Si < Sn && i < n} 。
-因为在求 dp[n] 时可能无法找到一个满足条件的递增子序列,此时 {Sn } 就构成了递增子序列,因此需要对前面的求解方程做修改,令 dp[n] 最小为 1,即:
+因为在求 dp[n] 时可能无法找到一个满足条件的递增子序列,此时 {Sn } 就构成了递增子序列,需要对前面的求解方程做修改,令 dp[n] 最小为 1,即:
@@ -1981,15 +1965,18 @@ public int lengthOfLIS(int[] nums) {
}
```
-以上解法的时间复杂度为 O(n2 ) ,可以使用二分查找使得时间复杂度降低为 O(nlogn )。定义一个 tails 数组,其中 tails[i] 存储长度为 i + 1 的最长递增子序列的最后一个元素,例如对于数组 [4,5,6,3],有
+以上解法的时间复杂度为 O(n2 ) ,可以使用二分查找使得时间复杂度降低为 O(nlogn)。定义一个 tails 数组,其中 tails[i] 存储长度为 i + 1 的最长递增子序列的最后一个元素,例如对于数组 [4,5,6,3],有
```html
len = 1 : [4], [5], [6], [3] => tails[0] = 3
len = 2 : [4, 5], [5, 6] => tails[1] = 5
-len = 3 : [4, 5, 6] => tails[2] = 6
+len = 3 : [4, 5, 6] => tails[2] = 6
```
-对于一个元素 x,如果它大于 tails 数组所有的值,那么把它添加到 tails 后面;如果 tails[i-1] < x <= tails[i],那么更新 tails[i] = x 。
+对于一个元素 x,
+
+- 如果它大于 tails 数组所有的值,那么把它添加到 tails 后面,表示最长递增子序列长度加 1;
+- 如果 tails[i-1] < x <= tails[i],那么更新 tails[i-1] = x。
可以看出 tails 数组保持有序,因此在查找 Si 位于 tails 数组的位置时就可以使用二分查找。
@@ -1998,22 +1985,59 @@ public int lengthOfLIS(int[] nums) {
int n = nums.length;
int[] tails = new int[n];
int size = 0;
- for(int i = 0; i < n; i++){
- int idx = binarySearch(tails, 0, size, nums[i]);
- tails[idx] = nums[i];
- if(idx == size) size++;
+ for (int i = 0; i < n; i++) {
+ int index = binarySearch(tails, 0, size, nums[i]);
+ tails[index] = nums[i];
+ if (index == size) size++;
}
return size;
}
-private int binarySearch(int[] nums, int sIdx, int eIdx, int key){
- while(sIdx < eIdx){
- int mIdx = sIdx + (eIdx - sIdx) / 2;
- if(nums[mIdx] == key) return mIdx;
- else if(nums[mIdx] > key) eIdx = mIdx;
- else sIdx = mIdx + 1;
+private int binarySearch(int[] nums, int first, int last, int key) {
+ while (first < last) {
+ int mid = first + (last - first) / 2;
+ if (nums[mid] == key) return mid;
+ else if (nums[mid] > key) last = mid;
+ else first = mid + 1;
}
- return sIdx;
+ return first;
+}
+```
+
+**一组整数对能够构成的最长链**
+
+[Leetcode : 646. Maximum Length of Pair Chain (Medium)](https://leetcode.com/problems/maximum-length-of-pair-chain/description/)
+
+```html
+Input: [[1,2], [2,3], [3,4]]
+Output: 2
+Explanation: The longest chain is [1,2] -> [3,4]
+```
+
+题目描述:对于 (a, b) 和 (c, d) ,如果 b < c,则它们可以构成一条链。
+
+```java
+public int findLongestChain(int[][] pairs) {
+ if(pairs == null || pairs.length == 0) {
+ return 0;
+ }
+ Arrays.sort(pairs, (a, b) -> (a[0] - b[0]));
+ int n = pairs.length;
+ int[] dp = new int[n];
+ Arrays.fill(dp, 1);
+ for(int i = 0; i < n; i++) {
+ for(int j = 0; j < i; j++) {
+ if(pairs[i][0] > pairs[j][1]){
+ dp[i] = Math.max(dp[i], dp[j] + 1);
+ }
+ }
+ }
+
+ int ret = 0;
+ for(int num : dp) {
+ ret = Math.max(ret, num);
+ }
+ return ret;
}
```
@@ -2021,6 +2045,19 @@ private int binarySearch(int[] nums, int sIdx, int eIdx, int key){
[Leetcode : 376. Wiggle Subsequence (Medium)](https://leetcode.com/problems/wiggle-subsequence/description/)
+```html
+Input: [1,7,4,9,2,5]
+Output: 6
+The entire sequence is a wiggle sequence.
+
+Input: [1,17,5,10,13,15,10,5,16,8]
+Output: 7
+There are several subsequences that achieve this length. One is [1,17,10,13,10,16,8].
+
+Input: [1,2,3,4,5,6,7,8,9]
+Output: 2
+```
+
要求:使用 O(n) 时间复杂度求解。
使用两个状态 up 和 down。
@@ -2038,17 +2075,16 @@ public int wiggleMaxLength(int[] nums) {
}
```
-### 最长公共子系列
+### 最长公共子序列
对于两个子序列 S1 和 S2,找出它们最长的公共子序列。
定义一个二维数组 dp 用来存储最长公共子序列的长度,其中 dp[i][j] 表示 S1 的前 i 个字符与 S2 的前 j 个字符最长公共子序列的长度。考虑 S1i 与 S2j 值是否相等,分为两种情况:
-① 当 S1i ==S2j 时,那么就能在 S1 的前 i-1 个字符与 S2 的前 j-1 个字符最长公共子序列的基础上再加上 S1i 这个值,最长公共子序列长度加 1 ,即 dp[i][j] = dp[i-1][j-1] + 1。
+- 当 S1i ==S2j 时,那么就能在 S1 的前 i-1 个字符与 S2 的前 j-1 个字符最长公共子序列的基础上再加上 S1i 这个值,最长公共子序列长度加 1 ,即 dp[i][j] = dp[i-1][j-1] + 1。
+- 当 S1i != S2j 时,此时最长公共子序列为 S1 的前 i-1 个字符和 S2 的前 j 个字符最长公共子序列,与 S1 的前 i 个字符和 S2 的前 j-1 个字符最长公共子序列,它们的最大者,即 dp[i][j] = max{ dp[i-1][j], dp[i][j-1] }。
-② 当 S1i != S2j 时,此时最长公共子序列为 S1 的前 i-1 个字符和 S2 的前 j 个字符最长公共子序列,与 S1 的前 i 个字符和 S2 的前 j-1 个字符最长公共子序列,它们的最大者,即 dp[i][j] = max{ dp[i-1][j], dp[i][j-1] }。
-
-综上,最长公共子系列的状态转移方程为:
+综上,最长公共子序列的状态转移方程为:
@@ -2056,9 +2092,9 @@ public int wiggleMaxLength(int[] nums) {
与最长递增子序列相比,最长公共子序列有以下不同点:
-① 针对的是两个序列,求它们的最长公共子序列。
-② 在最长递增子序列中,dp[i] 表示以 Si 为结尾的最长递增子序列长度,子序列必须包含 Si ;在最长公共子序列中,dp[i][j] 表示 S1 中前 i 个字符与 S2 中前 j 个字符的最长公共子序列长度,不一定包含 S1i 和 S2j 。
-③ 由于 2 ,在求最终解时,最长公共子序列中 dp[N][M] 就是最终解,而最长递增子序列中 dp[N] 不是最终解,因为以 SN 为结尾的最长递增子序列不一定是整个序列最长递增子序列,需要遍历一遍 dp 数组找到最大者。
+- 针对的是两个序列,求它们的最长公共子序列。
+- 在最长递增子序列中,dp[i] 表示以 Si 为结尾的最长递增子序列长度,子序列必须包含 Si ;在最长公共子序列中,dp[i][j] 表示 S1 中前 i 个字符与 S2 中前 j 个字符的最长公共子序列长度,不一定包含 S1i 和 S2j 。
+- 由于 2 ,在求最终解时,最长公共子序列中 dp[N][M] 就是最终解,而最长递增子序列中 dp[N] 不是最终解,因为以 SN 为结尾的最长递增子序列不一定是整个序列最长递增子序列,需要遍历一遍 dp 数组找到最大者。
```java
public int lengthOfLCS(int[] nums1, int[] nums2) {
@@ -2080,8 +2116,8 @@ public int lengthOfLCS(int[] nums1, int[] nums2) {
定义一个二维数组 dp 存储最大价值,其中 dp[i][j] 表示体积不超过 j 的情况下,前 i 件物品能达到的最大价值。设第 i 件物品体积为 w,价值为 v,根据第 i 件物品是否添加到背包中,可以分两种情况讨论:
-① 第 i 件物品没添加到背包,总体积不超过 j 的前 i 件物品的最大价值就是总体积不超过 j 的前 i-1 件物品的最大价值,dp[i][j] = dp[i-1][j]。
-② 第 i 件物品添加到背包中,dp[i][j] = dp[i-1][j-w] + v。
+- 第 i 件物品没添加到背包,总体积不超过 j 的前 i 件物品的最大价值就是总体积不超过 j 的前 i-1 件物品的最大价值,dp[i][j] = dp[i-1][j]。
+- 第 i 件物品添加到背包中,dp[i][j] = dp[i-1][j-w] + v。
第 i 件物品可添加也可以不添加,取决于哪种情况下最大价值更大。
@@ -2091,19 +2127,18 @@ public int lengthOfLCS(int[] nums1, int[] nums2) {
```java
public int knapsack(int W, int N, int[] weights, int[] values) {
- int[][] dp = new int[N][W];
- for (int i = W - 1; i >= 0; i--) {
- dp[0][i] = i > weights[0] ? values[0] : 0;
- }
- for (int i = 1; i < N; i++) {
- for (int j = W - 1; j >= weights[i]; j--) {
- dp[i][j] = Math.max(dp[i - 1][j], dp[i - 1][j - weights[i]] + values[i]);
- }
- for (int j = weights[i - 1] - 1; j >= 0; j--) {
- dp[i][j] = dp[i - 1][j];
+ int[][] dp = new int[N + 1][W + 1];
+ for (int i = 1; i <= N; i++) {
+ int w = weights[i - 1], v = values[i - 1];
+ for (int j = 1; j <= W; j++) {
+ if (j >= w) {
+ dp[i][j] = Math.max(dp[i - 1][j], dp[i - 1][j - w] + v);
+ } else {
+ dp[i][j] = dp[i - 1][j];
+ }
}
}
- return dp[N - 1][W - 1];
+ return dp[N][W];
}
```
@@ -2113,7 +2148,22 @@ public int knapsack(int W, int N, int[] weights, int[] values) {
-因为 dp[j-w] 表示 dp[i-1][j-w],因此不能先求 dp[i][j-w] 防止将 dp[i-1][j-w] 覆盖。也就是说要先计算 dp[i][j] 再计算 dp[i][j-w],在程序实现时需要按倒序来循环求解。
+因为 dp[j-w] 表示 dp[i-1][j-w],因此不能先求 dp[i][j-w],以防止将 dp[i-1][j-w] 覆盖。也就是说要先计算 dp[i][j] 再计算 dp[i][j-w],在程序实现时需要按倒序来循环求解。
+
+```java
+public int knapsack(int W, int N, int[] weights, int[] values) {
+ int[] dp = new int[W + 1];
+ for (int i = 1; i <= N; i++) {
+ int w = weights[i - 1], v = values[i - 1];
+ for (int j = W; j >= 1; j--) {
+ if (j >= w) {
+ dp[j] = Math.max(dp[j], dp[j - w] + v);
+ }
+ }
+ }
+ return dp[W];
+}
+```
**无法使用贪心算法的解释**
@@ -2127,45 +2177,45 @@ public int knapsack(int W, int N, int[] weights, int[] values) {
**变种**
-完全背包:物品可以无限个,可以转换为 0-1 背包,令每种物品的体积和价值变为 1/2/4... 倍数,把它们都当成一个新物品,然后一种物品只能添加一次。
+- 完全背包:物品数量为无限个
-多重背包:物品数量有限制,同样可以转换为 0-1 背包。
+- 多重背包:物品数量有限制
-多维费用背包:物品不仅有重量,还有体积,同时考虑这两种限制。
+- 多维费用背包:物品不仅有重量,还有体积,同时考虑这两种限制
-其它:物品之间相互约束或者依赖。
+- 其它:物品之间相互约束或者依赖
**划分数组为和相等的两部分**
[Leetcode : 416. Partition Equal Subset Sum (Medium)](https://leetcode.com/problems/partition-equal-subset-sum/description/)
-可以看成一个背包大小为 sum/2 的 0-1 背包问题,但是也有不同的地方,这里没有价值属性,并且背包必须被填满。
+```html
+Input: [1, 5, 11, 5]
-以下实现使用了空间优化。
+Output: true
+
+Explanation: The array can be partitioned as [1, 5, 5] and [11].
+```
+
+可以看成一个背包大小为 sum/2 的 0-1 背包问题。
```java
-public boolean canPartition(int[] nums) {
- int sum = 0;
- for (int num : nums) {
- sum += num;
- }
- if (sum % 2 != 0) {
- return false;
- }
- int W = sum / 2;
- boolean[] dp = new boolean[W + 1];
- int n = nums.length;
- for(int i = 0; i <= W; i++) {
- if(nums[0] == i) dp[i] = true;
- }
- for(int i = 1; i < n; i++) {
- for(int j = W; j >= nums[i]; j--) {
- dp[j] = dp[j] || dp[j - nums[i]];
- }
- }
-
- return dp[W];
-}
+ public boolean canPartition(int[] nums) {
+ int sum = 0;
+ for (int num : nums) sum += num;
+ if (sum % 2 != 0) return false;
+ int W = sum / 2;
+ boolean[] dp = new boolean[W + 1];
+ dp[0] = true;
+ for (int num : nums) { // 0-1 背包一个物品只能用一次
+ for (int i = W; i >= 0; i--) { // 从后往前,先计算 dp[i] 再计算 dp[i-num]
+ if (num <= i) {
+ dp[i] = dp[i] || dp[i - num];
+ }
+ }
+ }
+ return dp[W];
+ }
```
**字符串按单词列表分割**
@@ -2178,16 +2228,20 @@ dict = ["leet", "code"].
Return true because "leetcode" can be segmented as "leet code".
```
+这是一个完全背包问题,和 0-1 背包不同的是,完全背包中物品可以使用多次。在这一题当中,词典中的单词可以被使用多次。
+
+0-1 背包和完全背包在实现上的不同之处是,0-1 背包对物品的迭代是在最外层,而完全背包对物品的迭代是在最里层。
+
```java
public boolean wordBreak(String s, List wordDict) {
int n = s.length();
boolean[] dp = new boolean[n + 1];
dp[0] = true;
for (int i = 1; i <= n; i++) {
- for (String word : wordDict) {
- if (word.length() <= i
- && word.equals(s.substring(i - word.length(), i))) {
- dp[i] = dp[i] || dp[i - word.length()];
+ for (String word : wordDict) { // 每个单词可以使用多次
+ int len = word.length();
+ if (len <= i && word.equals(s.substring(i - len, i))) {
+ dp[i] = dp[i] || dp[i - len];
}
}
}
@@ -2213,7 +2267,9 @@ Explanation:
There are 5 ways to assign symbols to make the sum of nums be target 3.
```
-该问题可以转换为 subset sum 问题,从而使用 0-1 背包的方法来求解。可以将这组数看成两部分,P 和 N,其中 P 使用正号,N 使用负号,有以下推导:
+该问题可以转换为 Subset Sum 问题,从而使用 0-1 背包的方法来求解。
+
+可以将这组数看成两部分,P 和 N,其中 P 使用正号,N 使用负号,有以下推导:
```html
sum(P) - sum(N) = target
@@ -2226,26 +2282,34 @@ sum(P) + sum(N) + sum(P) - sum(N) = target + sum(P) + sum(N)
```java
public int findTargetSumWays(int[] nums, int S) {
int sum = 0;
- for (int num : nums) {
- sum += num;
- }
- if (sum < S || (sum + S) % 2 == 1) {
- return 0;
- }
- return subsetSum(nums, (sum + S) >>> 1);
-}
-
-private int subsetSum(int[] nums, int targetSum) {
- Arrays.sort(nums);
- int[] dp = new int[targetSum + 1];
+ for (int num : nums) sum += num;
+ if (sum < S || (sum + S) % 2 == 1) return 0;
+ int W = (sum + S) / 2;
+ int[] dp = new int[W + 1];
dp[0] = 1;
- for (int i = 0; i < nums.length; i++) {
- int num = nums[i];
- for (int j = targetSum; j >= num; j--) {
- dp[j] = dp[j] + dp[j - num];
+ for (int num : nums) {
+ for (int i = W; i >= 0; i--) {
+ if (num <= i) {
+ dp[i] = dp[i] + dp[i - num];
+ }
}
}
- return dp[targetSum];
+ return dp[W];
+}
+```
+
+DFS 解法:
+
+```java
+public int findTargetSumWays(int[] nums, int S) {
+ return findTargetSumWays(nums, 0, S);
+}
+
+private int findTargetSumWays(int[] nums, int start, int S) {
+ if (start == nums.length) {
+ return S == 0 ? 1 : 0;
+ }
+ return findTargetSumWays(nums, start + 1, S + nums[start]) + findTargetSumWays(nums, start + 1, S - nums[start]);
}
```
@@ -2257,7 +2321,7 @@ private int subsetSum(int[] nums, int targetSum) {
Input: Array = {"10", "0001", "111001", "1", "0"}, m = 5, n = 3
Output: 4
-Explanation: This are totally 4 strings can be formed by the using of 5 0s and 3 1s, which are “10,”0001”,”1”,”0”
+Explanation: There are totally 4 strings can be formed by the using of 5 0s and 3 1s, which are "10","0001","1","0"
```
这是一个多维费用的 0-1 背包问题,有两个背包大小,0 的数量和 1 的数量。
@@ -2265,20 +2329,16 @@ Explanation: This are totally 4 strings can be formed by the using of 5 0s and 3
```java
public int findMaxForm(String[] strs, int m, int n) {
if (strs == null || strs.length == 0) return 0;
- int l = strs.length;
int[][] dp = new int[m + 1][n + 1];
- for (int i = 0; i < l; i++) {
- String s = strs[i];
+ for (String s : strs) { // 每个字符串只能用一次
int ones = 0, zeros = 0;
for (char c : s.toCharArray()) {
if (c == '0') zeros++;
- else if (c == '1') ones++;
+ else ones++;
}
- for (int j = m; j >= zeros; j--) {
- for (int k = n; k >= ones; k--) {
- if (zeros <= j && ones <= k) {
- dp[j][k] = Math.max(dp[j][k], dp[j - zeros][k - ones] + 1);
- }
+ for (int i = m; i >= zeros; i--) {
+ for (int j = n; j >= ones; j--) {
+ dp[i][j] = Math.max(dp[i][j], dp[i - zeros][j - ones] + 1);
}
}
}
@@ -2290,19 +2350,30 @@ public int findMaxForm(String[] strs, int m, int n) {
[Leetcode : 322. Coin Change (Medium)](https://leetcode.com/problems/coin-change/description/)
+```html
+Example 1:
+coins = [1, 2, 5], amount = 11
+return 3 (11 = 5 + 5 + 1)
+
+Example 2:
+coins = [2], amount = 3
+return -1.
+```
+
题目描述:给一些面额的硬币,要求用这些硬币来组成给定面额的钱数,并且使得硬币数量最少。硬币可以重复使用。
-这是一个完全背包问题,完全背包问题和 0-1 背包问题在实现上唯一的不同是,第二层循环是从 0 开始的,而不是从尾部开始。
+这是一个完全背包问题。
```java
public int coinChange(int[] coins, int amount) {
+ if (coins == null || coins.length == 0) return 0;
int[] dp = new int[amount + 1];
Arrays.fill(dp, amount + 1);
dp[0] = 0;
for (int i = 1; i <= amount; i++) {
- for (int j = 0; j < coins.length; j++) {
- if (coins[j] <= i) {
- dp[i] = Math.min(dp[i], dp[i - coins[j]] + 1);
+ for (int c : coins) {
+ if (c <= i) {
+ dp[i] = Math.min(dp[i], dp[i - c] + 1);
}
}
}
@@ -2332,14 +2403,17 @@ Note that different sequences are counted as different combinations.
Therefore the output is 7.
```
+完全背包。
+
```java
public int combinationSum4(int[] nums, int target) {
+ if (nums == null || nums.length == 0) return 0;
int[] dp = new int[target + 1];
dp[0] = 1;
for (int i = 1; i <= target; i++) {
- for (int j = 0; j < nums.length; j++) {
- if(nums[j] <= i) {
- dp[i] += dp[i - nums[j]];
+ for (int num : nums) {
+ if (num <= i) {
+ dp[i] += dp[i - num];
}
}
}
@@ -2347,6 +2421,38 @@ public int combinationSum4(int[] nums, int target) {
}
```
+**只能进行 k 次的股票交易**
+
+[Leetcode : 188. Best Time to Buy and Sell Stock IV (Hard)](https://leetcode.com/problems/best-time-to-buy-and-sell-stock-iv/description/)
+
+```html
+dp[i, j] = max(dp[i, j-1], prices[j] - prices[jj] + dp[i-1, jj]) { jj in range of [0, j-1] }
+ = max(dp[i, j-1], prices[j] + max(dp[i-1, jj] - prices[jj]))
+```
+
+```java
+public int maxProfit(int k, int[] prices) {
+ int n = prices.length;
+ if (k >= n/2) { // 这种情况下该问题退化为普通的股票交易问题
+ int maxPro = 0;
+ for (int i = 1; i < n; i++) {
+ if (prices[i] > prices[i-1])
+ maxPro += prices[i] - prices[i-1];
+ }
+ return maxPro;
+ }
+ int[][] dp = new int[k + 1][n];
+ for (int i = 1; i <= k; i++) {
+ int localMax = dp[i - 1][0] - prices[0];
+ for (int j = 1; j < n; j++) {
+ dp[i][j] = Math.max(dp[i][j - 1], prices[j] + localMax);
+ localMax = Math.max(localMax, dp[i - 1][j] - prices[j]);
+ }
+ }
+ return dp[k][n - 1];
+}
+```
+
**只能进行两次的股票交易**
[Leetcode : 123. Best Time to Buy and Sell Stock III (Hard)](https://leetcode.com/problems/best-time-to-buy-and-sell-stock-iii/description/)
@@ -2365,58 +2471,35 @@ public int maxProfit(int[] prices) {
}
```
-**只能进行 k 次的股票交易**
-
-[Leetcode : 188. Best Time to Buy and Sell Stock IV (Hard)](https://leetcode.com/problems/best-time-to-buy-and-sell-stock-iv/description/)
-
-```html
-dp[i, j] = max(dp[i, j-1], prices[j] - prices[jj] + dp[i-1, jj]) { jj in range of [0, j-1] } = max(dp[i, j-1], prices[j] + max(dp[i-1, jj] - prices[jj]))
-```
-
-```java
-public int maxProfit(int k, int[] prices) {
- int n = prices.length;
- if (k >= n/2) {
- int maxPro = 0;
- for (int i = 1; i < n; i++) {
- if (prices[i] > prices[i-1])
- maxPro += prices[i] - prices[i-1];
- }
- return maxPro;
- }
- int[][] dp = new int[k + 1][n];
- for (int i = 1; i <= k; i++) {
- int localMax = dp[i - 1][0] - prices[0];
- for (int j = 1; j < n; j++) {
- dp[i][j] = Math.max(dp[i][j - 1], prices[j] + localMax);
- localMax = Math.max(localMax, dp[i - 1][j] - prices[j]);
- }
- }
- return dp[k][n - 1];
-}
-```
-
### 数组区间
**数组区间和**
[Leetcode : 303. Range Sum Query - Immutable (Easy)](https://leetcode.com/problems/range-sum-query-immutable/description/)
-求区间 i \~ j 的和,可以转换为 sum[j] - sum[i-1],其中 sum[i] 为 0 \~ j 的和。
+```html
+Given nums = [-2, 0, 3, -5, 2, -1]
+
+sumRange(0, 2) -> 1
+sumRange(2, 5) -> -1
+sumRange(0, 5) -> -3
+```
+
+求区间 i \~ j 的和,可以转换为 sum[j] - sum[i-1],其中 sum[i] 为 0 \~ i 的和。
```java
class NumArray {
-
- int[] nums;
+ private int[] sums;
public NumArray(int[] nums) {
- for(int i = 1; i < nums.length; i++)
- nums[i] += nums[i - 1];
- this.nums = nums;
+ sums = new int[nums.length];
+ for (int i = 0; i < nums.length; i++) {
+ sums[i] = i == 0 ? nums[0] : sums[i - 1] + nums[i];
+ }
}
public int sumRange(int i, int j) {
- return i == 0 ? nums[j] : nums[j] - nums[i - 1];
+ return i == 0 ? sums[j] : sums[j] - sums[i - 1];
}
}
```
@@ -2425,33 +2508,21 @@ class NumArray {
[Leetcode : 53. Maximum Subarray (Easy)](https://leetcode.com/problems/maximum-subarray/description/)
-令 sum[i] 为以 num[i] 为结尾的子数组最大的和,可以由 sum[i-1] 得到 sum[i] 的值,如果 sum[i-1] 小于 0,那么以 num[i] 为结尾的子数组不能包含前面的内容,因为加上前面的部分,那么和一定会比 num[i] 还小。
-
-```java
-public int maxSubArray(int[] nums) {
- int n = nums.length;
- int[] sum = new int[n];
- sum[0] = nums[0];
- int max = sum[0];
- for(int i = 1; i < n; i++){
- sum[i] = (sum[i-1] > 0 ? sum[i-1] : 0) + nums[i];
- max = Math.max(max, sum[i]);
- }
- return max;
-}
+```html
+For example, given the array [-2,1,-3,4,-1,2,1,-5,4],
+the contiguous subarray [4,-1,2,1] has the largest sum = 6.
```
-空间复杂度可以优化成 O(1) 空间复杂度
-
```java
public int maxSubArray(int[] nums) {
- int max = nums[0];
- int oldsum = nums[0];
+ if (nums == null || nums.length == 0) return 0;
+ int preSum = nums[0];
+ int maxSum = preSum;
for (int i = 1; i < nums.length; i++) {
- oldsum = (oldsum > 0 ? oldsum: 0) + nums[i];
- max = Math.max(max, oldsum);
+ preSum = preSum > 0 ? preSum + nums[i] : nums[i];
+ maxSum = Math.max(maxSum, preSum);
}
- return max;
+ return maxSum;
}
```
@@ -2461,25 +2532,25 @@ public int maxSubArray(int[] nums) {
```html
A = [1, 2, 3, 4]
-
return: 3, for 3 arithmetic slices in A: [1, 2, 3], [2, 3, 4] and [1, 2, 3, 4] itself.
```
-对于 (1,2,3,4),它有三种组成递增子区间的方式,而对于 (1,2,3,4,5),它组成递增子区间的方式除了 (1,2,3,4) 的三种外还多了一种,即 (1,2,3,4,5),因此 dp[i] = dp[i - 1] + 1。
+dp[i] 表示以 A[i] 为结尾的等差递增子区间的个数。
+
+如果 A[i] - A[i - 1] == A[i - 1] - A[i - 2],表示 [A[i - 2], A[i - 1], A[i]] 是一个等差递增子区间。如果 [A[i - 3], A[i - 2], A[i - 1]] 是一个等差递增子区间,那么 [A[i - 3], A[i - 2], A[i - 1], A[i]] 也是。因此在这个条件下,dp[i] = dp[i-1] + 1。
```java
public int numberOfArithmeticSlices(int[] A) {
+ if (A == null || A.length == 0) return 0;
int n = A.length;
int[] dp = new int[n];
- for(int i = 2; i < n; i++) {
- if(A[i] - A[i - 1] == A[i - 1] - A[i - 2]) {
+ for (int i = 2; i < n; i++) {
+ if (A[i] - A[i - 1] == A[i - 1] - A[i - 2]) {
dp[i] = dp[i - 1] + 1;
}
}
int ret = 0;
- for(int cnt : dp) {
- ret += cnt;
- }
+ for (int cnt : dp) ret += cnt;
return ret;
}
```
@@ -2490,6 +2561,12 @@ public int numberOfArithmeticSlices(int[] A) {
[Leetcode : 583. Delete Operation for Two Strings (Medium)](https://leetcode.com/problems/delete-operation-for-two-strings/description/)
+```html
+Input: "sea", "eat"
+Output: 2
+Explanation: You need one step to make "sea" to "ea" and another step to make "eat" to "ea".
+```
+
可以转换为求两个字符串的最长公共子序列问题。
```java
@@ -2499,18 +2576,189 @@ public int minDistance(String word1, String word2) {
for (int i = 0; i <= m; i++) {
for (int j = 0; j <= n; j++) {
if (i == 0 || j == 0) continue;
- dp[i][j] = word1.charAt(i - 1) == word2.charAt(j - 1) ? dp[i - 1][j - 1] + 1
- : Math.max(dp[i][j - 1], dp[i - 1][j]);
+ dp[i][j] = word1.charAt(i - 1) == word2.charAt(j - 1) ?
+ dp[i - 1][j - 1] + 1 : Math.max(dp[i][j - 1], dp[i - 1][j]);
}
}
return m + n - 2 * dp[m][n];
}
```
-**修改一个字符串称为另一个字符串** // TODO
+**修改一个字符串称为另一个字符串**
[Leetcode : 72. Edit Distance (Hard)](https://leetcode.com/problems/edit-distance/description/)
+```java
+public int minDistance(String word1, String word2) {
+ if (word1 == null || word2 == null) {
+ return 0;
+ }
+ int m = word1.length(), n = word2.length();
+ int[][] dp = new int[m + 1][n + 1];
+ for (int i = 1; i <= m; i++) {
+ dp[i][0] = i;
+ }
+ for (int i = 1; i <= n; i++) {
+ dp[0][i] = i;
+ }
+ for (int i = 1; i <= m; i++) {
+ for (int j = 1; j <= n; j++) {
+ if (word1.charAt(i - 1) == word2.charAt(j - 1)) {
+ dp[i][j] = dp[i - 1][j - 1];
+ } else {
+ dp[i][j] = Math.min(dp[i - 1][j - 1], Math.min(dp[i][j - 1], dp[i - 1][j])) + 1;
+ }
+ }
+ }
+ return dp[m][n];
+}
+```
+
+### 分割整数
+
+**分割整数的最大乘积**
+
+[Leetcode : 343. Integer Break (Medim)](https://leetcode.com/problems/integer-break/description/)
+
+题目描述:For example, given n = 2, return 1 (2 = 1 + 1); given n = 10, return 36 (10 = 3 + 3 + 4).
+
+```java
+public int integerBreak(int n) {
+ int[] dp = new int[n + 1];
+ dp[1] = 1;
+ for(int i = 2; i <= n; i++) {
+ for(int j = 1; j <= i - 1; j++) {
+ dp[i] = Math.max(dp[i], Math.max(j * dp[i - j], j * (i - j)));
+ }
+ }
+ return dp[n];
+}
+```
+
+**按平方数来分割整数**
+
+[Leetcode : 279. Perfect Squares(Medium)](https://leetcode.com/problems/perfect-squares/description/)
+
+题目描述:For example, given n = 12, return 3 because 12 = 4 + 4 + 4; given n = 13, return 2 because 13 = 4 + 9.
+
+```java
+public int numSquares(int n) {
+ List squareList = generateSquareList(n);
+ int[] dp = new int[n + 1];
+ for (int i = 1; i <= n; i++) {
+ int max = Integer.MAX_VALUE;
+ for (int square : squareList) {
+ if (square > i) break;
+ max = Math.min(max, dp[i - square] + 1);
+ }
+ dp[i] = max;
+ }
+ return dp[n];
+}
+
+private List generateSquareList(int n) {
+ List squareList = new ArrayList<>();
+ int diff = 3;
+ int square = 1;
+ while (square <= n) {
+ squareList.add(square);
+ square += diff;
+ diff += 2;
+ }
+ return squareList;
+}
+```
+
+**分割整数构成字母字符串**
+
+[Leetcode : 91. Decode Ways (Medium)](https://leetcode.com/problems/decode-ways/description/)
+
+题目描述:Given encoded message "12", it could be decoded as "AB" (1 2) or "L" (12).
+
+```java
+public int numDecodings(String s) {
+ if(s == null || s.length() == 0) return 0;
+ int n = s.length();
+ int[] dp = new int[n + 1];
+ dp[0] = 1;
+ dp[1] = s.charAt(0) == '0' ? 0 : 1;
+ for(int i = 2; i <= n; i++) {
+ int one = Integer.valueOf(s.substring(i - 1, i));
+ if(one != 0) dp[i] += dp[i - 1];
+ if(s.charAt(i - 2) == '0') continue;
+ int two = Integer.valueOf(s.substring(i - 2, i));
+ if(two <= 26) dp[i] += dp[i - 2];
+ }
+ return dp[n];
+}
+```
+
+### 矩阵路径
+
+**矩阵的总路径数**
+
+[Leetcode : 62. Unique Paths (Medium)](https://leetcode.com/problems/unique-paths/description/)
+
+题目描述:统计从矩阵左上角到右下角的路径总数,每次只能向右或者向下移动。
+
+
+
+```java
+public int uniquePaths(int m, int n) {
+ int[] dp = new int[n];
+ Arrays.fill(dp, 1);
+ for (int i = 1; i < m; i++) {
+ for (int j = 1; j < n; j++) {
+ dp[j] = dp[j] + dp[j - 1];
+ }
+ }
+ return dp[n - 1];
+}
+```
+
+也可以直接用数学公式求解,这是一个组合问题。机器人总共移动的次数 S=m+n-2,向下移动的次数 D=m-1,那么问题可以看成从 S 从取出 D 个位置的组合数量,这个问题的解为 C(S, D)。
+
+```java
+public int uniquePaths(int m, int n) {
+ int S = m + n - 2; // 总共的移动次数
+ int D = m - 1; // 向下的移动次数
+ long ret = 1;
+ for (int i = 1; i <= D; i++) {
+ ret = ret * (S - D + i) / i;
+ }
+ return (int) ret;
+}
+```
+
+**矩阵的最小路径和**
+
+[Leetcode : 64. Minimum Path Sum (Medium)](https://leetcode.com/problems/minimum-path-sum/description/)
+
+```html
+[[1,3,1],
+ [1,5,1],
+ [4,2,1]]
+Given the above grid map, return 7. Because the path 1→3→1→1→1 minimizes the sum.
+```
+
+题目描述:求从矩阵的左上角到右下角的最小路径和,每次只能向左和向下移动。
+
+```java
+public int minPathSum(int[][] grid) {
+ if (grid.length == 0 || grid[0].length == 0) return 0;
+ int m = grid.length, n = grid[0].length;
+ int[] dp = new int[n];
+ for (int i = 0; i < m; i++) {
+ for (int j = 0; j < n; j++) {
+ if (j == 0) dp[0] = dp[0] + grid[i][0]; // 只能从上侧走到该位置
+ else if (i == 0) dp[j] = dp[j - 1] + grid[0][j]; // 只能从右侧走到该位置
+ else dp[j] = Math.min(dp[j - 1], dp[j]) + grid[i][j];
+ }
+ }
+ return dp[n - 1];
+}
+```
+
### 其它问题
**需要冷却期的股票交易**
@@ -2519,77 +2767,63 @@ public int minDistance(String word1, String word2) {
题目描述:交易之后需要有一天的冷却时间。
-
-
-```html
-s0[i] = max(s0[i - 1], s2[i - 1]); // Stay at s0, or rest from s2
-s1[i] = max(s1[i - 1], s0[i - 1] - prices[i]); // Stay at s1, or buy from s0
-s2[i] = s1[i - 1] + prices[i]; // Only one way from s1
-```
+
```java
public int maxProfit(int[] prices) {
if (prices == null || prices.length == 0) return 0;
- int n = prices.length;
- int[] s0 = new int[n];
- int[] s1 = new int[n];
- int[] s2 = new int[n];
- s0[0] = 0;
- s1[0] = -prices[0];
- s2[0] = Integer.MIN_VALUE;
- for (int i = 1; i < n; i++) {
- s0[i] = Math.max(s0[i - 1], s2[i - 1]);
- s1[i] = Math.max(s1[i - 1], s0[i - 1] - prices[i]);
- s2[i] = Math.max(s2[i - 1], s1[i - 1] + prices[i]);
+ int N = prices.length;
+ int[] buy = new int[N];
+ int[] s1 = new int[N];
+ int[] sell = new int[N];
+ int[] s2 = new int[N];
+ s1[0] = buy[0] = -prices[0];
+ sell[0] = s2[0] = 0;
+ for (int i = 1; i < N; i++) {
+ buy[i] = s2[i - 1] - prices[i];
+ s1[i] = Math.max(buy[i - 1], s1[i - 1]);
+ sell[i] = Math.max(buy[i - 1], s1[i - 1]) + prices[i];
+ s2[i] = Math.max(s2[i - 1], sell[i - 1]);
}
- return Math.max(s0[n - 1], s2[n - 1]);
+ return Math.max(sell[N - 1], s2[N - 1]);
}
```
-**统计从 0 \~ n 每个数的二进制表示中 1 的个数**
+**需要交易费用的股票交易**
-[Leetcode : 338. Counting Bits (Medium)](https://leetcode.com/problems/counting-bits/description/)
+[Leetcode : 714. Best Time to Buy and Sell Stock with Transaction Fee (Medium)](https://leetcode.com/problems/best-time-to-buy-and-sell-stock-with-transaction-fee/description/)
-对于数字 6(110),它可以看成是数字 2(10) 前面加上一个 1 ,因此 dp[i] = dp[i&(i-1)] + 1;
-
-```java
- public int[] countBits(int num) {
- int[] ret = new int[num + 1];
- for(int i = 1; i <= num; i++){
- ret[i] = ret[i&(i-1)] + 1;
- }
- return ret;
- }
+```html
+Input: prices = [1, 3, 2, 8, 4, 9], fee = 2
+Output: 8
+Explanation: The maximum profit can be achieved by:
+Buying at prices[0] = 1
+Selling at prices[3] = 8
+Buying at prices[4] = 4
+Selling at prices[5] = 9
+The total profit is ((8 - 1) - 2) + ((9 - 4) - 2) = 8.
```
-**一组整数对能够构成的最长链**
+题目描述:每交易一次,都要支付一定的费用。
-[Leetcode : 646. Maximum Length of Pair Chain (Medium)](https://leetcode.com/problems/maximum-length-of-pair-chain/description/)
-
-对于 (a, b) 和 (c, d) ,如果 b < c,则它们可以构成一条链。
+
```java
-public int findLongestChain(int[][] pairs) {
- if(pairs == null || pairs.length == 0) {
- return 0;
+public int maxProfit(int[] prices, int fee) {
+ int N = prices.length;
+ int[] buy = new int[N];
+ int[] s1 = new int[N];
+ int[] sell = new int[N];
+ int[] s2 = new int[N];
+ s1[0] = buy[0] = -prices[0];
+ sell[0] = s2[0] = 0;
+ for (int i = 1; i < N; i++) {
+ buy[i] = Math.max(sell[i - 1], s2[i - 1]) - prices[i];
+ s1[i] = Math.max(buy[i - 1], s1[i - 1]);
+ sell[i] = Math.max(buy[i - 1], s1[i - 1]) - fee + prices[i];
+ s2[i] = Math.max(s2[i - 1], sell[i - 1]);
}
- Arrays.sort(pairs, (a, b) -> (a[0] - b[0]));
- int n = pairs.length;
- int[] dp = new int[n];
- Arrays.fill(dp, 1);
- for(int i = 0; i < n; i++) {
- for(int j = 0; j < i; j++) {
- if(pairs[i][0] > pairs[j][1]){
- dp[i] = Math.max(dp[i], dp[j] + 1);
- }
- }
- }
-
- int ret = 0;
- for(int num : dp) {
- ret = Math.max(ret, num);
- }
- return ret;
+ return Math.max(sell[N - 1], s2[N - 1]);
}
```
@@ -2619,6 +2853,18 @@ public int maxProfit(int[] prices) {
[Leetcode : 650. 2 Keys Keyboard (Medium)](https://leetcode.com/problems/2-keys-keyboard/description/)
+题目描述:最开始只有一个字符 A,问需要多少次操作能够得到 n 个字符 A,每次操作可以复制当前所有的字符,或者粘贴。
+
+```
+Input: 3
+Output: 3
+Explanation:
+Intitally, we have one character 'A'.
+In step 1, we use Copy All operation.
+In step 2, we use Paste operation to get 'AA'.
+In step 3, we use Paste operation to get 'AAA'.
+```
+
```java
public int minSteps(int n) {
int[] dp = new int[n + 1];
@@ -2690,12 +2936,11 @@ public int countPrimes(int n) {
```java
int gcd(int a, int b) {
- if (b == 0) return a;
- return gcd(b, a % b);
+ return b == 0 ? a : gcd(b, a% b);
}
```
-最大公倍数为两数的乘积除以最大公约数。
+最小公倍数为两数的乘积除以最大公约数。
```java
int lcm(int a, int b){
@@ -2707,47 +2952,80 @@ int lcm(int a, int b){
对于 a 和 b 的最大公约数 f(a, b),有:
-1\. 如果 a 和 b 均为偶数,f(a, b) = 2\*f(a/2, b/2);
-2\. 如果 a 是偶数 b 是奇数,f(a, b) = f(a/2, b);
-3\. 如果 b 是偶数 a 是奇数,f(a, b) = f(a, b/2);
-4\. 如果 a 和 b 均为奇数,f(a, b) = f(a, a-b);
+- 如果 a 和 b 均为偶数,f(a, b) = 2\*f(a/2, b/2);
+- 如果 a 是偶数 b 是奇数,f(a, b) = f(a/2, b);
+- 如果 b 是偶数 a 是奇数,f(a, b) = f(a, b/2);
+- 如果 a 和 b 均为奇数,f(a, b) = f(a, a-b);
乘 2 和除 2 都可以转换为移位操作。
### 进制转换
-Java 中 static String toString(int num, int radix) 可以将一个整数装换为 redix 进制表示的字符串。
-
**7 进制**
[Leetcode : 504. Base 7 (Easy)](https://leetcode.com/problems/base-7/description/)
```java
public String convertToBase7(int num) {
- if (num < 0) {
- return '-' + convertToBase7(-num);
- }
- if (num < 7) {
- return num + "";
- }
+ if (num < 0) return '-' + convertToBase7(-num);
+ if (num < 7) return num + "";
return convertToBase7(num / 7) + num % 7;
}
```
+```java
+public String convertToBase7(int num) {
+ if (num == 0) return "0";
+ StringBuilder sb = new StringBuilder();
+ boolean isNegative = num < 0;
+ if (isNegative) num = -num;
+ while (num > 0) {
+ sb.append(num % 7);
+ num /= 7;
+ }
+ String ret = sb.reverse().toString();
+ return isNegative ? "-" + ret : ret;
+}
+```
+
+Java 中 static String toString(int num, int radix) 可以将一个整数转换为 redix 进制表示的字符串。
+
+```java
+public String convertToBase7(int num) {
+ return Integer.toString(num, 7);
+}
+```
+
**16 进制**
[Leetcode : 405. Convert a Number to Hexadecimal (Easy)](https://leetcode.com/problems/convert-a-number-to-hexadecimal/description/)
+负数要用它的补码形式。
+
+```html
+Input:
+26
+
+Output:
+"1a"
+
+Input:
+-1
+
+Output:
+"ffffffff"
+```
+
```java
public String toHex(int num) {
char[] map = {'0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f'};
if(num == 0) return "0";
- String ret = "";
+ StringBuilder sb = new StringBuilder();
while(num != 0){
- ret = map[(num & 0b1111)] + ret;
- num >>>= 4;
+ sb.append(map[num & 0b1111]);
+ num >>>= 4; // 无符号右移,左边填 0
}
- return ret;
+ return sb.reverse().toString();
}
```
@@ -2784,15 +3062,14 @@ Return "100".
```java
public String addBinary(String a, String b) {
int i = a.length() - 1, j = b.length() - 1, carry = 0;
- String str = "";
- while(i >= 0 || j >= 0){
- if(i >= 0 && a.charAt(i--) == '1') carry++;
- if(j >= 0 && b.charAt(j--) == '1') carry++;
- str = (carry % 2) + str;
+ StringBuilder str = new StringBuilder();
+ while (carry == 1 || i >= 0 || j >= 0) {
+ if (i >= 0 && a.charAt(i--) == '1') carry++;
+ if (j >= 0 && b.charAt(j--) == '1') carry++;
+ str.append(carry % 2);
carry /= 2;
}
- if(carry == 1) str = "1" + str;
- return str;
+ return str.reverse().toString();
}
```
@@ -2804,15 +3081,15 @@ public String addBinary(String a, String b) {
```java
public String addStrings(String num1, String num2) {
- StringBuilder sb = new StringBuilder();
- int carry = 0;
- for(int i = num1.length() - 1, j = num2.length() - 1; i >= 0 || j >= 0 || carry == 1; i--, j--){
- int x = i < 0 ? 0 : num1.charAt(i) - '0';
- int y = j < 0 ? 0 : num2.charAt(j) - '0';
- sb.append((x + y + carry) % 10);
+ StringBuilder str = new StringBuilder();
+ int carry = 0, i = num1.length() - 1, j = num2.length() - 1;
+ while (carry == 1 || i >= 0 || j >= 0) {
+ int x = i < 0 ? 0 : num1.charAt(i--) - '0';
+ int y = j < 0 ? 0 : num2.charAt(j--) - '0';
+ str.append((x + y + carry) % 10);
carry = (x + y + carry) / 10;
}
- return sb.reverse().toString();
+ return str.reverse().toString();
}
```
@@ -2845,7 +3122,7 @@ Only two moves are needed (remember each move increments or decrements one eleme
**解法 1**
-先排序,时间复杂度:O(NlgN)
+先排序,时间复杂度:O(NlogN)
```java
public int minMoves2(int[] nums) {
@@ -2863,7 +3140,7 @@ public int minMoves2(int[] nums) {
**解法 2**
-使用快速排序找到中位数,时间复杂度 O(N)
+使用快速选择找到中位数,时间复杂度 O(N)
```java
public int minMoves2(int[] nums) {
@@ -2910,7 +3187,7 @@ public int majorityElement(int[] nums) {
}
```
-可以利用 Boyer-Moore Majority Vote Algorithm 来解决这个问题,使得时间复杂度为 O(n)。可以这么理解该算法:使用 cnt 来统计一个元素出现的次数,当遍历到的元素和统计元素不想等时,令 cnt--。如果前面查找了 i 个元素,且 cnt == 0 ,说明前 i 个元素没有 majority,或者有 majority,但是出现的次数少于 i / 2 ,因为如果多于 i / 2 的话 cnt 就一定不会为 0 。此时剩下的 n - i 个元素中,majority 的数目依然多于 (n - i) / 2,因此继续查找就能找出 majority。
+可以利用 Boyer-Moore Majority Vote Algorithm 来解决这个问题,使得时间复杂度为 O(n)。可以这么理解该算法:使用 cnt 来统计一个元素出现的次数,当遍历到的元素和统计元素不相等时,令 cnt--。如果前面查找了 i 个元素,且 cnt == 0 ,说明前 i 个元素没有 majority,或者有 majority,但是出现的次数少于 i / 2 ,因为如果多于 i / 2 的话 cnt 就一定不会为 0 。此时剩下的 n - i 个元素中,majority 的数目依然多于 (n - i) / 2,因此继续查找就能找出 majority。
```java
public int majorityElement(int[] nums) {
@@ -3016,11 +3293,13 @@ public int[] productExceptSelf(int[] nums) {
int n = nums.length;
int[] ret = new int[n];
ret[0] = 1;
- for(int i = 1; i < n; i++) {
- ret[i] = ret[i - 1] * nums[i - 1];
+ int left = 1;
+ for (int i = 1; i < n; i++) {
+ ret[i] = left * nums[i - 1];
+ left *= nums[i - 1];
}
int right = 1;
- for(int i = n - 1; i >= 0; i--) {
+ for (int i = n - 1; i >= 0; i--) {
ret[i] *= right;
right *= nums[i];
}
@@ -3039,16 +3318,16 @@ public int[] productExceptSelf(int[] nums) {
一个栈实现:
```java
-class MyQueue {
+class MyQueue {
private Stack st = new Stack();
public void push(int x) {
Stack temp = new Stack();
- while(!st.isEmpty()){
+ while (!st.isEmpty()) {
temp.push(st.pop());
}
st.push(x);
- while(!temp.isEmpty()){
+ while (!temp.isEmpty()) {
st.push(temp.pop());
}
}
@@ -3070,7 +3349,7 @@ class MyQueue {
两个栈实现:
```java
-class MyQueue {
+class MyQueue {
private Stack in = new Stack();
private Stack out = new Stack();
@@ -3088,9 +3367,9 @@ class MyQueue {
return out.peek();
}
- private void in2out(){
- if(out.isEmpty()){
- while(!in.isEmpty()){
+ private void in2out() {
+ if (out.isEmpty()) {
+ while (!in.isEmpty()) {
out.push(in.pop());
}
}
@@ -3117,8 +3396,9 @@ class MyStack {
public void push(int x) {
queue.add(x);
- for(int i = 1; i < queue.size(); i++){ // 翻转
- queue.add(queue.remove());
+ int cnt = queue.size();
+ while (cnt-- > 1) {
+ queue.add(queue.poll());
}
}
@@ -3157,20 +3437,14 @@ class MinStack {
public void push(int x) {
dataStack.add(x);
- if(x < min) {
- min = x;
- }
+ min = Math.min(min, x);
minStack.add(min);
}
public void pop() {
dataStack.pop();
minStack.pop();
- if(!minStack.isEmpty()) {
- min = minStack.peek();
- } else{
- min = Integer.MAX_VALUE;
- }
+ min = minStack.isEmpty() ? min = Integer.MAX_VALUE : minStack.peek();
}
public int top() {
@@ -3198,17 +3472,15 @@ Output : true
```java
public boolean isValid(String s) {
Stack stack = new Stack<>();
- for(int i = 0; i < s.length(); i++){
- char c = s.charAt(i);
- if(c == '(' || c == '{' || c == '[') stack.push(c);
- else{
- if(stack.isEmpty()) return false;
+ for (char c : s.toCharArray()) {
+ if (c == '(' || c == '{' || c == '[') stack.push(c);
+ else {
+ if (stack.isEmpty()) return false;
char cStack = stack.pop();
- if(c == ')' && cStack != '(' ||
- c == ']' && cStack != '[' ||
- c == '}' && cStack != '{' ) {
- return false;
- }
+ boolean b1 = c == ')' && cStack != '(';
+ boolean b2 = c == ']' && cStack != '[';
+ boolean b3 = c == '}' && cStack != '{';
+ if (b1 || b2 || b3) return false;
}
}
return stack.isEmpty();
@@ -3307,9 +3579,9 @@ HashMap 也可以用来对元素进行计数统计,此时键为元素,值为
[Leetcode : 1. Two Sum (Easy)](https://leetcode.com/problems/two-sum/description/)
-可以先对数组进行排序,然后使用双指针方法或者二分查找方法。这样做的时间复杂度为 O(nlgn ),空间复杂度为 O(1)。
+可以先对数组进行排序,然后使用双指针方法或者二分查找方法。这样做的时间复杂度为 O(NlogN),空间复杂度为 O(1)。
-用 HashMap 存储数组元素和索引的映射,在访问到 nums[i] 时,判断 HashMap 中是否存在 target - nums[i] ,如果存在说明 target - nums[i] 所在的索引和 i 就是要找的两个数。该方法的时间复杂度为 O(n),空间复杂度为 O(n),使用空间来换取时间。
+用 HashMap 存储数组元素和索引的映射,在访问到 nums[i] 时,判断 HashMap 中是否存在 target - nums[i] ,如果存在说明 target - nums[i] 所在的索引和 i 就是要找的两个数。该方法的时间复杂度为 O(N),空间复杂度为 O(N),使用空间来换取时间。
```java
public int[] twoSum(int[] nums, int target) {
@@ -3322,6 +3594,18 @@ public int[] twoSum(int[] nums, int target) {
}
```
+**判断数组是否含有相同元素**
+
+[Leetcode : 217. Contains Duplicate (Easy)](https://leetcode.com/problems/contains-duplicate/description/)
+
+```java
+public boolean containsDuplicate(int[] nums) {
+ Set set = new HashSet<>();
+ for (int num : nums) set.add(num);
+ return set.size() < nums.length;
+}
+```
+
**最长和谐序列**
[Leetcode : 594. Longest Harmonious Subsequence (Easy)](https://leetcode.com/problems/longest-harmonious-subsequence/description/)
@@ -3350,6 +3634,45 @@ public int findLHS(int[] nums) {
}
```
+**最长连续序列**
+
+[Leetcode : 128. Longest Consecutive Sequence (Medium)](https://leetcode.com/problems/longest-consecutive-sequence/description/)
+
+```html
+Given [100, 4, 200, 1, 3, 2],
+The longest consecutive elements sequence is [1, 2, 3, 4]. Return its length: 4.
+```
+
+```java
+public int longestConsecutive(int[] nums) {
+ Map numCnts = new HashMap<>();
+ for (int num : nums) {
+ numCnts.put(num, 1);
+ }
+ for (int num : nums) {
+ count(numCnts, num);
+ }
+ int max = 0;
+ for (int num : nums) {
+ max = Math.max(max, numCnts.get(num));
+ }
+ return max;
+}
+
+private int count(Map numCnts, int num) {
+ if (!numCnts.containsKey(num)) {
+ return 0;
+ }
+ int cnt = numCnts.get(num);
+ if (cnt > 1) {
+ return cnt;
+ }
+ cnt = count(numCnts, num + 1) + 1;
+ numCnts.put(num, cnt);
+ return cnt;
+}
+```
+
## 字符串
**两个字符串包含的字符是否完全相同**
@@ -3455,7 +3778,7 @@ Output: 6
Explanation: Six palindromic strings: "a", "a", "a", "aa", "aa", "aaa".
```
-解决方案是从字符串的某一位开始,尝试着去扩展子字符串。
+从字符串的某一位开始,尝试着去扩展子字符串。
```java
private int cnt = 0;
@@ -3545,18 +3868,229 @@ For example, given nums = [0, 1, 0, 3, 12], after calling your function, nums sh
```java
public void moveZeroes(int[] nums) {
- int n = nums.length;
int idx = 0;
- for(int i = 0; i < n; i++){
- if(nums[i] != 0) nums[idx++] = nums[i];
- }
- while(idx < n){
- nums[idx++] = 0;
- }
+ for (int num : nums) if (num != 0) nums[idx++] = num;
+ while (idx < nums.length) nums[idx++] = 0;
+}
+```
+
+**调整矩阵**
+
+[Leetcode : 566. Reshape the Matrix (Easy)](https://leetcode.com/problems/reshape-the-matrix/description/)
+
+```html
+Input:
+nums =
+[[1,2],
+ [3,4]]
+r = 1, c = 4
+Output:
+[[1,2,3,4]]
+Explanation:
+The row-traversing of nums is [1,2,3,4]. The new reshaped matrix is a 1 * 4 matrix, fill it row by row by using the previous list.
+```
+
+```java
+public int[][] matrixReshape(int[][] nums, int r, int c) {
+ int m = nums.length, n = nums[0].length;
+ if (m * n != r * c) return nums;
+ int[][] ret = new int[r][c];
+ int index = 0;
+ for (int i = 0; i < r; i++) {
+ for (int j = 0; j < c; j++) {
+ ret[i][j] = nums[index / n][index % n];
+ index++;
+ }
+ }
+ return ret;
+}
+```
+
+**找出数组中最长的连续 1**
+
+[Leetcode : 485. Max Consecutive Ones (Easy)](https://leetcode.com/problems/max-consecutive-ones/description/)
+
+```java
+public int findMaxConsecutiveOnes(int[] nums) {
+ int max = 0;
+ int cur = 0;
+ for (int num : nums) {
+ if (num == 0) cur = 0;
+ else {
+ cur++;
+ max = Math.max(max, cur);
+ }
+ }
+ return max;
+}
+```
+
+**数组相邻差值的个数**
+
+[Leetcode : 667. Beautiful Arrangement II (Medium)](https://leetcode.com/problems/beautiful-arrangement-ii/description/)
+
+```html
+Input: n = 3, k = 2
+Output: [1, 3, 2]
+Explanation: The [1, 3, 2] has three different positive integers ranging from 1 to 3, and the [2, 1] has exactly 2 distinct integers: 1 and 2.
+```
+
+题目描述:数组元素为 1\~n 的整数,要求构建数组,使得相邻元素的差值不相同的个数为 k。
+
+让前 k+1 个元素构建出 k 个不相同的差值,序列为:1 k+1 2 k 3 k-1 ... k/2 k/2+1.
+
+```java
+public int[] constructArray(int n, int k) {
+ int[] ret = new int[n];
+ ret[0] = 1;
+ for (int i = 1, interval = k; i <= k; i++, interval--) {
+ ret[i] = i % 2 == 1 ? ret[i - 1] + interval : ret[i - 1] - interval;
+ }
+ for (int i = k + 1; i < n; i++) {
+ ret[i] = i + 1;
+ }
+ return ret;
+}
+```
+
+**数组的度**
+
+[Leetcode : 697. Degree of an Array (Easy)](https://leetcode.com/problems/degree-of-an-array/description/)
+
+```html
+Input: [1,2,2,3,1,4,2]
+Output: 6
+```
+
+题目描述:数组的度定义为元素出现的最高频率,例如上面的数组度为 3。要求找到一个最小的子数组,这个子数组的度和原数组一样。
+
+```java
+public int findShortestSubArray(int[] nums) {
+ Map numsCnt = new HashMap<>();
+ Map numsLastIndex = new HashMap<>();
+ Map numsFirstIndex = new HashMap<>();
+ for (int i = 0; i < nums.length; i++) {
+ int num = nums[i];
+ numsCnt.put(num, numsCnt.getOrDefault(num, 0) + 1);
+ numsLastIndex.put(num, i);
+ if (!numsFirstIndex.containsKey(num)) {
+ numsFirstIndex.put(num, i);
+ }
+ }
+ int maxCnt = 0;
+ for (int num : nums) {
+ maxCnt = Math.max(maxCnt, numsCnt.get(num));
+ }
+ int ret = nums.length;
+ for (int i = 0; i < nums.length; i++) {
+ int num = nums[i];
+ int cnt = numsCnt.get(num);
+ if (cnt != maxCnt) continue;
+ ret = Math.min(ret, numsLastIndex.get(num) - numsFirstIndex.get(num) + 1);
+ }
+ return ret;
+}
+```
+
+**对角元素相等的矩阵**
+
+[Leetcode : 766. Toeplitz Matrix (Easy)](https://leetcode.com/problems/toeplitz-matrix/description/)
+
+```html
+1234
+5123
+9512
+
+In the above grid, the diagonals are "[9]", "[5, 5]", "[1, 1, 1]", "[2, 2, 2]", "[3, 3]", "[4]", and in each diagonal all elements are the same, so the answer is True.
+```
+
+```java
+public boolean isToeplitzMatrix(int[][] matrix) {
+ for (int i = 0; i < matrix[0].length; i++) {
+ if (!check(matrix, matrix[0][i], 0, i)) {
+ return false;
+ }
+ }
+ for (int i = 0; i < matrix.length; i++) {
+ if (!check(matrix, matrix[i][0], i, 0)) {
+ return false;
+ }
+ }
+ return true;
+}
+
+private boolean check(int[][] matrix, int expectValue, int row, int col) {
+ if (row >= matrix.length || col >= matrix[0].length) {
+ return true;
+ }
+ if (matrix[row][col] != expectValue) {
+ return false;
+ }
+ return check(matrix, expectValue, row + 1, col + 1);
+}
+```
+
+**嵌套数组**
+
+[Leetcode : 565. Array Nesting (Medium)](https://leetcode.com/problems/array-nesting/description/)
+
+```html
+Input: A = [5,4,0,3,1,6,2]
+Output: 4
+Explanation:
+A[0] = 5, A[1] = 4, A[2] = 0, A[3] = 3, A[4] = 1, A[5] = 6, A[6] = 2.
+
+One of the longest S[K]:
+S[0] = {A[0], A[5], A[6], A[2]} = {5, 6, 2, 0}
+```
+
+题目描述:S[i] 表示一个集合,集合的第一个元素是 A[i],第二个元素是 A[A[i]],如此嵌套下去。求最大的 S[i]。
+
+```java
+public int arrayNesting(int[] nums) {
+ int max = 0;
+ for (int i = 0; i < nums.length; i++) {
+ int cnt = 0;
+ for (int j = i; nums[j] != -1; ) {
+ cnt++;
+ int t = nums[j];
+ nums[j] = -1; // 标记该位置已经被访问
+ j = t;
+
+ }
+ max = Math.max(max, cnt);
+ }
+ return max;
+}
+```
+
+**分隔数组**
+
+[Leetcode : 769. Max Chunks To Make Sorted (Medium)](https://leetcode.com/problems/max-chunks-to-make-sorted/description/)
+
+```html
+Input: arr = [1,0,2,3,4]
+Output: 4
+Explanation:
+We can split into two chunks, such as [1, 0], [2, 3, 4].
+However, splitting into [1, 0], [2], [3], [4] is the highest number of chunks possible.
+```
+
+题目描述:分隔数组,使得对每部分排序后数组就为有序。
+
+```java
+public int maxChunksToSorted(int[] arr) {
+ if (arr == null) return 0;
+ int ret = 0;
+ int right = arr[0];
+ for (int i = 0; i < arr.length; i++) {
+ right = Math.max(right, arr[i]);
+ if (right == i) ret++;
+ }
+ return ret;
}
```
-### 1-n 分布
**一个数组元素在 [1, n] 之间,其中一个数被替换为另一个数,找出丢失的数和重复的数**
@@ -3567,9 +4101,14 @@ Input: nums = [1,2,2,4]
Output: [2,3]
```
-最直接的方法是先对数组进行排序,这种方法时间复杂度为 O(nlogn ).本题可以以 O(n) 的时间复杂度、O(1) 空间复杂度来求解。
+```html
+Input: nums = [1,2,2,4]
+Output: [2,3]
+```
-主要思想是让通过交换数组元素,使得数组上的元素在正确的位置上
+最直接的方法是先对数组进行排序,这种方法时间复杂度为 O(nlogn)。本题可以以 O(n) 的时间复杂度、O(1) 空间复杂度来求解。
+
+主要思想是通过交换数组元素,使得数组上的元素在正确的位置上。
遍历数组,如果第 i 位上的元素不是 i + 1 ,那么就交换第 i 位 和 nums[i] - 1 位上的元素,使得 num[i] - 1 的元素为 nums[i] ,也就是该位的元素是正确的。交换操作需要循环进行,因为一次交换没办法使得第 i 位上的元素是正确的。但是要交换的两个元素可能就是重复元素,那么循环就可能永远进行下去,终止循环的方法是加上 nums[i] != nums[nums[i] - 1 条件。
@@ -3580,32 +4119,29 @@ Output: [2,3]
```java
public int[] findErrorNums(int[] nums) {
- for(int i = 0; i < nums.length; i++){
- while(nums[i] != i + 1 && nums[i] != nums[nums[i] - 1]) {
+ for (int i = 0; i < nums.length; i++) {
+ while (nums[i] != i + 1) {
+ if (nums[i] == nums[nums[i] - 1]) {
+ return new int[]{nums[nums[i] - 1], i + 1};
+ }
swap(nums, i, nums[i] - 1);
}
}
- for(int i = 0; i < nums.length; i++){
- if(i + 1 != nums[i]) {
- return new int[]{nums[i], i + 1};
- }
- }
-
return null;
}
-private void swap(int[] nums, int i, int j){
- int tmp = nums[i];
- nums[i] = nums[j];
- nums[j] = tmp;
+private void swap(int[] nums, int i, int j) {
+ int tmp = nums[i]; nums[i] = nums[j]; nums[j] = tmp;
}
```
-**找出数组中重复的数,数组值在 [0, n-1] 之间**
+**找出数组中重复的数,数组值在 [1, n] 之间**
[Leetcode : 287. Find the Duplicate Number (Medium)](https://leetcode.com/problems/find-the-duplicate-number/description/)
+要求不能修改数组,也不能使用额外的空间。
+
二分查找解法:
```java
@@ -3628,24 +4164,23 @@ public int findDuplicate(int[] nums) {
```java
public int findDuplicate(int[] nums) {
- int slow = nums[0], fast = nums[nums[0]];
- while (slow != fast) {
- slow = nums[slow];
- fast = nums[nums[fast]];
- }
-
- fast = 0;
- while (slow != fast) {
- slow = nums[slow];
- fast = nums[fast];
- }
- return slow;
+ int slow = nums[0], fast = nums[nums[0]];
+ while (slow != fast) {
+ slow = nums[slow];
+ fast = nums[nums[fast]];
+ }
+ fast = 0;
+ while (slow != fast) {
+ slow = nums[slow];
+ fast = nums[fast];
+ }
+ return slow;
}
```
-### 有序矩阵
+**有序矩阵查找**
-有序矩阵指的是行和列分别有序的矩阵。一般可以利用有序性使用二分查找方法。
+[Leetocde : 240. Search a 2D Matrix II (Medium)](https://leetcode.com/problems/search-a-2d-matrix-ii/description/)
```html
[
@@ -3655,10 +4190,6 @@ public int findDuplicate(int[] nums) {
]
```
-**有序矩阵查找**
-
-[Leetocde : 240. Search a 2D Matrix II (Medium)](https://leetcode.com/problems/search-a-2d-matrix-ii/description/)
-
```java
public boolean searchMatrix(int[][] matrix, int target) {
if (matrix == null || matrix.length == 0 || matrix[0].length == 0) return false;
@@ -3741,7 +4272,7 @@ class Tuple implements Comparable {
## 链表
-**判断两个链表的交点**
+**找出两个链表的交点**
[Leetcode : 160. Intersection of Two Linked Lists (Easy)](https://leetcode.com/problems/intersection-of-two-linked-lists/description/)
@@ -3818,6 +4349,11 @@ public ListNode mergeTwoLists(ListNode l1, ListNode l2) {
[Leetcode : 83. Remove Duplicates from Sorted List (Easy)](https://leetcode.com/problems/remove-duplicates-from-sorted-list/description/)
+```html
+Given 1->1->2, return 1->2.
+Given 1->1->2->3->3, return 1->2->3.
+```
+
```java
public ListNode deleteDuplicates(ListNode head) {
if(head == null || head.next == null) return head;
@@ -3826,22 +4362,203 @@ public ListNode deleteDuplicates(ListNode head) {
}
```
+**删除链表的倒数第 n 个节点**
+
+[Leetcode : 19. Remove Nth Node From End of List (Medium)](https://leetcode.com/problems/remove-nth-node-from-end-of-list/description/)
+
+```html
+Given linked list: 1->2->3->4->5, and n = 2.
+After removing the second node from the end, the linked list becomes 1->2->3->5.
+```
+
+```java
+public ListNode removeNthFromEnd(ListNode head, int n) {
+ ListNode newHead = new ListNode(-1);
+ newHead.next = head;
+ ListNode fast = newHead;
+ while (n-- > 0) {
+ fast = fast.next;
+ }
+ ListNode slow = newHead;
+ while (fast.next != null) {
+ fast = fast.next;
+ slow = slow.next;
+ }
+ slow.next = slow.next.next;
+ return newHead.next;
+}
+```
+
+**交换链表中的相邻结点**
+
+[Leetcode : 24. Swap Nodes in Pairs (Medium)](https://leetcode.com/problems/swap-nodes-in-pairs/description/)
+
+```html
+Given 1->2->3->4, you should return the list as 2->1->4->3.
+```
+
+题目要求:不能修改结点的 val 值;O(1) 空间复杂度。
+
+```java
+public ListNode swapPairs(ListNode head) {
+ ListNode newHead = new ListNode(-1);
+ newHead.next = head;
+ ListNode cur = head, pre = newHead;
+ while (cur != null && cur.next != null) {
+ ListNode next = cur.next;
+ pre.next = next;
+ cur.next = next.next;
+ next.next = cur;
+ pre = cur;
+ cur = cur.next;
+ }
+ return newHead.next;
+}
+```
+
+**根据有序链表构造平衡的 BST**
+
+[Leetcode : 109. Convert Sorted List to Binary Search Tree (Medium)](https://leetcode.com/problems/convert-sorted-list-to-binary-search-tree/description/)
+
+```html
+Given the sorted linked list: [-10,-3,0,5,9],
+
+One possible answer is: [0,-3,9,-10,null,5], which represents the following height balanced BST:
+
+ 0
+ / \
+ -3 9
+ / /
+ -10 5
+```
+
+```java
+public TreeNode sortedListToBST(ListNode head) {
+ if (head == null) return null;
+ int size = size(head);
+ if (size == 1) return new TreeNode(head.val);
+ ListNode pre = head, mid = pre.next;
+ int step = 2;
+ while (step <= size / 2) {
+ pre = mid;
+ mid = mid.next;
+ step++;
+ }
+ pre.next = null;
+ TreeNode t = new TreeNode(mid.val);
+ t.left = sortedListToBST(head);
+ t.right = sortedListToBST(mid.next);
+ return t;
+}
+
+private int size(ListNode node) {
+ int size = 0;
+ while (node != null) {
+ size++;
+ node = node.next;
+ }
+ return size;
+}
+```
+
+
+**链表求和**
+
+[Leetcode : 445. Add Two Numbers II (Medium)](https://leetcode.com/problems/add-two-numbers-ii/description/)
+
+```html
+Input: (7 -> 2 -> 4 -> 3) + (5 -> 6 -> 4)
+Output: 7 -> 8 -> 0 -> 7
+```
+
+题目要求:不能修改原始链表。
+
+```java
+public ListNode addTwoNumbers(ListNode l1, ListNode l2) {
+ Stack l1Stack = buildStack(l1);
+ Stack l2Stack = buildStack(l2);
+ ListNode head = new ListNode(-1);
+ int carry = 0;
+ while (!l1Stack.isEmpty() || !l2Stack.isEmpty() || carry != 0) {
+ int x = l1Stack.isEmpty() ? 0 : l1Stack.pop();
+ int y = l2Stack.isEmpty() ? 0 : l2Stack.pop();
+ int sum = x + y + carry;
+ ListNode node = new ListNode(sum % 10);
+ node.next = head.next;
+ head.next = node;
+ carry = sum / 10;
+ }
+ return head.next;
+}
+
+private Stack buildStack(ListNode l) {
+ Stack stack = new Stack<>();
+ while (l != null) {
+ stack.push(l.val);
+ l = l.next;
+ }
+ return stack;
+}
+```
+
+**分隔链表**
+
+[Leetcode : 725. Split Linked List in Parts(Medium)](https://leetcode.com/problems/split-linked-list-in-parts/description/)
+
+```html
+Input:
+root = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10], k = 3
+Output: [[1, 2, 3, 4], [5, 6, 7], [8, 9, 10]]
+Explanation:
+The input has been split into consecutive parts with size difference at most 1, and earlier parts are a larger size than the later parts.
+```
+
+题目描述:把链表分隔成 k 部分,每部分的长度都应该尽可能相同,排在前面的长度应该大于等于后面的。
+
+```java
+public ListNode[] splitListToParts(ListNode root, int k) {
+ int N = 0;
+ ListNode cur = root;
+ while (cur != null) {
+ N++;
+ cur = cur.next;
+ }
+ int mod = N % k;
+ int size = N / k;
+ ListNode[] ret = new ListNode[k];
+ cur = root;
+ for (int i = 0; cur != null && i < k; i++) {
+ ret[i] = cur;
+ int curSize = size + (mod-- > 0 ? 1 : 0);
+ for (int j = 0; j < curSize - 1; j++) {
+ cur = cur.next;
+ }
+ ListNode next = cur.next;
+ cur.next = null;
+ cur = next;
+ }
+ return ret;
+}
+```
+
**回文链表**
[Leetcode : 234. Palindrome Linked List (Easy)](https://leetcode.com/problems/palindrome-linked-list/description/)
+要求以 O(1) 的空间复杂度来求解。
+
切成两半,把后半段反转,然后比较两半是否相等。
```java
public boolean isPalindrome(ListNode head) {
- if(head == null || head.next == null) return true;
+ if (head == null || head.next == null) return true;
ListNode slow = head, fast = head.next;
- while(fast != null && fast.next != null){
+ while (fast != null && fast.next != null) {
slow = slow.next;
fast = fast.next.next;
}
- if(fast != null){ // 偶数节点,让 slow 指向下一个节点
+ if (fast != null) { // 偶数节点,让 slow 指向下一个节点
slow = slow.next;
}
@@ -3851,14 +4568,14 @@ public boolean isPalindrome(ListNode head) {
return isEqual(l1, l2);
}
-private void cut(ListNode head, ListNode cutNode){
- while( head.next != cutNode ) head = head.next;
+private void cut(ListNode head, ListNode cutNode) {
+ while (head.next != cutNode) head = head.next;
head.next = null;
}
-private ListNode reverse(ListNode head){
+private ListNode reverse(ListNode head) {
ListNode newHead = null;
- while(head != null){
+ while (head != null) {
ListNode nextNode = head.next;
head.next = newHead;
newHead = head;
@@ -3867,9 +4584,9 @@ private ListNode reverse(ListNode head){
return newHead;
}
-private boolean isEqual(ListNode l1, ListNode l2){
- while(l1 != null && l2 != null){
- if(l1.val != l2.val) return false;
+private boolean isEqual(ListNode l1, ListNode l2) {
+ while (l1 != null && l2 != null) {
+ if (l1.val != l2.val) return false;
l1 = l1.next;
l2 = l2.next;
}
@@ -4016,7 +4733,7 @@ Return 3. The paths that sum to 8 are:
3. -3 -> 11
```
-路径不一定以 root 开头并以 leaf 结尾,但是必须连续
+路径不一定以 root 开头,也不一定以 leaf 结尾,但是必须连续。
```java
public int pathSum(TreeNode root, int sum) {
@@ -4123,19 +4840,13 @@ There are two left leaves in the binary tree, with values 9 and 15 respectively.
```java
public int sumOfLeftLeaves(TreeNode root) {
- if(root == null) {
- return 0;
- }
- if(isLeaf(root.left)) {
- return root.left.val + sumOfLeftLeaves(root.right);
- }
+ if(root == null) return 0;
+ if(isLeaf(root.left)) return root.left.val + sumOfLeftLeaves(root.right);
return sumOfLeftLeaves(root.left) + sumOfLeftLeaves(root.right);
}
private boolean isLeaf(TreeNode node){
- if(node == null) {
- return false;
- }
+ if(node == null) return false;
return node.left == null && node.right == null;
}
```
@@ -4192,7 +4903,7 @@ Given tree s:
/ \
1 2
Given tree t:
- 4
+ 4
/ \
1 2
Return true, because t has the same structure and node values with a subtree of s.
@@ -4235,9 +4946,10 @@ private TreeNode toBST(int[] nums, int sIdx, int eIdx){
**两节点的最长路径**
+[Leetcode : 543. Diameter of Binary Tree (Easy)](https://leetcode.com/problems/diameter-of-binary-tree/description/)
+
```html
Input:
-
1
/ \
2 3
@@ -4247,8 +4959,6 @@ Input:
Return 3, which is the length of the path [4,2,1,3] or [5,2,1,3].
```
-[Leetcode : 543. Diameter of Binary Tree (Easy)](https://leetcode.com/problems/diameter-of-binary-tree/description/)
-
```java
private int max = 0;
@@ -4468,11 +5178,11 @@ public int findBottomLeftValue(TreeNode root) {
### 前中后序遍历
```html
- 1
- / \
- 2 3
- / \ \
-4 5 6
+ 1
+ / \
+ 2 3
+ / \ \
+4 5 6
```
层次遍历顺序:[1 2 3 4 5 6]
@@ -4521,14 +5231,14 @@ void dfs(TreeNode root){
```java
public List preorderTraversal(TreeNode root) {
List ret = new ArrayList<>();
- if (root == null) return ret;
Stack stack = new Stack<>();
stack.push(root);
while (!stack.isEmpty()) {
TreeNode node = stack.pop();
+ if (node == null) continue;
ret.add(node.val);
- if (node.right != null) stack.push(node.right);
- if (node.left != null) stack.push(node.left); // 先添加右子树再添加左子树,这样是为了让左子树在栈顶
+ stack.push(node.right); // 先右后左,保证左子树先遍历
+ stack.push(node.left);
}
return ret;
}
@@ -4543,14 +5253,14 @@ public List preorderTraversal(TreeNode root) {
```java
public List postorderTraversal(TreeNode root) {
List ret = new ArrayList<>();
- if (root == null) return ret;
Stack stack = new Stack<>();
stack.push(root);
while (!stack.isEmpty()) {
TreeNode node = stack.pop();
+ if (node == null) continue;
ret.add(node.val);
- if (node.left != null) stack.push(node.left);
- if (node.right != null) stack.push(node.right);
+ stack.push(node.left);
+ stack.push(node.right);
}
Collections.reverse(ret);
return ret;
@@ -4564,11 +5274,12 @@ public List postorderTraversal(TreeNode root) {
```java
public List inorderTraversal(TreeNode root) {
List ret = new ArrayList<>();
+ if (root == null) return ret;
Stack stack = new Stack<>();
TreeNode cur = root;
- while(cur != null || !stack.isEmpty()) {
- while(cur != null) { // 模拟递归栈的不断深入
- stack.add(cur);
+ while (cur != null || !stack.isEmpty()) {
+ while (cur != null) {
+ stack.push(cur);
cur = cur.left;
}
TreeNode node = stack.pop();
@@ -4588,7 +5299,7 @@ public List inorderTraversal(TreeNode root) {
[653. Two Sum IV - Input is a BST (Easy)](https://leetcode.com/problems/two-sum-iv-input-is-a-bst/description/)
```html
-Input:
+Input:
5
/ \
3 6
@@ -4689,17 +5400,11 @@ public TreeNode convertBST(TreeNode root) {
}
private void traver(TreeNode root) {
- if (root == null) {
- return;
- }
- if (root.right != null) {
- traver(root.right);
- }
+ if (root == null) return;
+ if (root.right != null) traver(root.right);
sum += root.val;
root.val = sum;
- if (root.left != null) {
- traver(root.left);
- }
+ if (root.left != null) traver(root.left);
}
```
@@ -4724,31 +5429,31 @@ private List list;
public int[] findMode(TreeNode root) {
list = new ArrayList<>();
- inorder(root);
+ inOrder(root);
int[] ret = new int[list.size()];
int idx = 0;
- for(int num : list){
+ for (int num : list) {
ret[idx++] = num;
}
return ret;
}
-private void inorder(TreeNode node){
- if(node == null) return;
- inorder(node.left);
- if(preNode != null){
- if(preNode.val == node.val) cnt++;
+private void inOrder(TreeNode node) {
+ if (node == null) return;
+ inOrder(node.left);
+ if (preNode != null) {
+ if (preNode.val == node.val) cnt++;
else cnt = 1;
}
- if(cnt > maxCnt){
+ if (cnt > maxCnt) {
maxCnt = cnt;
list.clear();
list.add(node.val);
- } else if(cnt == maxCnt){
+ } else if (cnt == maxCnt) {
list.add(node.val);
}
preNode = node;
- inorder(node.right);
+ inOrder(node.right);
}
```
@@ -4779,19 +5484,19 @@ private int cnt = 0;
private int val;
public int kthSmallest(TreeNode root, int k) {
- inorder(root, k);
+ inOrder(root, k);
return val;
}
-private void inorder(TreeNode node, int k) {
- if(node == null) return;
- inorder(node.left, k);
+private void inOrder(TreeNode node, int k) {
+ if (node == null) return;
+ inOrder(node.left, k);
cnt++;
- if(cnt == k) {
+ if (cnt == k) {
val = node.val;
return;
}
- inorder(node.right, k);
+ inOrder(node.right, k);
}
```
@@ -4808,33 +5513,24 @@ Trie,又称前缀树或字典树,用于判断字符串是否存在或者是
```java
class Trie {
- private class Node{
- Node[] childs = new Node[26];
- boolean isLeaf;
- }
-
private Node root = new Node();
- /** Initialize your data structure here. */
public Trie() {
}
- /** Inserts a word into the trie. */
public void insert(String word) {
- int idx = word.charAt(0) - 'a';
insert(word, root);
}
private void insert(String word, Node node){
int idx = word.charAt(0) - 'a';
- if(node.childs[idx] == null){
- node.childs[idx] = new Node();
+ if(node.child[idx] == null){
+ node.child[idx] = new Node();
}
- if(word.length() == 1) node.childs[idx].isLeaf = true;
- else insert(word.substring(1), node.childs[idx]);
+ if(word.length() == 1) node.child[idx].isLeaf = true;
+ else insert(word.substring(1), node.child[idx]);
}
- /** Returns if the word is in the trie. */
public boolean search(String word) {
return search(word, root);
}
@@ -4842,12 +5538,11 @@ class Trie {
private boolean search(String word, Node node){
if(node == null) return false;
int idx = word.charAt(0) - 'a';
- if(node.childs[idx] == null) return false;
- if(word.length() == 1) return node.childs[idx].isLeaf;
- return search(word.substring(1), node.childs[idx]);
+ if(node.child[idx] == null) return false;
+ if(word.length() == 1) return node.child[idx].isLeaf;
+ return search(word.substring(1), node.child[idx]);
}
- /** Returns if there is any word in the trie that starts with the given prefix. */
public boolean startsWith(String prefix) {
return startWith(prefix, root);
}
@@ -4856,7 +5551,12 @@ class Trie {
if(node == null) return false;
if(prefix.length() == 0) return true;
int idx = prefix.charAt(0) - 'a';
- return startWith(prefix.substring(1), node.childs[idx]);
+ return startWith(prefix.substring(1), node.child[idx]);
+ }
+
+ private class Node{
+ Node[] child = new Node[26];
+ boolean isLeaf;
}
}
```
@@ -4874,51 +5574,50 @@ Input: sum("ap"), Output: 5
```java
class MapSum {
- private class Trie {
- int val;
- Map childs;
- boolean isWord;
- Trie() {
- childs = new HashMap<>();
- }
+ private class Node {
+ Node[] child = new Node[26];
+ int value;
}
- private Trie root;
+ private Node root = new Node();
public MapSum() {
- root = new Trie();
+
}
public void insert(String key, int val) {
- Trie cur = root;
- for(char c : key.toCharArray()) {
- if(!cur.childs.containsKey(c)) {
- Trie next = new Trie();
- cur.childs.put(c, next);
- }
- cur = cur.childs.get(c);
+ insert(key, root, val);
+ }
+
+ private void insert(String key, Node node, int val) {
+ int idx = key.charAt(0) - 'a';
+ if (node.child[idx] == null) {
+ node.child[idx] = new Node();
+ }
+ if (key.length() == 1) {
+ node.child[idx].value = val;
+ } else {
+ insert(key.substring(1), node.child[idx], val);
}
- cur.val = val;
- cur.isWord = true;
}
public int sum(String prefix) {
- Trie cur = root;
- for(char c : prefix.toCharArray()) {
- if(!cur.childs.containsKey(c)) return 0;
- cur = cur.childs.get(c);
- }
- return dfs(cur);
+ return sum(prefix, root);
}
- private int dfs(Trie cur) {
- int sum = 0;
- if(cur.isWord) {
- sum += cur.val;
+ private int sum(String prefix, Node node) {
+ if (node == null) {
+ return 0;
}
- for(Trie next : cur.childs.values()) {
- sum += dfs(next);
+ int sum = node.value;
+ if (prefix.length() == 0) {
+ for (Node next : node.child) {
+ sum += sum(prefix, next);
+ }
+ } else {
+ int idx = prefix.charAt(0) - 'a';
+ sum = sum(prefix.substring(1), node.child[idx]);
}
return sum;
}
@@ -4927,11 +5626,71 @@ class MapSum {
## 图
+**冗余连接**
+
+[Leetcode : 684. Redundant Connection (Medium)](https://leetcode.com/problems/redundant-connection/description/)
+
+```html
+Input: [[1,2], [1,3], [2,3]]
+Output: [2,3]
+Explanation: The given undirected graph will be like this:
+ 1
+ / \
+2 - 3
+```
+
+题目描述:有一系列的边连成的图,找出一条边,移除它之后该图能够成为一棵树。
+
+使用 Union-Find。
+
+```java
+public int[] findRedundantConnection(int[][] edges) {
+ int N = edges.length;
+ UF uf = new UF(N);
+ for (int[] e : edges) {
+ int u = e[0], v = e[1];
+ if (uf.find(u) == uf.find(v)) {
+ return e;
+ }
+ uf.union(u, v);
+ }
+ return new int[]{-1, -1};
+}
+
+private class UF {
+ int[] id;
+
+ UF(int N) {
+ id = new int[N + 1];
+ for (int i = 0; i < id.length; i++) {
+ id[i] = i;
+ }
+ }
+
+ void union(int u, int v) {
+ int uID = find(u);
+ int vID = find(v);
+ if (uID == vID) {
+ return;
+ }
+ for (int i = 0; i < id.length; i++) {
+ if (id[i] == uID) {
+ id[i] = vID;
+ }
+ }
+ }
+
+ int find(int p) {
+ return id[p];
+ }
+}
+```
+
## 位运算
**1. 基本原理**
-0s 表示 一串 0 ,1s 表示一串 1。
+0s 表示一串 0,1s 表示一串 1。
```
x ^ 0s = x x & 0s = 0 x | 0s = x
@@ -4939,9 +5698,9 @@ x ^ 1s = ~x x & 1s = x x | 1s = 1s
x ^ x = 0 x & x = x x | x = x
```
-① 利用 x ^ 1s = \~x 的特点,可以将位级表示翻转;利用 x ^ x = 0 的特点,可以将三个数中重复的两个数去除,只留下另一个数;
-② 利用 x & 0s = 0 和 x & 1s = x 的特点,可以实现掩码操作。一个数 num 与 mask :00111100 进行位与操作,只保留 num 中与 mask 的 1 部分相对应的位;
-③ 利用 x | 0s = x 和 x | 1s = 1s 的特点,可以实现设置操作。一个数 num 与 mask:00111100 进行位或操作,将 num 中与 mask 的 1 部分相对应的位都设置为 1 。
+- 利用 x ^ 1s = \~x 的特点,可以将位级表示翻转;利用 x ^ x = 0 的特点,可以将三个数中重复的两个数去除,只留下另一个数;
+- 利用 x & 0s = 0 和 x & 1s = x 的特点,可以实现掩码操作。一个数 num 与 mask :00111100 进行位与操作,只保留 num 中与 mask 的 1 部分相对应的位;
+- 利用 x | 0s = x 和 x | 1s = 1s 的特点,可以实现设值操作。一个数 num 与 mask:00111100 进行位或操作,将 num 中与 mask 的 1 部分相对应的位都设置为 1 。
\>\> n 为算术右移,相当于除以 2n ;
\>\>\> n 为无符号右移,左边会补上 0。
@@ -4949,7 +5708,7 @@ x ^ x = 0 x & x = x x | x = x
n&(n-1) 该位运算是去除 n 的位级表示中最低的那一位。例如对于二进制表示 10110 **100** ,减去 1 得到 10110**011**,这两个数相与得到 10110**000**。
-n-n&(\~n+1) 概运算是去除 n 的位级表示中最高的那一位。
+n-n&(\~n+1) 运算是去除 n 的位级表示中最高的那一位。
n&(-n) 该运算得到 n 的位级表示中最低的那一位。-n 得到 n 的反码加 1,对于二进制表示 10110 **100** ,-n 得到 01001**100**,相与得到 00000**100**
@@ -4957,7 +5716,7 @@ n&(-n) 该运算得到 n 的位级表示中最低的那一位。-n 得到 n 的
要获取 111111111,将 0 取反即可,\~0。
-要得到只有第 i 位为 1 的 mask,将 1 向左移动 i 位即可,1<<i 。例如 1<<5 得到只有第 5 位为 1 的 mask :00010000。
+要得到只有第 i 位为 1 的 mask,将 1 向左移动 i-1 位即可,1<<(i-1) 。例如 1<<4 得到只有第 5 位为 1 的 mask :00010000。
要得到 1 到 i 位为 1 的 mask,1<<(i+1)-1 即可,例如将 1<<(4+1)-1 = 00010000-1 = 00001111。
@@ -5016,16 +5775,16 @@ num & (~((1 << (i+1)) - 1));
**4. Java 中的位操作**
```html
-static int Integer.bitCount() // 统计 1 的数量
-static int Integer.highestOneBit() // 获得最高位
-static String toBinaryString(int i) // 转换位二进制表示的字符串
+static int Integer.bitCount(); // 统计 1 的数量
+static int Integer.highestOneBit(); // 获得最高位
+static String toBinaryString(int i); // 转换为二进制表示的字符串
```
**统计两个数的二进制表示有多少位不同**
[Leetcode : 461. Hamming Distance (Easy)](https://leetcode.com/problems/hamming-distance/)
-对两个数进行异或操作,不同的那一位结果为 1 ,统计有多少个 1 即可。
+对两个数进行异或操作,位级表示不同的那一位为 1,统计有多少个 1 即可。
```java
public int hammingDistance(int x, int y) {
@@ -5047,6 +5806,28 @@ public int hammingDistance(int x, int y) {
}
```
+**找出数组中缺失的那个数**
+
+[Leetcode : 268. Missing Number (Easy)](https://leetcode.com/problems/missing-number/description/)
+
+```html
+Input: [3,0,1]
+Output: 2
+```
+
+题目描述:数组元素在 0-n 之间,但是有一个数是缺失的,要求找到这个缺失的数。
+ `
+
+```java
+public int missingNumber(int[] nums) {
+ int ret = 0;
+ for (int i = 0; i <= nums.length; i++) {
+ ret = i == nums.length ? ret ^ i : ret ^ i ^ nums[i];
+ }
+ return ret;
+}
+```
+
**翻转一个数的比特位**
[Leetcode : 190. Reverse Bits (Easy)](https://leetcode.com/problems/reverse-bits/description/)
@@ -5054,7 +5835,7 @@ public int hammingDistance(int x, int y) {
```java
public int reverseBits(int n) {
int ret = 0;
- for(int i = 0; i < 32; i++){
+ for (int i = 0; i < 32; i++) {
ret <<= 1;
ret |= (n & 1);
n >>>= 1;
@@ -5063,6 +5844,37 @@ public int reverseBits(int n) {
}
```
+如果该函数需要被调用很多次,可以将 int 拆成 4 个 byte,然后缓存 byte 对应的比特位翻转,最后再拼接起来。
+
+```java
+private static Map cache = new HashMap<>();
+
+public int reverseBits(int n) {
+ int ret = 0;
+ for (int i = 0; i < 4; i++) {
+ byte b = (byte) (n & 0b11111111);
+ ret <<= 8;
+ ret |= reverseByte(b);
+ n >>= 8;
+ }
+ return ret;
+
+}
+
+private int reverseByte(byte b) {
+ if (cache.containsKey(b)) return cache.get(b);
+ int ret = 0;
+ byte t = b;
+ for (int i = 0; i < 8; i++) {
+ ret <<= 1;
+ ret |= t & 1;
+ t >>= 1;
+ }
+ cache.put(b, ret);
+ return ret;
+}
+```
+
**不用额外变量交换两个整数**
[程序员代码面试指南 :P317](#)
@@ -5073,38 +5885,13 @@ b = a ^ b;
a = a ^ b;
```
-将 c = a ^ b,那么 b ^ c = b ^ b ^ a = a,a ^ c = a ^ a ^ b = b。
-
-**判断一个数是不是 4 的 n 次方**
-
-[Leetcode : 342. Power of Four (Easy)](https://leetcode.com/problems/power-of-four/)
-
-该数二进制表示有且只有一个奇数位为 1 ,其余的都为 0 ,例如 16 : 10000。可以每次把 1 向左移动 2 位,就能构造出这种数字,然后比较构造出来的数与要判断的数是否相同。
-
-```java
-public boolean isPowerOfFour(int num) {
- int i = 1;
- while(i > 0){
- if(i == num) return true;
- i = i << 2;
- }
- return false;
-}
-```
-
-也可以用 Java 的 Integer.toString() 方法将该数转换为 4 进制形式的字符串,然后判断字符串是否以 1 开头。
-
-```java
-public boolean isPowerOfFour(int num) {
- return Integer.toString(num, 4).matches("10*");
-}
-```
+令 c = a ^ b,那么 b ^ c = b ^ b ^ a = a,a ^ c = a ^ a ^ b = b。
**判断一个数是不是 2 的 n 次方**
[Leetcode : 231. Power of Two (Easy)](https://leetcode.com/problems/power-of-two/description/)
-同样可以用 Power of Four 的方法,但是 2 的 n 次方更特殊,它的二进制表示只有一个 1 存在。
+二进制表示只有一个 1 存在。
```java
public boolean isPowerOfTwo(int n) {
@@ -5120,6 +5907,35 @@ public boolean isPowerOfTwo(int n) {
}
```
+**判断一个数是不是 4 的 n 次方**
+
+[Leetcode : 342. Power of Four (Easy)](https://leetcode.com/problems/power-of-four/)
+
+该数二进制表示有且只有一个奇数位为 1 ,其余的都为 0 ,例如 16 :10000。可以每次把 1 向左移动 2 位,就能构造出这种数字,然后比较构造出来的数与要判断的数是否相同。
+
+```java
+public boolean isPowerOfFour(int num) {
+ int i = 1;
+ while(i > 0){
+ if(i == num) return true;
+ i = i << 2;
+ }
+ return false;
+}
+```
+
+```java
+public boolean isPowerOfFour(int num) {
+ return Integer.toString(num, 4).matches("10*");
+}
+```
+
+```java
+public boolean isPowerOfFour(int num) {
+ return num > 0 && (num & (num - 1)) == 0 && (num & 0b01010101010101010101010101010101) != 0;
+}
+```
+
**数组中唯一一个不重复的元素**
[Leetcode : 136. Single Number (Easy)](https://leetcode.com/problems/single-number/description/)
@@ -5178,7 +5994,13 @@ public boolean hasAlternatingBits(int n) {
[Leetcode : 476. Number Complement (Easy)](https://leetcode.com/problems/number-complement/description/)
-不考虑二进制表示中的首 0 部分
+```html
+Input: 5
+Output: 2
+Explanation: The binary representation of 5 is 101 (no leading zero bits), and its complement is 010. So you need to output 2.
+```
+
+不考虑二进制表示中的首 0 部分。
对于 00000101,要求补码可以将它与 00000111 进行异或操作。那么问题就转换为求掩码 00000111。
@@ -5239,6 +6061,12 @@ public int getSum(int a, int b) {
[Leetcode : 318. Maximum Product of Word Lengths (Medium)](https://leetcode.com/problems/maximum-product-of-word-lengths/description/)
+```html
+Given ["abcw", "baz", "foo", "bar", "xtfn", "abcdef"]
+Return 16
+The two words can be "abcw", "xtfn".
+```
+
题目描述:字符串数组的字符串只含有小写字符。求解字符串数组中两个字符串长度的最大乘积,要求这两个字符串不能含有相同字符。
解题思路:本题主要问题是判断两个字符串是否含相同字符,由于字符串只含有小写字符,总共 26 位,因此可以用一个 32 位的整数来存储每个字符是否出现过。
@@ -5265,6 +6093,22 @@ public int maxProduct(String[] words) {
}
```
+**统计从 0 \~ n 每个数的二进制表示中 1 的个数**
+
+[Leetcode : 338. Counting Bits (Medium)](https://leetcode.com/problems/counting-bits/description/)
+
+对于数字 6(110),它可以看成是数字 (10) 前面加上一个 1 ,因此 dp[i] = dp[i&(i-1)] + 1;
+
+```java
+public int[] countBits(int num) {
+ int[] ret = new int[num + 1];
+ for(int i = 1; i <= num; i++){
+ ret[i] = ret[i&(i-1)] + 1;
+ }
+ return ret;
+}
+```
+
# 参考资料
- [Leetcode](https://leetcode.com/problemset/algorithms/?status=Todo)
diff --git a/notes/Linux.md b/notes/Linux.md
index 2c072e75..dbeddf16 100644
--- a/notes/Linux.md
+++ b/notes/Linux.md
@@ -1,121 +1,99 @@
-* [常用操作以及概念](#常用操作以及概念)
+* [一、常用操作以及概念](#一常用操作以及概念)
* [求助](#求助)
* [关机](#关机)
- * [查看进程](#查看进程)
- * [查看端口](#查看端口)
* [PATH](#path)
* [运行等级](#运行等级)
* [sudo](#sudo)
* [GNU](#gnu)
* [包管理工具](#包管理工具)
- * [常见发行版本](#常见发行版本)
-* [分区](#分区)
+ * [发行版](#发行版)
+ * [VIM 三个模式](#vim-三个模式)
+* [二、分区](#二分区)
* [磁盘的文件名](#磁盘的文件名)
* [分区表](#分区表)
- * [1. MBR](#1-mbr)
- * [2. GPT](#2-gpt)
* [开机检测程序](#开机检测程序)
- * [1. BIOS](#1-bios)
- * [2. UEFI](#2-uefi)
* [挂载](#挂载)
-* [文件权限与目录配置](#文件权限与目录配置)
+* [三、文件](#三文件)
* [文件权限概念](#文件权限概念)
* [文件属性以及权限的修改](#文件属性以及权限的修改)
- * [1. 修改文件所属群组](#1-修改文件所属群组)
- * [2. 修改文件拥有者](#2-修改文件拥有者)
- * [3. 修改权限](#3-修改权限)
* [目录的权限](#目录的权限)
* [文件默认权限](#文件默认权限)
* [目录配置](#目录配置)
-* [文件与目录](#文件与目录)
* [文件时间](#文件时间)
* [文件与目录的基本操作](#文件与目录的基本操作)
- * [1. ls](#1-ls)
- * [2. cp](#2-cp)
- * [3. rm](#3-rm)
- * [4. mv](#4-mv)
* [获取文件内容](#获取文件内容)
- * [1. cat](#1-cat)
- * [2. tac](#2-tac)
- * [3. more](#3-more)
- * [4. less](#4-less)
- * [5. head](#5-head)
- * [6. tail](#6-tail)
- * [7. od](#7-od)
- * [8. touch](#8-touch)
* [指令与文件搜索](#指令与文件搜索)
- * [1. which](#1-which)
- * [2. whereis](#2-whereis)
- * [3. locate](#3-locate)
- * [4. find](#4-find)
- * [4.1 与时间有关的选项](#41-与时间有关的选项)
- * [4.2 与文件拥有者和所属群组有关的选项](#42-与文件拥有者和所属群组有关的选项)
- * [4.3 与文件权限和名称有关的选项](#43-与文件权限和名称有关的选项)
-* [磁盘与文件系统](#磁盘与文件系统)
+* [四、磁盘与文件系统](#四磁盘与文件系统)
* [文件系统的组成](#文件系统的组成)
* [inode](#inode)
* [目录的 inode 与 block](#目录的-inode-与-block)
* [实体链接与符号链接](#实体链接与符号链接)
- * [1. 实体链接](#1-实体链接)
- * [2. 符号链接](#2-符号链接)
-* [压缩与打包](#压缩与打包)
+* [五、压缩与打包](#五压缩与打包)
* [压缩](#压缩)
- * [1. gzip](#1-gzip)
- * [2. bzip2](#2-bzip2)
- * [3. xz](#3-xz)
* [打包](#打包)
-* [Bash](#bash)
- * [Bash 特性](#bash-特性)
+* [六、Bash](#六bash)
+ * [特性](#特性)
* [变量操作](#变量操作)
* [指令搜索顺序](#指令搜索顺序)
* [数据流重定向](#数据流重定向)
- * [管线指令](#管线指令)
- * [1. 提取指令:cut](#1-提取指令cut)
- * [2. 排序命令:sort、uniq](#2-排序命令sortuniq)
- * [3. 双向输出重定向:tee](#3-双向输出重定向tee)
- * [4. 字符转换指令:tr、col、expand、join、paste](#4-字符转换指令trcolexpandjoinpaste)
- * [5. 分区指令:split](#5-分区指令split)
-* [正规表示法与文件格式化处理](#正规表示法与文件格式化处理)
+* [七、管线指令](#七管线指令)
+ * [提取指令](#提取指令)
+ * [排序指令](#排序指令)
+ * [双向输出重定向](#双向输出重定向)
+ * [字符转换指令](#字符转换指令)
+ * [分区指令](#分区指令)
+* [八、正则表达式](#八正则表达式)
* [grep](#grep)
* [printf](#printf)
* [awk](#awk)
-* [vim 三个模式](#vim-三个模式)
+* [九、进程管理](#九进程管理)
+ * [查看进程](#查看进程)
+ * [进程状态](#进程状态)
+ * [SIGCHILD](#sigchild)
+ * [孤儿进程和僵死进程](#孤儿进程和僵死进程)
+* [十、I/O 复用](#十io-复用)
+ * [概念理解](#概念理解)
+ * [I/O 模型](#io-模型)
+ * [select poll epoll](#select-poll-epoll)
+ * [select 和 poll 比较](#select-和-poll-比较)
+ * [eopll 工作模式](#eopll-工作模式)
+ * [select poll epoll 应用场景](#select-poll-epoll-应用场景)
* [参考资料](#参考资料)
-# 常用操作以及概念
+# 一、常用操作以及概念
## 求助
-**1. --help**
+### 1. --help
指令的基本用法与选项介绍。
-**2. man**
+### 2. man
man 是 manual 的缩写,将指令的具体信息显示出来。
当执行 man date 时,有 DATE(1) 出现,其中的数字代表指令的类型,常用的数字及其类型如下:
| 代号 | 类型 |
-| -- | -- |
+| :--: | -- |
| 1 | 用户在 shell 环境中可以操作的指令或者可执行文件 |
| 5 | 配置文件 |
| 8 | 系统管理员可以使用的管理指令 |
-**3. info**
+### 3. info
info 与 man 类似,但是 info 将文档分成一个个页面,每个页面可以进行跳转。
## 关机
-**1. sync**
+### 1. sync
-为了加快对磁盘上文件的读写速度,位于内存中的文件数据不会立即同步到磁盘上,因此关机之前需要先进行 sync 同步操作。
+为了加快对磁盘文件的读写速度,位于内存中的文件数据不会立即同步到磁盘上,因此关机之前需要先进行 sync 同步操作。
-**2. shutdown**
+### 2. shutdown
```html
# /sbin/shutdown [-krhc] [时间] [警告讯息]
@@ -125,22 +103,10 @@ info 与 man 类似,但是 info 将文档分成一个个页面,每个页面
-c : 取消已经在进行的 shutdown 指令内容
```
-**3. 其它关机指令**
+### 3. 其它关机指令
reboot、halt、poweroff。
-## 查看进程
-
-```html
-ps aux | grep threadx
-```
-
-## 查看端口
-
-```html
-netstat -anp | grep 80
-```
-
## PATH
可以在环境变量 PATH 中声明可执行文件的路径,路径之间用 : 分隔。
@@ -165,7 +131,7 @@ netstat -anp | grep 80
## GNU
-GNU 计划,又译为革奴计划,它的目标是创建一套完全自由的操作系统,称为 GNU,其内容软件完全以 GPL 方式发布。其中 GPL 全称为 GNU 通用公共许可协议,包含了以下内容:
+GNU 计划,译为革奴计划,它的目标是创建一套完全自由的操作系统,称为 GNU,其内容软件完全以 GPL 方式发布。其中 GPL 全称为 GNU 通用公共许可协议,包含了以下内容:
- 以任何目的运行此程序的自由;
- 再复制的自由;
@@ -177,7 +143,7 @@ RPM 和 DPKG 为最常见的两类软件包管理工具。RPM 全称为 Redhat P
YUM 基于 RPM 包管理工具,具有依赖管理功能,并具有软件升级的功能。
-## 常见发行版本
+## 发行版
Linux 发行版是 Linux 内核及各种应用软件的集成版本。
@@ -186,13 +152,30 @@ Linux 发行版是 Linux 内核及各种应用软件的集成版本。
| DPKG | Ubuntu | Debian |
| RPM | Red Hat | Fedora / CentOS |
-# 分区
+## VIM 三个模式
+
+- 一般指令模式(Command mode):进入 VIM 的默认模式,可以用于移动游标查看内容;
+- 编辑模式(Insert mode):按下 "i" 等按键之后进入,可以对文本进行编辑;
+- 指令列模式(Bottom-line mode):按下 ":" 按键之后进入,用于保存退出等操作。
+
+
+
+在指令列模式下,有以下命令用于离开或者保存文件。
+
+| 命令 | 作用 |
+| :--: | -- |
+| :w | 写入磁盘|
+| :w! | 当文件为只读时,强制写入磁盘。到底能不能写入,与用户对该文件的权限有关 |
+| :q | 离开 |
+| :q! | 强制离开不保存 |
+| :wq | 写入磁盘后离开 |
+| :wq!| 强制写入磁盘后离开 |
+
+# 二、分区
## 磁盘的文件名
-Linux 中每个硬件都被当做一个文件。
-
-常见磁盘的文件名:
+Linux 中每个硬件都被当做一个文件,包括磁盘。常见磁盘的文件名如下:
- SCSI/SATA/USB 磁盘:/dev/sd[a-p]
- IDE 磁盘:/dev/hd[a-d]
@@ -205,9 +188,9 @@ Linux 中每个硬件都被当做一个文件。
### 1. MBR
-MBR 中,第一个扇区最重要,里面有主要开机记录(Master boot record, MBR)及分区表(partition table),其中 MBR 占 446 bytes,partition table 占 64 bytes。
+MBR 中,第一个扇区最重要,里面有主要开机记录(Master boot record, MBR)及分区表(partition table),其中 MBR 占 446 bytes,分区表占 64 bytes。
-分区表只有 64 bytes,最多只能存储 4 个分区,这 4 个分区为主分区(Primary)和扩展分区(Extended)。其中扩展分区只有一个,它将其它空间用来记录分区表,因此通过扩展分区可以分出更多区分,这些分区称为逻辑分区。
+分区表只有 64 bytes,最多只能存储 4 个分区,这 4 个分区为主分区(Primary)和扩展分区(Extended)。其中扩展分区只有一个,它将其它空间用来记录分区表,因此通过扩展分区可以分出更多分区,这些分区称为逻辑分区。
Linux 也把分区当成文件,分区文件的命名方式为:磁盘文件名 + 编号,例如 /dev/sda1。注意,逻辑分区的编号从 5 开始。
@@ -219,20 +202,20 @@ GPT 第 1 个区块记录了 MBR,紧接着是 33 个区块记录分区信息
GPT 没有扩展分区概念,都是主分区,最多可以分 128 个分区。
-
+
## 开机检测程序
### 1. BIOS
-BIOS 是开机的时候计算机执行的第一个程序,这个程序知道可以开机的磁盘,并读取磁盘第一个扇区的 MBR,由 MBR 执行其中的开机管理程序,这个开机管理程序的会加载操作系统的核心文件。
+BIOS 是开机的时候计算机执行的第一个程序,这个程序知道可以开机的磁盘,并读取磁盘第一个扇区的 MBR,由 MBR 执行其中的开机管理程序,这个开机管理程序会加载操作系统的核心文件。
MBR 中的开机管理程序提供以下功能:选单、载入核心文件以及转交其它开机管理程序。转交这个功能可以用来实现了多重引导,只需要将另一个操作系统的开机管理程序安装在其它分区的启动扇区上,在启动 MBR 中的开机管理程序时,就可以选择启动当前的操作系统或者转交给其它开机管理程序从而启动另一个操作系统。
-
-
安装多重引导,最好先安装 Windows 再安装 Linux。因为安装 Windows 时会覆盖掉 MBR,而 Linux 可以选择将开机管理程序安装在 MBR 或者其它分区的启动扇区,并且可以设置开机管理程序的选单。
+
+
### 2. UEFI
UEFI 相比于 BIOS 来说功能更为全面,也更为安全。
@@ -241,15 +224,15 @@ UEFI 相比于 BIOS 来说功能更为全面,也更为安全。
挂载利用目录作为分区的进入点,也就是说,进入目录之后就可以读取分区的数据。
-
+
-# 文件权限与目录配置
+# 三、文件
## 文件权限概念
把用户分为三种:文件拥有者、群组以及其它人,对不同的用户有不同的文件权限。
-使用 ls 查看一个文件时,会显示一个文件的信息,例如 drwxr-xr-x. 3 root root 17 May 6 00:14 .config,对这个信息的解释如下:
+使用 ls 查看一个文件时,会显示一个文件的信息,例如 `drwxr-xr-x. 3 root root 17 May 6 00:14 .config`,对这个信息的解释如下:
- drwxr-xr-x:文件类型以及权限,第 1 位为文件类型字段,后 9 位为文件权限字段。
- 3:链接数;
@@ -325,8 +308,8 @@ UEFI 相比于 BIOS 来说功能更为全面,也更为安全。
## 文件默认权限
-文件默认权限:文件默认没有可执行权限,因此为 666,也就是 -rw-rw-rw- 。
-目录默认权限:目录必须要能够进入,也就是必须拥有可执行权限,因此为 777 ,也就是 drwxrwxrwx。
+- 文件默认权限:文件默认没有可执行权限,因此为 666,也就是 -rw-rw-rw- 。
+- 目录默认权限:目录必须要能够进入,也就是必须拥有可执行权限,因此为 777 ,也就是 drwxrwxrwx。
可以通过 umask 设置或者查看文件的默认权限,通常以掩码的形式来表示,例如 002 表示其它用户的权限去除了一个 2 的权限,也就是写权限,因此建立新文件时默认的权限为 -rw-rw-r-- 。
@@ -338,11 +321,7 @@ UEFI 相比于 BIOS 来说功能更为全面,也更为安全。
- /usr (unix software resource):所有系统默认软件都会安装到这个目录;
- /var (variable):存放系统或程序运行过程中的数据文件。
-完整的目录树如下:
-
-
-
-# 文件与目录
+
## 文件时间
@@ -448,8 +427,8 @@ cp [-adfilprsu] source destination
-a : 更新 atime
-c : 更新 ctime,若该文件不存在则不建立新文件
-m : 更新 mtime
--d : 后面可以接欲更新的日期而不用当前的日期,也可以使用 --date="日期或时间"
--t : 后面可以接欲更新的时间而不用当前的时间,格式为[YYYYMMDDhhmm]
+-d : 后面可以接更新日期而不使用当前日期,也可以使用 --date="日期或时间"
+-t : 后面可以接更新时间而不使用当前时间,格式为[YYYYMMDDhhmm]
```
## 指令与文件搜索
@@ -487,10 +466,11 @@ locate 使用 /var/lib/mlocate/ 这个数据库来进行搜索,它存储在内
find 可以使用文件的属性和权限进行搜索。
```html
-# find filename [option]
+# find [basedir] [option]
+example: find . -name "shadow*"
```
-#### 4.1 与时间有关的选项
+(一)与时间有关的选项
```html
-mtime n :列出在 n 天前的那一天修改过内容的文件
@@ -501,9 +481,9 @@ find 可以使用文件的属性和权限进行搜索。
+4、4 和 -4 的指示的时间范围如下:
-
+
-#### 4.2 与文件拥有者和所属群组有关的选项
+(二)与文件拥有者和所属群组有关的选项
```html
-uid n
@@ -514,7 +494,7 @@ find 可以使用文件的属性和权限进行搜索。
-nogroup:搜索所属群组不存在于 /etc/group 的文件
```
-#### 4.3 与文件权限和名称有关的选项
+(三)与文件权限和名称有关的选项
```html
-name filename
@@ -525,7 +505,7 @@ find 可以使用文件的属性和权限进行搜索。
-perm /mode :搜索权限包含任一 mode 的文件
```
-# 磁盘与文件系统
+# 四、磁盘与文件系统
## 文件系统的组成
@@ -537,21 +517,19 @@ find 可以使用文件的属性和权限进行搜索。
2. inode:一个文件占用一个 inode,记录文件的属性,同时记录此文件的内容所在的 block 号码;
3. block:记录文件的内容,文件太大时,会占用多个 block。
+
+
当要读取一个文件的内容时,先在 inode 中去查找文件内容所在的所有 block,然后把所有 block 的内容读出来。
磁盘碎片是指一个文件内容所在的 block 过于分散。
-Ext2 文件系统使用了上述的文件结构,并在此之上加入了 block 群组的概念,也就是将一个文件系统划分为多个 block 群组,方便管理。
-
-
-
## inode
Ext2 文件系统支持的 block 大小有 1k、2k 和 4k 三种,不同的 block 大小限制了单一文件的大小。而每个 inode 大小是固定为 128 bytes。
inode 中记录了文件内容所在的 block,但是每个 block 非常小,一个大文件随便都需要几十万的 block。而一个 inode 大小有限,无法直接引用这么多 block。因此引入了间接、双间接、三间接引用。间接引用是指,让 inode 记录的引用 block 块当成 inode 用来记录引用信息。
-
+
inode 具体包含以下信息:
@@ -578,9 +556,11 @@ inode 具体包含以下信息:
### 1. 实体链接
-hard link 只是在某个目录下新增一个条目,使得新增的条目链接到文件的 inode 上。删除任意一个条目,文件还是存在,只要引用数量不为 0。
+它和普通文件类似,实体链接文件的 inode 都指向源文件所在的 block 上,也就是说读取文件直接从源文件的 block 上读取。
-有以下限制:不能跨越 File System;不能对目录进行链接。
+删除任意一个条目,文件还是存在,只要引用数量不为 0。
+
+有以下限制:不能跨越 File System、不能对目录进行链接。
```html
# ln /etc/crontab .
@@ -591,9 +571,11 @@ hard link 只是在某个目录下新增一个条目,使得新增的条目链
### 2. 符号链接
-symbolic link 可以理解为 Windows 的快捷方式,通过建立一个独立的文件,这个文件的数据的读取指向链接的那个文件。当源文件被删除了,链接文件就打不开了。
+符号链接文件保存着源文件所在的绝对路径,在读取时会定位到源文件上,可以理解为 Windows 的快捷方式。
-symbolic link 可以为目录建立链接。
+当源文件被删除了,链接文件就打不开了。
+
+可以为目录建立链接。
```html
# ll -i /etc/crontab /root/crontab2
@@ -601,7 +583,7 @@ symbolic link 可以为目录建立链接。
53745909 lrwxrwxrwx. 1 root root 12 Jun 23 22:31 /root/crontab2 -> /etc/crontab
```
-# 压缩与打包
+# 五、压缩与打包
## 压缩
@@ -653,7 +635,7 @@ $ bzip2 [-cdkzv#] filename
提供比 bzip2 更佳的压缩比。
-可以看到,gzip、bzip2、xz 的压缩比不断优化。不过要注意,压缩比越高,压缩的时间也越长。
+可以看到,gzip、bzip2、xz 的压缩比不断优化。不过要注意的是,压缩比越高,压缩的时间也越长。
查看命令:xzcat、xzmore、xzless、xzgrep。
@@ -686,29 +668,21 @@ $ tar [-z|-j|-J] [xv] [-f 已有的 tar 文件] [-C 目录] ==解压缩
| 查 看 | tar -jtv -f filename.tar.bz2 |
| 解压缩 | tar -jxv -f filename.tar.bz2 -C 要解压缩的目录 |
-# Bash
+# 六、Bash
可以通过 Shell 请求内核提供服务,Bash 正是 Shell 的一种。
-## Bash 特性
+## 特性
-**1. 命令历史**
+1. 命令历史:记录使用过的命令。本次登录所执行的命令都会暂时存放到内存中,\~/.bash_history 文件中记录的是前一次登录所执行过的命令。
-记录使用过的命令。本次登录所执行的命令都会暂时存放到内存中, \~/.bash_history 文件中记录的是前一次登录所执行过的命令。
+2. 命令与文件补全:快捷键:tab。
-**2. 命令与文件补全**
+3. 命名别名:例如 lm 是 ls -al 的别名。
-快捷键:tab
+4. shell scripts。
-**3. 命名别名**
-
-例如 lm 是 ls -al 的别名。
-
-**4. shell scripts**
-
-**5. 通配符**
-
-例如 ls -l /usr/bin/X\* 列出 /usr/bin 下面所有以 X 开头的文件。
+5. 通配符:例如 ls -l /usr/bin/X\* 列出 /usr/bin 下面所有以 X 开头的文件。
## 变量操作
@@ -722,7 +696,10 @@ $ echo $var
$ echo ${var}
```
-变量内容如果有空格,需要使用双引号或者单引号。双引号内的特殊字符可以保留原本特性,例如 var="lang is \$LANG",则 var 的值为 lang is zh_TW.UTF-8;而单引号内的特殊字符就是特殊字符本身,例如 var='lang is \$LANG',则 var 的值为 lang is \$LANG。
+变量内容如果有空格,必须需要使用双引号或者单引号。
+
+- 双引号内的特殊字符可以保留原本特性,例如 var="lang is \$LANG",则 var 的值为 lang is zh_TW.UTF-8;
+- 单引号内的特殊字符就是特殊字符本身,例如 var='lang is \$LANG',则 var 的值为 lang is \$LANG。
可以使用 \`指令\` 或者 \$(指令) 的方式将指令的执行结果赋值给变量。例如 version=\$(uname -r),则 version 的值为 3.10.0-229.el7.x86_64。
@@ -755,11 +732,13 @@ $ echo ${array[1]}
## 数据流重定向
-重定向就是使用文件代替标准输入、标准输出和标准错误输出。
+重定向指的是使用文件代替标准输入、标准输出和标准错误输出。
-1. 标准输入 (stdin) :代码为 0 ,使用 < 或 << ;
-2. 标准输出 (stdout) :代码为 1 ,使用 > 或 >> ;
-3. 标准错误输出 (stderr):代码为 2 ,使用 2> 或 2>> ;
+| 1 | 代码 | 运算符 |
+| :---: | :---: | :---:|
+| 标准输入 (stdin) | 0 | < 或 << |
+| 标准输出 (stdout) | 1 | > 或 >> |
+| 标准错误输出 (stderr) | 2 | 2> 或 2>> |
其中,有一个箭头的表示以覆盖的方式重定向,而有两个箭头的表示以追加的方式重定向。
@@ -771,7 +750,7 @@ $ echo ${array[1]}
$ find /home -name .bashrc > list 2>&1
```
-## 管线指令
+# 七、管线指令
管线是将一个命令的标准输出作为另一个命令的标准输入,在数据需要经过多个步骤的处理之后才能得到我们想要的内容时就可以使用管线。在命令之间使用 | 分隔各个管线命令。
@@ -779,7 +758,7 @@ $ find /home -name .bashrc > list 2>&1
$ ls -al /etc | less
```
-### 1. 提取指令:cut
+## 提取指令
cut 对数据进行切分,取出想要的部分。提取过程一行一行地进行。
@@ -814,7 +793,7 @@ declare -x HOSTNAME="study.centos.vbird"
$ export | cut -c 12
```
-### 2. 排序命令:sort、uniq
+## 排序指令
**sort** 进行排序。
@@ -860,7 +839,7 @@ $ last | cut -d ' ' -f 1 | sort | uniq -c
1 wtmp
```
-### 3. 双向输出重定向:tee
+## 双向输出重定向
输出重定向会将输出内容重定向到文件中,而 **tee** 不仅能够完成这个功能,还能保留屏幕上的输出。也就是说,使用 tee 指令,一个输出会同时传送到文件和屏幕上。
@@ -868,7 +847,7 @@ $ last | cut -d ' ' -f 1 | sort | uniq -c
$ tee [-a] file
```
-### 4. 字符转换指令:tr、col、expand、join、paste
+## 字符转换指令
**tr** 用来删除一行中的字符,或者对字符进行替换。
@@ -914,7 +893,7 @@ $ paste [-d] file1 file2
-d :分隔符,默认为 tab
```
-### 5. 分区指令:split
+## 分区指令
**split** 将一个文件划分成多个文件。
@@ -925,7 +904,7 @@ $ split [-bl] file PREFIX
- PREFIX :分区文件的前导名称
```
-# 正规表示法与文件格式化处理
+# 八、正则表达式
## grep
@@ -952,7 +931,7 @@ $ grep -n 'the' regular_express.txt
18:google is the best tools for search keyword
```
-因为 { 与 } 的符号在 shell 是有特殊意义的,因此必须要使用使用转义字符进行转义。
+因为 { 和 } 在 shell 是有特殊意义的,因此必须要使用转义字符进行转义。
```html
$ grep -n 'go\{2,5\}g' regular_express.txt
@@ -973,8 +952,10 @@ $ printf '%10s %5i %5i %5i %8.2f \n' $(cat printf.txt)
## awk
+可以根据字段的某些条件进行匹配,例如匹配字段小于某个值的那一行数据。
+
```html
-$ awk '条件类型 1 {动作 1} 条件类型 2 {动作 2} ...' filename
+$ awk ' 条件类型 1 {动作 1} 条件类型 2 {动作 2} ...' filename
```
awk 每次处理一行,处理的最小单位是字段,每个字段的命名方式为:\$n,n 为字段号,从 1 开始,\$0 表示一整行。
@@ -1016,28 +997,403 @@ dmtsai lines: 5 columns: 9
范例 3:/etc/passwd 文件第三个字段为 UID,对 UID 小于 10 的数据进行处理。
```text
-cat /etc/passwd | awk 'BEGIN {FS=":"} $3 < 10 {print $1 "\t " $3}'
+$ cat /etc/passwd | awk 'BEGIN {FS=":"} $3 < 10 {print $1 "\t " $3}'
root 0
bin 1
daemon 2
```
-# vim 三个模式
+# 九、进程管理
-
+## 查看进程
-在指令列模式下,有以下命令用于离开或者存储文件。
+### 1. ps
-| 命令 | 作用 |
-| -- | -- |
-| :w | 写入磁盘|
-| :w! | 当文件为只读时,强制写入磁盘。到底能不能写入,与用户对该文件的权限有关 |
-| :q | 离开 |
-| :q! | 强制离开不保存 |
-| :wq | 写入磁盘后离开 |
-| :wq!| 强制写入磁盘后离开 |
+查看某个时间点的进程信息
+
+示例一:查看自己的进程
+
+```
+# ps -l
+```
+
+示例二:查看系统所有进程
+
+```
+# ps aux
+```
+
+示例三:查看特定的进程
+
+```
+# ps aux | grep threadx
+```
+
+### 2. top
+
+实时显示进程信息
+
+示例:两秒钟刷新一次
+
+```
+# top -d 2
+```
+
+### 3. pstree
+
+查看进程树
+
+示例:查看所有进程树
+
+```
+# pstree -A
+```
+
+### 4. netstat
+
+查看占用端口的进程
+
+```
+# netstat -anp | grep port
+```
+
+## 进程状态
+
+
+
+
+| 状态 | 说明 |
+| :---: | --- |
+| R | running or runnable (on run queue) |
+| D | uninterruptible sleep (usually IO) |
+| S | interruptible sleep (waiting for an event to complete) |
+| Z | defunct/zombie, terminated but not reaped by its parent |
+| T | stopped, either by a job control signal or because it is being traced|
+
+## SIGCHILD
+
+当一个子进程改变了它的状态时:停止运行,继续运行或者退出,有两件事会发生在父进程中:
+
+- 得到 SIGCHLD 信号;
+- 阻塞的 waitpid(2)(或者 wait)调用会返回。
+
+
+
+## 孤儿进程和僵死进程
+
+### 1. 孤儿进程
+
+一个父进程退出,而它的一个或多个子进程还在运行,那么这些子进程将成为孤儿进程。孤儿进程将被 init 进程(进程号为 1)所收养,并由 init 进程对它们完成状态收集工作。
+
+由于孤儿进程会被 init 进程收养,所以孤儿进程不会对系统造成危害。
+
+### 2. 僵死进程
+
+一个子进程的进程描述符在子进程退出时不会释放,只有当父进程通过 wait 或 waitpid 获取了子进程信息后才会释放。如果子进程退出,而父进程并没有调用 wait 或 waitpid,那么子进程的进程描述符仍然保存在系统中,这种进程称之为僵死进程。
+
+僵死进程通过 ps 命令显示出来的状态为 Z。
+
+系统所能使用的进程号是有限的,如果大量的产生僵死进程,将因为没有可用的进程号而导致系统不能产生新的进程。
+
+要消灭系统中大量的僵死进程,只需要将其父进程杀死,此时所有的僵死进程就会变成孤儿进程,从而被 init 所收养,这样 init 就会释放所有的僵死进程所占有的资源,从而结束僵死进程。
+
+# 十、I/O 复用
+
+## 概念理解
+
+I/O Multiplexing 又被称为 Event Driven I/O,它可以让单个进程具有处理多个 I/O 事件的能力。
+
+当某个 I/O 事件条件满足时,进程会收到通知。
+
+如果一个 Web 服务器没有 I/O 复用,那么每一个 Socket 连接都需要创建一个线程去处理。如果同时连接几万个连接,那么就需要创建相同数量的线程。并且相比于多进程和多线程技术,I/O 复用不需要进程线程创建和切换的开销,系统开销更小。
+
+## I/O 模型
+
+- 阻塞(Blocking)
+- 非阻塞(Non-blocking)
+- 同步(Synchronous)
+- 异步(Asynchronous)
+
+阻塞非阻塞是等待 I/O 完成的方式,阻塞要求用户程序停止执行,直到 I/O 完成,而非阻塞在 I/O 完成之前还可以继续执行。
+
+同步异步是获知 I/O 完成的方式,同步需要时刻关心 I/O 是否已经完成,异步无需主动关心,在 I/O 完成时它会收到通知。
+
+
+
+### 1. 同步-阻塞
+
+这是最常见的一种模型,用户程序在使用 read() 时会执行系统调用从而陷入内核,之后就被阻塞直到系统调用完成。
+
+应该注意到,在阻塞的过程中,其他程序还可以执行,因此阻塞不意味着整个操作系统都被阻塞。因为其他程序还可以执行,因此不消耗 CPU 时间,这种模型的执行效率会比较高。
+
+
+
+### 2. 同步-非阻塞
+
+非阻塞意味着用户程序在执行系统调用后还可以继续执行,内核并不是马上执行完 I/O,而是以一个错误码来告知用户程序 I/O 还未完成。为了获得 I/O 完成事件,用户程序必须调用多次系统调用去询问内核,甚至是忙等,也就是在一个循环里面一直询问并等待。
+
+由于 CPU 要处理更多的用户程序的询问,因此这种模型的效率是比较低的。
+
+
+
+### 3. 异步-阻塞
+
+这是 I/O 复用使用的一种模式,通过使用 select,它可以监听多个 I/O 事件,当这些事件至少有一个发生时,用户程序会收到通知。
+
+
+
+### 4. 异步-非阻塞
+
+该模式下,I/O 操作会立即返回,之后可以处理其它操作,并且在 I/O 完成时会收到一个通知,此时会中断正在处理的操作,然后继续之前的操作。
+
+
+
+## select poll epoll
+
+这三个都是 I/O 多路复用的具体实现,select 出现的最早,之后是 poll,再是 epoll。
+
+### 1. select
+
+```c
+int select (int n, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);
+```
+
+- fd_set 表示描述符集合;
+- readset、writeset 和 exceptset 这三个参数指定让操作系统内核测试读、写和异常条件的描述符;
+- timeout 参数告知内核等待所指定描述符中的任何一个就绪可花多少时间;
+- 成功调用返回结果大于 0;出错返回结果为 -1;超时返回结果为 0。
+
+```c
+fd_set fd_in, fd_out;
+struct timeval tv;
+
+// Reset the sets
+FD_ZERO( &fd_in );
+FD_ZERO( &fd_out );
+
+// Monitor sock1 for input events
+FD_SET( sock1, &fd_in );
+
+// Monitor sock2 for output events
+FD_SET( sock2, &fd_out );
+
+// Find out which socket has the largest numeric value as select requires it
+int largest_sock = sock1 > sock2 ? sock1 : sock2;
+
+// Wait up to 10 seconds
+tv.tv_sec = 10;
+tv.tv_usec = 0;
+
+// Call the select
+int ret = select( largest_sock + 1, &fd_in, &fd_out, NULL, &tv );
+
+// Check if select actually succeed
+if ( ret == -1 )
+ // report error and abort
+else if ( ret == 0 )
+ // timeout; no event detected
+else
+{
+ if ( FD_ISSET( sock1, &fd_in ) )
+ // input event on sock1
+
+ if ( FD_ISSET( sock2, &fd_out ) )
+ // output event on sock2
+}
+```
+
+每次调用 select() 都需要将 fd_set \*readfds, fd_set \*writefds, fd_set \*exceptfds 链表内容全部从用户进程内存中复制到操作系统内核中,内核需要将所有 fd_set 遍历一遍,这个过程非常低效。
+
+返回结果中内核并没有声明哪些 fd_set 已经准备好了,所以如果返回值大于 0 时,程序需要遍历所有的 fd_set 判断哪个 I/O 已经准备好。
+
+在 Linux 中 select 最多支持 1024 个 fd_set 同时轮询,其中 1024 由 Linux 内核的 FD_SETSIZE 决定。如果需要打破该限制可以修改 FD_SETSIZE,然后重新编译内核。
+
+### 2. poll
+
+```c
+int poll (struct pollfd *fds, unsigned int nfds, int timeout);
+```
+
+```c
+struct pollfd {
+ int fd; //文件描述符
+ short events; //监视的请求事件
+ short revents; //已发生的事件
+};
+```
+
+```c
+// The structure for two events
+struct pollfd fds[2];
+
+// Monitor sock1 for input
+fds[0].fd = sock1;
+fds[0].events = POLLIN;
+
+// Monitor sock2 for output
+fds[1].fd = sock2;
+fds[1].events = POLLOUT;
+
+// Wait 10 seconds
+int ret = poll( &fds, 2, 10000 );
+// Check if poll actually succeed
+if ( ret == -1 )
+ // report error and abort
+else if ( ret == 0 )
+ // timeout; no event detected
+else
+{
+ // If we detect the event, zero it out so we can reuse the structure
+ if ( pfd[0].revents & POLLIN )
+ pfd[0].revents = 0;
+ // input event on sock1
+
+ if ( pfd[1].revents & POLLOUT )
+ pfd[1].revents = 0;
+ // output event on sock2
+}
+```
+
+它和 select() 功能基本相同。同样需要每次将 struct pollfd \*fds 复制到内核,返回后同样需要进行轮询每一个 pollfd 是否已经 I/O 准备好。poll() 取消了 1024 个描述符数量上限,但是数量太大以后不能保证执行效率,因为复制大量内存到内核十分低效,所需时间与描述符数量成正比。poll() 在 pollfd 的重复利用上比 select() 的 fd_set 会更好。
+
+如果在多线程下,如果一个线程对某个描述符调用了 poll() 系统调用,但是另一个线程关闭了该描述符,会导致 poll() 调用结果不确定,该问题同样出现在 select() 中。
+
+### 3. epoll
+
+```c
+int epoll_create(int size);
+int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
+int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout);
+```
+
+```c
+// Create the epoll descriptor. Only one is needed per app, and is used to monitor all sockets.
+// The function argument is ignored (it was not before, but now it is), so put your favorite number here
+int pollingfd = epoll_create( 0xCAFE );
+
+if ( pollingfd < 0 )
+ // report error
+
+// Initialize the epoll structure in case more members are added in future
+struct epoll_event ev = { 0 };
+
+// Associate the connection class instance with the event. You can associate anything
+// you want, epoll does not use this information. We store a connection class pointer, pConnection1
+ev.data.ptr = pConnection1;
+
+// Monitor for input, and do not automatically rearm the descriptor after the event
+ev.events = EPOLLIN | EPOLLONESHOT;
+// Add the descriptor into the monitoring list. We can do it even if another thread is
+// waiting in epoll_wait - the descriptor will be properly added
+if ( epoll_ctl( epollfd, EPOLL_CTL_ADD, pConnection1->getSocket(), &ev ) != 0 )
+ // report error
+
+// Wait for up to 20 events (assuming we have added maybe 200 sockets before that it may happen)
+struct epoll_event pevents[ 20 ];
+
+// Wait for 10 seconds, and retrieve less than 20 epoll_event and store them into epoll_event array
+int ready = epoll_wait( pollingfd, pevents, 20, 10000 );
+// Check if epoll actually succeed
+if ( ret == -1 )
+ // report error and abort
+else if ( ret == 0 )
+ // timeout; no event detected
+else
+{
+ // Check if any events detected
+ for ( int i = 0; i < ret; i++ )
+ {
+ if ( pevents[i].events & EPOLLIN )
+ {
+ // Get back our connection pointer
+ Connection * c = (Connection*) pevents[i].data.ptr;
+ c->handleReadEvent();
+ }
+ }
+}
+```
+
+epoll 仅仅适用于 Linux OS。
+
+它是 select 和 poll 的增强版,更加灵活而且没有描述符限制。它将用户关心的描述符放到内核的一个事件表中,从而只需要在用户空间和内核空间拷贝一次。
+
+select 和 poll 方式中,进程只有在调用一定的方法后,内核才对所有监视的描述符进行扫描。而 epoll 事先通过 epoll_ctl() 来注册描述符,一旦基于某个描述符就绪时,内核会采用类似 callback 的回调机制,迅速激活这个描述符,当进程调用 epoll_wait() 时便得到通知。
+
+新版本的 epoll_create(int size) 参数 size 不起任何作用,在旧版本的 epoll 中如果描述符的数量大于 size,不保证服务质量。
+
+epoll_ctl() 执行一次系统调用,用于向内核注册新的描述符或者是改变某个文件描述符的状态。已注册的描述符在内核中会被维护在一棵红黑树上,通过回调函数内核会将 I/O 准备好的描述符加入到一个链表中管理。
+
+epoll_wait() 取出在内核中通过链表维护的 I/O 准备好的描述符,将他们从内核复制到程序中,不需要像 select/poll 对注册的所有描述符遍历一遍。
+
+epoll 对多线程编程更有友好,同时多个线程对同一个描述符调用了 epoll_wait 也不会产生像 select/poll 的不确定情况。或者一个线程调用了 epoll_wait 另一个线程关闭了同一个描述符也不会产生不确定情况。
+
+## select 和 poll 比较
+
+### 1. 功能
+
+它们提供了几乎相同的功能,但是在一些细节上有所不同:
+
+- select 会修改 fd_set 参数,而 poll 不会;
+- select 默认只能监听 1024 个描述符,如果要监听更多的话,需要修改 FD_SETSIZE 之后重新编译;
+- poll 提供了更多的事件类型。
+
+### 2. 速度
+
+poll 和 select 在速度上都很慢。
+
+- 它们都采取轮询的方式来找到 I/O 完成的描述符,如果描述符很多,那么速度就会很慢;
+- select 只使用每个描述符的 3 位,而 poll 通常需要使用 64 位,因此 poll 需要复制更多的内核空间。
+
+### 3. 可移植性
+
+几乎所有的系统都支持 select,但是只有比较新的系统支持 poll。
+
+## eopll 工作模式
+
+epoll_event 有两种触发模式:LT(level trigger)和 ET(edge trigger)。
+
+### 1. LT 模式
+
+当 epoll_wait() 检测到描述符事件发生并将此事件通知应用程序,应用程序可以不立即处理该事件。下次调用 epoll_wait() 时,会再次响应应用程序并通知此事件。是默认的一种模式,并且同时支持 Blocking 和 No-Blocking。
+
+### 2. ET 模式
+
+当 epoll_wait() 检测到描述符事件发生并将此事件通知应用程序,应用程序必须立即处理该事件。如果不处理,下次调用 epoll_wait() 时,不会再次响应应用程序并通知此事件。很大程度上减少了 epoll 事件被重复触发的次数,因此效率要比 LT 模式高。只支持 No-Blocking,以避免由于一个文件句柄的阻塞读/阻塞写操作把处理多个文件描述符的任务饿死。
+
+## select poll epoll 应用场景
+
+很容易产生一种错觉认为只要用 epoll 就可以了,select poll 都是历史遗留问题,并没有什么应用场景,其实并不是这样的。
+
+### 1. select 应用场景
+
+select() poll() epoll_wait() 都有一个 timeout 参数,在 select() 中 timeout 的精确度为 1ns,而 poll() 和 epoll_wait() 中则为 1ms。所以 select 更加适用于实时要求更高的场景,比如核反应堆的控制。
+
+select 历史更加悠久,它的可移植性更好,几乎被所有主流平台所支持。
+
+### 2. poll 应用场景
+
+poll 没有最大描述符数量的限制,如果平台支持应该采用 poll 且对实时性要求并不是十分严格,而不是 select。
+
+需要同时监控小于 1000 个描述符。那么也没有必要使用 epoll,因为这个应用场景下并不能体现 epoll 的优势。
+
+需要监控的描述符状态变化多,而且都是非常短暂的。因为 epoll 中的所有描述符都存储在内核中,造成每次需要对描述符的状态改变都需要通过 epoll_ctl() 进行系统调用,频繁系统调用降低效率。epoll 的描述符存储在内核,不容易调试。
+
+### 3. epoll 应用场景
+
+程序只需要运行在 Linux 平台上,有非常大量的描述符需要同时轮询,而且这些连接最好是长连接。
+
+### 4. 性能对比
+
+> [epoll Scalability Web Page](http://lse.sourceforge.net/epoll/index.html)
# 参考资料
- 鸟哥. 鸟 哥 的 Linux 私 房 菜 基 础 篇 第 三 版[J]. 2009.
- [Linux 平台上的软件包管理](https://www.ibm.com/developerworks/cn/linux/l-cn-rpmdpkg/index.html)
+- [Boost application performance using asynchronous I/O](https://www.ibm.com/developerworks/linux/library/l-async/)
+- [Synchronous and Asynchronous I/O](https://msdn.microsoft.com/en-us/library/windows/desktop/aa365683(v=vs.85).aspx)
+- [Linux IO 模式及 select、poll、epoll 详解](https://segmentfault.com/a/1190000003063859)
+- [poll vs select vs event-based](https://daniel.haxx.se/docs/poll-vs-select.html)
+- [Linux 之守护进程、僵死进程与孤儿进程](http://liubigbin.github.io/2016/03/11/Linux-%E4%B9%8B%E5%AE%88%E6%8A%A4%E8%BF%9B%E7%A8%8B%E3%80%81%E5%83%B5%E6%AD%BB%E8%BF%9B%E7%A8%8B%E4%B8%8E%E5%AD%A4%E5%84%BF%E8%BF%9B%E7%A8%8B/)
+- [Linux process states](https://idea.popcount.org/2012-12-11-linux-process-states/)
+- [GUID Partition Table](https://en.wikipedia.org/wiki/GUID_Partition_Table)
diff --git a/notes/MySQL.md b/notes/MySQL.md
index 8bded3c5..5cc1fd69 100644
--- a/notes/MySQL.md
+++ b/notes/MySQL.md
@@ -1,48 +1,32 @@
-* [存储引擎](#存储引擎)
- * [1. InnoDB](#1-innodb)
- * [2. MyISAM](#2-myisam)
- * [3. InnoDB 与 MyISAM 的比较](#3-innodb-与-myisam-的比较)
-* [数据类型](#数据类型)
- * [1. 整型](#1-整型)
- * [2. 浮点数](#2-浮点数)
- * [3. 字符串](#3-字符串)
- * [4. 时间和日期](#4-时间和日期)
-* [索引](#索引)
- * [1. 索引分类](#1-索引分类)
- * [1.1 B-Tree 索引](#11-b-tree-索引)
- * [1.2 哈希索引](#12-哈希索引)
- * [1.3. 空间索引(R-Tree)](#13-空间索引r-tree)
- * [1.4 全文索引](#14-全文索引)
- * [2. 索引的优点](#2-索引的优点)
- * [3. 索引优化](#3-索引优化)
- * [3.1 独立的列](#31-独立的列)
- * [3.2 前缀索引](#32-前缀索引)
- * [3.3 多列索引](#33-多列索引)
- * [3.4 索引列的顺序](#34-索引列的顺序)
- * [3.5 聚簇索引](#35-聚簇索引)
- * [3.6 覆盖索引](#36-覆盖索引)
- * [4. B-Tree 和 B+Tree 原理](#4-b-tree-和-b+tree-原理)
- * [4. 1 B-Tree](#4-1-b-tree)
- * [4.2 B+Tree](#42-b+tree)
- * [4.3 带有顺序访问指针的 B+Tree](#43-带有顺序访问指针的-b+tree)
- * [4.4 为什么使用 B-Tree 和 B+Tree](#44-为什么使用-b-tree-和-b+tree)
-* [查询性能优化](#查询性能优化)
- * [1. Explain](#1-explain)
- * [2. 减少返回的列](#2-减少返回的列)
- * [3. 减少返回的行](#3-减少返回的行)
- * [4. 拆分大的 DELETE 或 INSERT 语句](#4-拆分大的-delete-或-insert-语句)
-* [分库与分表](#分库与分表)
-* [故障转移和故障恢复](#故障转移和故障恢复)
- * [1. 故障转移](#1-故障转移)
- * [2. 故障恢复](#2-故障恢复)
+* [一、存储引擎](#一存储引擎)
+ * [InnoDB](#innodb)
+ * [MyISAM](#myisam)
+ * [比较](#比较)
+* [二、数据类型](#二数据类型)
+ * [整型](#整型)
+ * [浮点数](#浮点数)
+ * [字符串](#字符串)
+ * [时间和日期](#时间和日期)
+* [三、索引](#三索引)
+ * [索引分类](#索引分类)
+ * [索引的优点](#索引的优点)
+ * [索引优化](#索引优化)
+ * [B-Tree 和 B+Tree 原理](#b-tree-和-b+tree-原理)
+* [四、查询性能优化](#四查询性能优化)
+* [五、切分](#五切分)
+ * [垂直切分](#垂直切分)
+ * [水平切分](#水平切分)
+ * [切分的选择](#切分的选择)
+ * [存在的问题](#存在的问题)
+* [六、故障转移和故障恢复](#六故障转移和故障恢复)
* [参考资料](#参考资料)
-# 存储引擎
+# 一、存储引擎
-## 1. InnoDB
+## InnoDB
InnoDB 是 MySQL 默认的事务型存储引擎,只有在需要 InnoDB 不支持的特性时,才考虑使用其它存储引擎。
@@ -54,9 +38,9 @@ InnoDB 是 MySQL 默认的事务型存储引擎,只有在需要 InnoDB 不支
通过一些机制和工具支持真正的热备份。
-## 2. MyISAM
+## MyISAM
-MyISAM 提供了大量的特性,包括全文索引、压缩、空间函数(GIS)等。但 MyISAM 不支持事务和行级锁,而且崩溃后无法安全恢复。
+MyISAM 提供了大量的特性,包括全文索引、压缩、空间函数(GIS)等。但 MyISAM 不支持事务和行级锁,而且崩溃后无法安全恢复。应该注意的是,MySQL5.6.4 添加了对 InnoDB 引擎的全文索引支持。
只能对整张表加锁,而不是针对行。
@@ -72,43 +56,29 @@ MyISAM 提供了大量的特性,包括全文索引、压缩、空间函数(G
MyISAM 设计简单,数据以紧密格式存储,所以在某些场景下性能很好。
-## 3. InnoDB 与 MyISAM 的比较
+## 比较
-**事务**
+1. 事务:InnoDB 是事务型的。
+2. 备份:InnoDB 支持在线热备份。
+3. 崩溃恢复:MyISAM 崩溃后发生损坏的概率比 InnoDB 高很多,而且恢复的速度也更慢。
+4. 并发:MyISAM 只支持表级锁,而 InnoDB 还支持行级锁。
+5. 其它特性:MyISAM 支持,地理空间索引。
-InnoDB 是事务型的。
+# 二、数据类型
-**备份**
-
-InnoDB 支持在线热备份。
-
-**崩溃恢复**
-
-MyISAM 崩溃后发生损坏的概率比 InnoDB 高很多,而且恢复的速度也更慢。
-
-**并发**
-
-MyISAM 只支持表级锁,而 InnoDB 还支持行级锁。
-
-**其它特性**
-
-MyISAM 支持全文索引,地理空间索引。
-
-# 数据类型
-
-## 1. 整型
+## 整型
TINYINT, SMALLINT, MEDIUMINT, INT, BIGINT 分别使用 8, 16, 24, 32, 64 位存储空间,一般情况下越小的列越好。
INT(11) 中的数字只是规定了交互工具显示字符的个数,对于存储和计算来说是没有意义的。
-## 2. 浮点数
+## 浮点数
FLOAT 和 DOUBLE 为浮点类型,DECIMAL 为高精度小数类型。CPU 原生支持浮点运算,但是不支持 DECIMAl 类型的计算,因此 DECIMAL 的计算比浮点类型需要更高的代价。
FLOAT、DOUBLE 和 DECIMAL 都可以指定列宽,例如 DECIMAL(18, 9) 表示总共 18 位,取 9 位存储小数部分,剩下 9 位存储整数部分。
-## 3. 字符串
+## 字符串
主要有 CHAR 和 VARCHAR 两种类型,一种是定长的,一种是变长的。
@@ -116,11 +86,11 @@ VARCHAR 这种变长类型能够节省空间,因为只需要存储必要的内
VARCHAR 会保留字符串末尾的空格,而 CHAR 会删除。
-## 4. 时间和日期
+## 时间和日期
MySQL 提供了两种相似的日期时间类型:DATATIME 和 TIMESTAMP。
-**DATATIME**
+### 1. DATATIME
能够保存从 1001 年到 9999 年的日期和时间,精度为秒,使用 8 字节的存储空间。
@@ -128,7 +98,7 @@ MySQL 提供了两种相似的日期时间类型:DATATIME 和 TIMESTAMP。
默认情况下,MySQL 以一种可排序的、无歧义的格式显示 DATATIME 值,例如“2008-01-16 22:37:08”,这是 ANSI 标准定义的日期和时间表示方法。
-**TIMESTAMP**
+### 2. TIMESTAMP
和 UNIX 时间戳相同,保存从 1970 年 1 月 1 日午夜(格林威治时间)以来的秒数,使用 4 个字节,只能表示从 1970 年 到 2038 年。
@@ -140,7 +110,7 @@ MySQL 提供了 FROM_UNIXTIME() 函数把 UNIX 时间戳转换为日期,并提
应该尽量使用 TIMESTAMP,因为它比 DATETIME 空间效率更高。
-# 索引
+# 三、索引
索引是在存储引擎层实现的,而不是在服务器层实现的,所以不同存储引擎具有不同的索引类型和实现。
@@ -148,43 +118,47 @@ MySQL 提供了 FROM_UNIXTIME() 函数把 UNIX 时间戳转换为日期,并提
对于非常小的表、大部分情况下简单的全表扫描比建立索引更高效。对于中到大型的表,索引就非常有效。但是对于特大型的表,建立和使用索引的代价将会随之增长。这种情况下,需要用到一种技术可以直接区分出需要查询的一组数据,而不是一条记录一条记录地匹配,例如可以使用分区技术。
-## 1. 索引分类
+## 索引分类
-### 1.1 B-Tree 索引
+### 1. B+Tree 索引
-B-Tree 索引是大多数 MySQL 存储引擎的默认索引类型。
+
+
+《高性能 MySQL》一书使用 B-Tree 进行描述,其实从技术上来说这种索引是 B+Tree。
+
+B+Tree 索引是大多数 MySQL 存储引擎的默认索引类型。
因为不再需要进行全表扫描,只需要对树进行搜索即可,因此查找速度快很多。
-可以指定多个列作为索引列,多个索引列共同组成键。B-Tree 索引适用于全键值、键值范围和键前缀查找,其中键前缀查找只适用于最左前缀查找。
+可以指定多个列作为索引列,多个索引列共同组成键。B+Tree 索引适用于全键值、键值范围和键前缀查找,其中键前缀查找只适用于最左前缀查找。
除了用于查找,还可以用于排序和分组。
如果不是按照索引列的顺序进行查找,则无法使用索引。
-### 1.2 哈希索引
+### 2. 哈希索引
基于哈希表实现,优点是查找非常快。
在 MySQL 中只有 Memory 引擎显式支持哈希索引。
-InnoDB 引擎有一个特殊的功能叫“自适应哈希索引”,当某个索引值被使用的非常频繁时,会在 B-Tree 索引之上再创建一个哈希索引,这样就让 B-Tree 索引具有哈希索引的一些优点,比如快速的哈希查找。
+InnoDB 引擎有一个特殊的功能叫“自适应哈希索引”,当某个索引值被使用的非常频繁时,会在 B+Tree 索引之上再创建一个哈希索引,这样就让 B+Tree 索引具有哈希索引的一些优点,比如快速的哈希查找。
限制:哈希索引只包含哈希值和行指针,而不存储字段值,所以不能使用索引中的值来避免读取行。不过,访问内存中的行的速度很快,所以大部分情况下这一点对性能影响并不明显;无法用于分组与排序;只支持精确查找,无法用于部分查找和范围查找;如果哈希冲突很多,查找速度会变得很慢。
-### 1.3. 空间索引(R-Tree)
+### 3. 空间索引(R-Tree)
MyISAM 存储引擎支持空间索引,可以用于地理数据存储。
空间索引会从所有维度来索引数据,可以有效地使用任意维度来进行组合查询。
-### 1.4 全文索引
+### 4. 全文索引
MyISAM 存储引擎支持全文索引,用于查找文本中的关键词,而不是直接比较索引中的值。
使用 MATCH AGAINST,而不是普通的 WHERE。
-## 2. 索引的优点
+## 索引的优点
- 大大减少了服务器需要扫描的数据量;
@@ -192,9 +166,9 @@ MyISAM 存储引擎支持全文索引,用于查找文本中的关键词,而
- 将随机 I/O 变为顺序 I/O。
-## 3. 索引优化
+## 索引优化
-### 3.1 独立的列
+### 1. 独立的列
在进行查询时,索引列不能是表达式的一部分,也不能是函数的参数,否则无法使用索引。
@@ -204,22 +178,22 @@ MyISAM 存储引擎支持全文索引,用于查找文本中的关键词,而
SELECT actor_id FROM sakila.actor WHERE actor_id + 1 = 5;
```
-### 3.2 前缀索引
+### 2. 前缀索引
对于 BLOB、TEXT 和 VARCHAR 类型的列,必须使用前缀索引,只索引开始的部分字符。
-对于前缀长度的选取需要根据 **索引选择性** 来确定:不重复的索引值和记录总数的比值。选择性越高,查询效率也越高。最大值为 1 ,此时每个记录都有唯一的索引与其对应。
+对于前缀长度的选取需要根据 **索引选择性** 来确定:不重复的索引值和记录总数的比值。选择性越高,查询效率也越高。最大值为 1,此时每个记录都有唯一的索引与其对应。
-### 3.3 多列索引
+### 3. 多列索引
-在需要使用多个列作为条件进行查询时,使用多列索引比使用多个单列索引性能更好。例如下面的语句中,最好把 actor_id 和 file_id 设置为多列索引。
+在需要使用多个列作为条件进行查询时,使用多列索引比使用多个单列索引性能更好。例如下面的语句中,最好把 actor_id 和 film_id 设置为多列索引。
```sql
-SELECT file_id, actor_ id FROM sakila.film_actor
-WhERE actor_id = 1 OR film_id = 1;
+SELECT film_id, actor_ id FROM sakila.film_actor
+WhERE actor_id = 1 AND film_id = 1;
```
-### 3.4 索引列的顺序
+### 4. 索引列的顺序
让选择性最强的索引列放在前面,例如下面显示的结果中 customer_id 的选择性比 staff_id 更高,因此最好把 customer_id 列放在多列索引的前面。
@@ -236,20 +210,20 @@ customer_id_selectivity: 0.0373
COUNT(*): 16049
```
-### 3.5 聚簇索引
+### 5. 聚簇索引
-
+
聚簇索引并不是一种索引类型,而是一种数据存储方式。
-术语“聚簇”表示数据行和相邻的键值紧密地存储在一起,InnoDB 的聚簇索引的数据行存放在 B-Tree 的叶子页中。
+术语“聚簇”表示数据行和相邻的键值紧密地存储在一起,InnoDB 的聚簇索引的数据行存放在 B+Tree 的叶子页中。
因为无法把数据行存放在两个不同的地方,所以一个表只能有一个聚簇索引。
**优点**
1. 可以把相关数据保存在一起,减少 I/O 操作;
-2. 因为数据保存在 B-Tree 中,因此数据访问更快。
+2. 因为数据保存在 B+Tree 中,因此数据访问更快。
**缺点**
@@ -259,13 +233,19 @@ customer_id_selectivity: 0.0373
4. 当插入到某个已满的页中,存储引擎会将该页分裂成两个页面来容纳该行,页分裂会导致表占用更多的磁盘空间。
5. 如果行比较稀疏,或者由于页分裂导致数据存储不连续时,聚簇索引可能导致全表扫描速度变慢。
-### 3.6 覆盖索引
+### 6. 覆盖索引
索引包含所有需要查询的字段的值。
-## 4. B-Tree 和 B+Tree 原理
+**优点**
-### 4. 1 B-Tree
+1. 因为索引条目通常远小于数据行的大小,所以若只读取索引,能大大减少数据访问量。
+2. 一些存储引擎(例如 MyISAM)在内存中只缓存索引,而数据依赖于操作系统来缓存。因此,只访问索引可以不使用系统调用(通常比较费时)。
+3. 对于 InnoDB 引擎,若二级索引能够覆盖查询,则无需访问聚簇索引。
+
+## B-Tree 和 B+Tree 原理
+
+### 1. B-Tree
@@ -281,7 +261,7 @@ B-Tree 是满足下列条件的数据结构:
由于插入删除新的数据记录会破坏 B-Tree 的性质,因此在插入删除时,需要对树进行一个分裂、合并、转移等操作以保持 B-Tree 性质。
-### 4.2 B+Tree
+### 2. B+Tree
@@ -290,25 +270,25 @@ B-Tree 是满足下列条件的数据结构:
- 每个节点的指针上限为 2d 而不是 2d+1;
- 内节点不存储 data,只存储 key,叶子节点不存储指针。
-### 4.3 带有顺序访问指针的 B+Tree
+### 3. 带有顺序访问指针的 B+Tree
一般在数据库系统或文件系统中使用的 B+Tree 结构都在经典 B+Tree 基础上进行了优化,在叶子节点增加了顺序访问指针,做这个优化的目的是为了提高区间访问的性能。
-### 4.4 为什么使用 B-Tree 和 B+Tree
+### 4. 为什么使用 B-Tree 和 B+Tree
红黑树等数据结构也可以用来实现索引,但是文件系统及数据库系统普遍采用 B-/+Tree 作为索引结构。
-页是计算机管理存储器的逻辑块,硬件及操作系统往往将主存和磁盘存储区分割为连续的大小相等的块,每个存储块称为一页(在许多操作系统中,页得大小通常为 4k),主存和磁盘以页为单位交换数据。
+页是计算机管理存储器的逻辑块,硬件及操作系统往往将主存和磁盘存储区分割为连续的大小相等的块,每个存储块称为一页(在许多操作系统中,页的大小通常为 4k),主存和磁盘以页为单位交换数据。
-一般来说,索引本身也很大,不可能全部存储在内存中,因此索引往往以索引文件的形式存储的磁盘上。为了减少磁盘 I/O,磁盘往往不是严格按需读取,而是每次都会预读。这样做的理论依据是计算机科学中著名的局部性原理:当一个数据被用到时,其附近的数据也通常会马上被使用。数据库系统的设计者巧妙利用了磁盘预读原理,将一个节点的大小设为等于一个页,这样每个节点只需要一次 I/O 就可以完全载入。B-Tree 中一次检索最多需要 h-1 次 I/O(根节点常驻内存),渐进复杂度为 O(h)=O(logdN)。一般实际应用中,出度 d 是非常大的数字,通常超过 100,因此 h 非常小(通常不超过 3)。而红黑树这种结构,h 明显要深的多。并且于逻辑上很近的节点(父子)物理上可能很远,无法利用局部性,效率明显比 B-Tree 差很多。
+一般来说,索引本身也很大,不可能全部存储在内存中,因此索引往往以索引文件的形式存储的磁盘上。为了减少磁盘 I/O,磁盘往往不是严格按需读取,而是每次都会预读。这样做的理论依据是计算机科学中著名的局部性原理:当一个数据被用到时,其附近的数据也通常会马上被使用。数据库系统的设计者巧妙利用了磁盘预读原理,将一个节点的大小设为等于一个页,这样每个节点只需要一次 I/O 就可以完全载入。B-Tree 中一次检索最多需要 h-1 次 I/O(根节点常驻内存),渐进复杂度为 O(h)=O(logd N)。一般实际应用中,出度 d 是非常大的数字,通常超过 100,因此 h 非常小(通常不超过 3)。而红黑树这种结构,h 明显要深的多。并且于逻辑上很近的节点(父子)物理上可能很远,无法利用局部性,效率明显比 B-Tree 差很多。
B+Tree 更适合外存索引,原因和内节点出度 d 有关。由于 B+Tree 内节点去掉了 data 域,因此可以拥有更大的出度,拥有更好的性能。
-# 查询性能优化
+# 四、查询性能优化
-## 1. Explain
+### Explain
用来分析 SQL 语句,分析结果中比较重要的字段有:
@@ -318,29 +298,30 @@ B+Tree 更适合外存索引,原因和内节点出度 d 有关。由于 B+Tree
- rows : 扫描的行数
-## 2. 减少返回的列
+### 减少返回的列
慢查询主要是因为访问了过多数据,除了访问过多行之外,也包括访问过多列。
最好不要使用 SELECT * 语句,要根据需要选择查询的列。
-## 3. 减少返回的行
+### 减少返回的行
最好使用 LIMIT 语句来取出想要的那些行。
-还可以建立索引来减少条件语句的全表扫描。例如对于下面的语句,不适用索引的情况下需要进行全表扫描,而使用索引只需要扫描几行记录即可,使用 Explain 语句可以通过观察 rows 字段来看出这种差异。
+还可以建立索引来减少条件语句的全表扫描。例如对于下面的语句,不使用索引的情况下需要进行全表扫描,而使用索引只需要扫描几行记录即可,使用 Explain 语句可以通过观察 rows 字段来看出这种差异。
```sql
SELECT * FROM sakila.film_actor WHERE film_id = 1;
```
-## 4. 拆分大的 DELETE 或 INSERT 语句
+### 拆分大的 DELETE 或 INSERT 语句
如果一次性执行的话,可能一次锁住很多数据、占满整个事务日志、耗尽系统资源、阻塞很多小的但重要的查询。
```sql
DELEFT FROM messages WHERE create < DATE_SUB(NOW(), INTERVAL 3 MONTH);
```
+
```sql
rows_affected = 0
do {
@@ -349,74 +330,64 @@ do {
} while rows_affected > 0
```
-# 分库与分表
+# 五、切分
-**1. 分表与分区的不同**
-
-分表,就是讲一张表分成多个小表,这些小表拥有不同的表名;而分区是将一张表的数据分为多个区块,这些区块可以存储在同一个磁盘上,也可以存储在不同的磁盘上,这种方式下表仍然只有一个。
-
-**2. 使用分库与分表的原因**
随着时间和业务的发展,数据库中的表会越来越多,并且表中的数据量也会越来越大,那么读写操作的开销也会随着增大。
-**3. 垂直切分**
+## 垂直切分
将表按功能模块、关系密切程度划分出来,部署到不同的库上。例如,我们会建立商品数据库 payDB、用户数据库 userDB 等,分别用来存储项目与商品有关的表和与用户有关的表。
-**4. 水平切分**
+## 水平切分
-把表中的数据按照某种规则存储到多个结构相同的表中,例如按 id 的散列值、性别等进行划分,
+把表中的数据按照某种规则存储到多个结构相同的表中,例如按 id 的散列值、性别等进行划分。
-**5. 垂直切分与水平切分的选择**
+## 切分的选择
如果数据库中的表太多,并且项目各项业务逻辑清晰,那么垂直切分是首选。
如果数据库的表不多,但是单表的数据量很大,应该选择水平切分。
-**6. 水平切分的实现方式**
+## 存在的问题
-最简单的是使用 merge 存储引擎。
-
-**7. 分库与分表存在的问题**
-
-(1) 事务问题
+### 1. 事务问题
在执行分库分表之后,由于数据存储到了不同的库上,数据库事务管理出现了困难。如果依赖数据库本身的分布式事务管理功能去执行事务,将付出高昂的性能代价;如果由应用程序去协助控制,形成程序逻辑上的事务,又会造成编程方面的负担。
-(2) 跨库跨表连接问题
+### 2. 跨库跨表连接问题
在执行了分库分表之后,难以避免会将原本逻辑关联性很强的数据划分到不同的表、不同的库上。这时,表的连接操作将受到限制,我们无法连接位于不同分库的表,也无法连接分表粒度不同的表,导致原本只需要一次查询就能够完成的业务需要进行多次才能完成。
-# 故障转移和故障恢复
+### 3. 额外的数据管理负担和数据运算压力
+
+最显而易见的就是数据的定位问题和数据的增删改查的重复执行问题,这些都可以通过应用程序解决,但必然引起额外的逻辑运算。
+
+
+# 六、故障转移和故障恢复
故障转移也叫做切换,当主库出现故障时就切换到备库,使备库成为主库。故障恢复顾名思义就是从故障中恢复过来,并且保证数据的正确性。
-## 1. 故障转移
-
-**1.1 提升备库或切换角色**
+### 提升备库或切换角色
提升一台备库为主库,或者在一个主-主复制结构中调整主动和被动角色。
-**1.2 虚拟 IP 地址和 IP 托管**
+### 虚拟 IP 地址和 IP 托管
为 MySQL 实例指定一个逻辑 IP 地址,当 MySQL 实例失效时,可以将 IP 地址转移到另一台 MySQL 服务器上。
-**1.3 中间件解决方案**
+### 中间件解决方案
通过代理,可以路由流量到可以使用的服务器上。
-
-
-**1.4 在应用中处理故障转移**
+### 在应用中处理故障转移
将故障转移整合到应用中可能导致应用变得太过笨拙。
-## 2. 故障恢复
-
-
# 参考资料
-- 高性能 MySQL
+- BaronScbwartz, PeterZaitsev, VadimTkacbenko, 等. 高性能 MySQL[M]. 电子工业出版社, 2013.
+- [How Sharding Works](https://medium.com/@jeeyoungk/how-sharding-works-b4dec46b3f6)
- [MySQL 索引背后的数据结构及算法原理 ](http://blog.codinglabs.org/articles/theory-of-mysql-index.html)
-- [MySQL 索引优化全攻略 ](http://www.runoob.com/w3cnote/mysql-index.html)
- [20+ 条 MySQL 性能优化的最佳经验 ](https://www.jfox.info/20-tiao-mysql-xing-nen-you-hua-de-zui-jia-jing-yan.html)
+- [数据库为什么分库分表?mysql的分库分表方案](https://www.i3geek.com/archives/1108)
diff --git a/notes/Redis.md b/notes/Redis.md
new file mode 100644
index 00000000..08b1f54f
--- /dev/null
+++ b/notes/Redis.md
@@ -0,0 +1,474 @@
+
+* [一、Redis 是什么](#一redis-是什么)
+* [二、五种基本类型](#二五种基本类型)
+ * [1. STRING](#1-string)
+ * [2. LIST](#2-list)
+ * [3. SET](#3-set)
+ * [4. HASH](#4-hash)
+ * [5. ZSET](#5-zset)
+* [三、键的过期时间](#三键的过期时间)
+* [四、发布与订阅](#四发布与订阅)
+* [五、事务](#五事务)
+* [六、持久化](#六持久化)
+ * [1. 快照持久化](#1-快照持久化)
+ * [2. AOF 持久化](#2-aof-持久化)
+* [七、复制](#七复制)
+ * [从服务器连接主服务器的过程](#从服务器连接主服务器的过程)
+ * [主从链](#主从链)
+* [八、处理故障](#八处理故障)
+* [九、分片](#九分片)
+ * [1. 客户端分片](#1-客户端分片)
+ * [2. 代理分片](#2-代理分片)
+ * [3. 服务器分片](#3-服务器分片)
+* [十、事件](#十事件)
+ * [事件类型](#事件类型)
+ * [事件的调度与执行](#事件的调度与执行)
+* [十一、Redis 与 Memcached 的区别](#十一redis-与-memcached-的区别)
+ * [数据类型](#数据类型)
+ * [数据持久化](#数据持久化)
+ * [分布式](#分布式)
+ * [内存管理机制](#内存管理机制)
+* [十二、Redis 适用场景](#十二redis-适用场景)
+ * [缓存](#缓存)
+ * [消息队列](#消息队列)
+ * [计数器](#计数器)
+ * [好友关系](#好友关系)
+* [十三、数据淘汰策略](#十三数据淘汰策略)
+* [十四、一个简单的论坛系统分析](#十四一个简单的论坛系统分析)
+ * [文章信息](#文章信息)
+ * [点赞功能](#点赞功能)
+ * [对文章进行排序](#对文章进行排序)
+* [参考资料](#参考资料)
+
+
+
+# 一、Redis 是什么
+
+Redis 是速度非常快的非关系型(NoSQL)内存键值数据库,可以存储键和五种不同类型的值之间的映射。
+
+五种类型数据类型为:字符串、列表、集合、有序集合、散列表。
+
+Redis 支持很多特性,例如将内存中的数据持久化到硬盘中,使用复制来扩展读性能,使用分片来扩展写性能。
+
+# 二、五种基本类型
+
+| 数据类型 | 可以存储的值 | 操作 |
+| :--: | :--: | :--: |
+| STRING | 字符串、整数或者浮点数 | 对整个字符串或者字符串的其中一部分执行操作 对整数和浮点数执行自增或者自减操作 |
+| LIST | 链表 | 从两端压入或者弹出元素 读取单个或者多个元素 进行修剪,只保留一个范围内的元素 |
+| SET | 无序集合 | 添加、获取、移除单个元素 检查一个元素是否存在于集合中 计算交集、并集、差集 从集合里面随机获取元素 |
+| HASH | 包含键值对的无序散列表 | 添加、获取、移除单个键值对 获取所有键值对 检查某个键是否存在|
+| ZSET | 有序集合 | 添加、获取、删除元素 根据分值范围或者成员来获取元素 计算一个键的排名 |
+
+> [What Redis data structures look like](https://redislabs.com/ebook/part-1-getting-started/chapter-1-getting-to-know-redis/1-2-what-redis-data-structures-look-like/)
+
+## 1. STRING
+
+
+
+```html
+> set hello world
+OK
+> get hello
+"world"
+> del hello
+(integer) 1
+> get hello
+(nil)
+```
+
+## 2. LIST
+
+
+
+```html
+> rpush list-key item
+(integer) 1
+> rpush list-key item2
+(integer) 2
+> rpush list-key item
+(integer) 3
+
+> lrange list-key 0 -1
+1) "item"
+2) "item2"
+3) "item"
+
+> lindex list-key 1
+"item2"
+
+> lpop list-key
+"item"
+
+> lrange list-key 0 -1
+1) "item2"
+2) "item"
+```
+
+## 3. SET
+
+
+
+```html
+> sadd set-key item
+(integer) 1
+> sadd set-key item2
+(integer) 1
+> sadd set-key item3
+(integer) 1
+> sadd set-key item
+(integer) 0
+
+> smembers set-key
+1) "item"
+2) "item2"
+3) "item3"
+
+> sismember set-key item4
+(integer) 0
+> sismember set-key item
+(integer) 1
+
+> srem set-key item2
+(integer) 1
+> srem set-key item2
+(integer) 0
+
+> smembers set-key
+1) "item"
+2) "item3"
+```
+
+## 4. HASH
+
+
+
+```html
+> hset hash-key sub-key1 value1
+(integer) 1
+> hset hash-key sub-key2 value2
+(integer) 1
+> hset hash-key sub-key1 value1
+(integer) 0
+
+> hgetall hash-key
+1) "sub-key1"
+2) "value1"
+3) "sub-key2"
+4) "value2"
+
+> hdel hash-key sub-key2
+(integer) 1
+> hdel hash-key sub-key2
+(integer) 0
+
+> hget hash-key sub-key1
+"value1"
+
+> hgetall hash-key
+1) "sub-key1"
+2) "value1"
+```
+
+## 5. ZSET
+
+
+
+```html
+> zadd zset-key 728 member1
+(integer) 1
+> zadd zset-key 982 member0
+(integer) 1
+> zadd zset-key 982 member0
+(integer) 0
+
+> zrange zset-key 0 -1 withscores
+1) "member1"
+2) "728"
+3) "member0"
+4) "982"
+
+> zrangebyscore zset-key 0 800 withscores
+1) "member1"
+2) "728"
+
+> zrem zset-key member1
+(integer) 1
+> zrem zset-key member1
+(integer) 0
+
+> zrange zset-key 0 -1 withscores
+1) "member0"
+2) "982"
+```
+
+# 三、键的过期时间
+
+Redis 可以为每个键设置过期时间,当键过期时,会自动删除该键。
+
+对于散列表这种容器,只能为整个键设置过期时间(整个散列表),而不能为键里面的单个元素设置过期时间。
+
+过期时间对于清理缓存数据非常有用。
+
+# 四、发布与订阅
+
+订阅者订阅了频道之后,发布者向频道发送字符串消息会被所有订阅者接收到。
+
+发布与订阅模式和观察者模式有以下不同:
+
+- 观察者模式中,观察者和主题都知道对方的存在;而在发布与订阅模式中,发布者与订阅者不知道对方的存在,它们之间通过频道进行通信。
+- 观察者模式是同步的,当事件触发时,主题会去调度观察者的方法;而发布与订阅模式是异步的;
+
+
+
+发布与订阅有一些问题,很少使用它,而是使用替代的解决方案。问题如下:
+
+1. 如果订阅者读取消息的速度很慢,会使得消息不断积压在发布者的输出缓存区中,造成内存占用过多;
+2. 如果订阅者在执行订阅的过程中网络出现问题,那么就会丢失断线期间发送的所有消息。
+
+# 五、事务
+
+Redis 最简单的事务实现方式是使用 MULTI 和 EXEC 命令将事务操作包围起来。
+
+MULTI 和 EXEC 中的操作将会一次性发送给服务器,而不是一条一条发送,这种方式称为流水线,它可以减少客户端与服务器之间的网络通信次数从而提升性能。
+
+# 六、持久化
+
+Redis 是内存型数据库,为了保证数据在断电后不会丢失,需要将内存中的数据持久化到硬盘上。
+
+## 1. 快照持久化
+
+将某个时间点的所有数据都存放到硬盘上。
+
+可以将快照复制到其它服务器从而创建具有相同数据的服务器副本。
+
+如果系统发生故障,将会丢失最后一次创建快照之后的数据。
+
+如果数据量很大,保存快照的时间会很长。
+
+## 2. AOF 持久化
+
+将写命令添加到 AOF 文件(Append Only File)的末尾。
+
+对硬盘的文件进行写入时,写入的内容首先会被存储到缓冲区,然后由操作系统决定什么时候将该内容同步到硬盘,用户可以调用 file.flush() 方法请求操作系统尽快将缓冲区存储的数据同步到硬盘。
+
+将写命令添加到 AOF 文件时,要根据需求来保证何时将添加的数据同步到硬盘上,有以下同步选项:
+
+| 选项 | 同步频率 |
+| :--: | :--: |
+| always | 每个写命令都同步 |
+| everysec | 每秒同步一次 |
+| no | 让操作系统来决定何时同步 |
+
+always 选项会严重减低服务器的性能;everysec 选项比较合适,可以保证系统奔溃时只会丢失一秒左右的数据,并且 Redis 每秒执行一次同步对服务器性能几乎没有任何影响;no 选项并不能给服务器性能带来多大的提升,而且也会增加系统奔溃时数据丢失的数量。
+
+随着服务器写请求的增多,AOF 文件会越来越大;Redis 提供了一种将 AOF 重写的特性,能够去除 AOF 文件中的冗余写命令。
+
+# 七、复制
+
+通过使用 slaveof host port 命令来让一个服务器成为另一个服务器的从服务器。
+
+一个从服务器只能有一个主服务器,并且不支持主主复制。
+
+## 从服务器连接主服务器的过程
+
+1. 主服务器创建快照文件,发送给从服务器,并在发送期间使用缓冲区记录执行的写命令。快照文件发送完毕之后,开始向从服务器发送存储在缓冲区中的写命令;
+
+2. 从服务器丢弃所有旧数据,载入主服务器发来的快照文件,之后从服务器开始接受主服务器发来的写命令;
+
+3. 主服务器每执行一次写命令,就向从服务器发送相同的写命令。
+
+## 主从链
+
+随着负载不断上升,主服务器可能无法很快地更新所有从服务器,或者重新连接和重新同步从服务器将导致系统超载。为了解决这个问题,可以创建一个中间层来分担主服务器的复制工作。中间层的服务器是最上层服务器的从服务器,又是最下层服务器的主服务器。
+
+
+
+# 八、处理故障
+
+要用到持久化文件来恢复服务器的数据。
+
+持久化文件可能因为服务器出错也有错误,因此要先对持久化文件进行验证和修复。对 AOF 文件就行验证和修复很容易,修复操作将第一个出错命令和其后的所有命令都删除;但是只能验证快照文件,无法对快照文件进行修复,因为快照文件进行了压缩,出现在快照文件中间的错误可能会导致整个快照文件的剩余部分无法读取。
+
+当主服务器出现故障时,Redis 常用的做法是新开一台服务器作为主服务器,具体步骤如下:假设 A 为主服务器,B 为从服务器,当 A 出现故障时,让 B 生成一个快照文件,将快照文件发送给 C,并让 C 恢复快照文件的数据。最后,让 B 成为 C 的从服务器。
+
+# 九、分片
+
+Redis 中的分片类似于 MySQL 的分表操作,分片是将数据划分为多个部分的方法,对数据的划分可以基于键包含的 ID、基于键的哈希值,或者基于以上两者的某种组合。通过对数据进行分片,用户可以将数据存储到多台机器里面,也可以从多台机器里面获取数据,这种方法在解决某些问题时可以获得线性级别的性能提升。
+
+假设有 4 个 Reids 实例 R0,R1,R2,R3,还有很多表示用户的键 user:1,user:2,... 等等,有不同的方式来选择一个指定的键存储在哪个实例中。最简单的方式是范围分片,例如用户 id 从 0\~1000 的存储到实例 R0 中,用户 id 从 1001\~2000 的存储到实例 R1 中,等等。但是这样需要维护一张映射范围表,维护操作代价很高。还有一种方式是哈希分片,使用 CRC32 哈希函数将键转换为一个数字,再对实例数量求模就能知道应该存储的实例。
+
+## 1. 客户端分片
+
+客户端使用一致性哈希等算法决定键应当分布到哪个节点。
+
+## 2. 代理分片
+
+将客户端请求发送到代理上,由代理转发请求到正确的节点上。
+
+## 3. 服务器分片
+
+Redis Cluster。
+
+# 十、事件
+
+## 事件类型
+
+### 1. 文件事件
+
+服务器有许多套接字,事件产生时会对这些套接字进行操作,服务器通过监听套接字来处理事件。常见的文件事件有:客户端的连接事件;客户端的命令请求事件;服务器向客户端返回命令结果的事件;
+
+### 2. 时间事件
+
+又分为两类:定时事件是让一段程序在指定的时间之内执行一次;周期性时间是让一段程序每隔指定时间就执行一次。
+
+## 事件的调度与执行
+
+服务器需要不断监听文件事件的套接字才能得到待处理的文件事件,但是不能监听太久,否则时间事件无法在规定的时间内执行,因此监听时间应该根据距离现在最近的时间事件来决定。
+
+事件调度与执行由 aeProcessEvents 函数负责,伪代码如下:
+
+```python
+def aeProcessEvents():
+
+ # 获取到达时间离当前时间最接近的时间事件
+ time_event = aeSearchNearestTimer()
+
+ # 计算最接近的时间事件距离到达还有多少毫秒
+ remaind_ms = time_event.when - unix_ts_now()
+
+ # 如果事件已到达,那么 remaind_ms 的值可能为负数,将它设为 0
+ if remaind_ms < 0:
+ remaind_ms = 0
+
+ # 根据 remaind_ms 的值,创建 timeval
+ timeval = create_timeval_with_ms(remaind_ms)
+
+ # 阻塞并等待文件事件产生,最大阻塞时间由传入的 timeval 决定
+ aeApiPoll(timeval)
+
+ # 处理所有已产生的文件事件
+ procesFileEvents()
+
+ # 处理所有已到达的时间事件
+ processTimeEvents()
+```
+
+将 aeProcessEvents 函数置于一个循环里面,加上初始化和清理函数,就构成了 Redis 服务器的主函数,伪代码如下:
+
+```python
+def main():
+
+ # 初始化服务器
+ init_server()
+
+ # 一直处理事件,直到服务器关闭为止
+ while server_is_not_shutdown():
+ aeProcessEvents()
+
+ # 服务器关闭,执行清理操作
+ clean_server()
+```
+
+从事件处理的角度来看,服务器运行流程如下:
+
+
+
+# 十一、Redis 与 Memcached 的区别
+
+两者都是非关系型内存键值数据库。有以下主要不同:
+
+## 数据类型
+
+Memcached 仅支持字符串类型,而 Redis 支持五种不同种类的数据类型,使得它可以更灵活地解决问题。
+
+## 数据持久化
+
+Redis 支持两种持久化策略:RDB 快照和 AOF 日志,而 Memcached 不支持持久化。
+
+## 分布式
+
+Memcached 不支持分布式,只能通过在客户端使用像一致性哈希这样的分布式算法来实现分布式存储,这种方式在存储和查询时都需要先在客户端计算一次数据所在的节点。
+
+Redis Cluster 实现了分布式的支持。
+
+## 内存管理机制
+
+在 Redis 中,并不是所有数据都一直存储在内存中,可以将一些很久没用的 value 交换到磁盘。而 Memcached 的数据则会一直在内存中。
+
+Memcached 将内存分割成特定长度的块来存储数据,以完全解决内存碎片的问题,但是这种方式会使得内存的利用率不高,例如块的大小为 128 bytes,只存储 100 bytes 的数据,那么剩下的 28 bytes 就浪费掉了。
+
+# 十二、Redis 适用场景
+
+## 缓存
+
+将热点数据放到内存中。
+
+## 消息队列
+
+List 类型是双向链表,很适合用于消息队列。
+
+## 计数器
+
+Redis 这种内存数据库能支持计数器频繁的读写操作。
+
+## 好友关系
+
+使用 Set 类型的交集操作很容易就可以知道两个用户的共同好友。
+
+# 十三、数据淘汰策略
+
+可以设置内存最大使用量,当内存使用量超过时施行淘汰策略,具体有 6 种淘汰策略。
+
+| 策略 | 描述 |
+| :--: | :--: |
+| volatile-lru | 从已设置过期时间的数据集中挑选最近最少使用的数据淘汰 |
+| volatile-ttl | 从已设置过期时间的数据集中挑选将要过期的数据淘汰 |
+|volatile-random | 从已设置过期时间的数据集中任意选择数据淘汰 |
+| allkeys-lru | 从所有数据集中挑选最近最少使用的数据淘汰 |
+| allkeys-random | 从所有数据集中任意选择数据进行淘汰 |
+| no-envicition | 禁止驱逐数据 |
+
+如果使用 Redis 来缓存数据时,要保证所有数据都是热点数据,可以将内存最大使用量设置为热点数据占用的内存量,然后启用 allkeys-lru 淘汰策略,将最近最少使用的数据淘汰。
+
+作为内存数据库,出于对性能和内存消耗的考虑,Redis 的淘汰算法(LRU、TTL)实际实现上并非针对所有 key,而是抽样一小部分 key 从中选出被淘汰 key。抽样数量可通过 maxmemory-samples 配置。
+
+# 十四、一个简单的论坛系统分析
+
+该论坛系统功能如下:
+
+- 可以发布文章;
+- 可以对文章进行点赞;
+- 在首页可以按文章的发布时间或者文章的点赞数进行排序显示;
+
+## 文章信息
+
+文章包括标题、作者、赞数等信息,在关系型数据库中很容易构建一张表来存储这些信息,在 Redis 中可以使用 HASH 来存储每种信息以及其对应的值的映射。
+
+Redis 没有关系型数据库中的表这一概念来将同类型的数据存放在一起,而是使用命名空间的方式来实现这一功能。键名的前面部分存储命名空间,后面部分的内容存储 ID,通常使用 : 来进行分隔。例如下面的 HASH 的键名为 article:92617,其中 article 为命名空间,ID 为 92617。
+
+
+
+## 点赞功能
+
+当有用户为一篇文章点赞时,除了要对该文章的 votes 字段进行加 1 操作,还必须记录该用户已经对该文章进行了点赞,防止用户点赞次数超过 1。可以建立文章的已投票用户集合来进行记录。
+
+为了节约内存,规定一篇文章发布满一周之后,就不能再对它进行投票,而文章的已投票集合也会被删除,可以为文章的已投票集合设置一个一周的过期时间就能实现这个规定。
+
+
+
+## 对文章进行排序
+
+为了按发布时间和点赞数进行排序,可以建立一个文章发布时间的有序集合和一个文章点赞数的有序集合。(下图中的 score 就是这里所说的点赞数;下面所示的有序集合分值并不直接是时间和点赞数,而是根据时间和点赞数间接计算出来的)
+
+
+
+# 参考资料
+
+- Carlson J L. Redis in Action[J]. Media.johnwiley.com.au, 2013.
+- 黄健宏. Redis 设计与实现 [M]. 机械工业出版社, 2014.
+- [REDIS IN ACTION](https://redislabs.com/ebook/foreword/)
+- [论述 Redis 和 Memcached 的差异](http://www.cnblogs.com/loveincode/p/7411911.html)
+- [Redis 3.0 中文版- 分片](http://wiki.jikexueyuan.com/project/redis-guide)
+- [Redis 应用场景](http://www.scienjus.com/redis-use-case/)
+- [Observer vs Pub-Sub](http://developers-club.com/posts/270339/)
diff --git a/notes/SQL 语法.md b/notes/SQL.md
similarity index 84%
rename from notes/SQL 语法.md
rename to notes/SQL.md
index 9345ea8d..d582dca2 100644
--- a/notes/SQL 语法.md
+++ b/notes/SQL.md
@@ -1,39 +1,32 @@
-* [基础](#基础)
-* [创建表](#创建表)
-* [插入](#插入)
-* [更新](#更新)
-* [删除](#删除)
-* [修改表](#修改表)
-* [查询](#查询)
-* [排序](#排序)
-* [过滤](#过滤)
-* [通配符](#通配符)
-* [计算字段](#计算字段)
-* [函数](#函数)
- * [文本处理](#文本处理)
- * [日期和时间处理](#日期和时间处理)
- * [数值处理](#数值处理)
- * [汇总](#汇总)
-* [分组](#分组)
-* [子查询](#子查询)
-* [连接](#连接)
- * [内连接](#内连接)
- * [自连接](#自连接)
- * [自然连接](#自然连接)
- * [外连接](#外连接)
-* [组合查询](#组合查询)
-* [视图](#视图)
-* [存储过程](#存储过程)
-* [游标](#游标)
-* [触发器](#触发器)
-* [事务处理](#事务处理)
-* [字符集](#字符集)
-* [权限管理](#权限管理)
+* [一、基础](#一基础)
+* [二、创建表](#二创建表)
+* [三、修改表](#三修改表)
+* [四、插入](#四插入)
+* [五、更新](#五更新)
+* [六、删除](#六删除)
+* [七、查询](#七查询)
+* [八、排序](#八排序)
+* [九、过滤](#九过滤)
+* [十、通配符](#十通配符)
+* [十一、计算字段](#十一计算字段)
+* [十二、函数](#十二函数)
+* [十三、分组](#十三分组)
+* [十四、子查询](#十四子查询)
+* [十五、连接](#十五连接)
+* [十六、组合查询](#十六组合查询)
+* [十七、视图](#十七视图)
+* [十八、存储过程](#十八存储过程)
+* [十九、游标](#十九游标)
+* [二十、触发器](#二十触发器)
+* [二十一、事务处理](#二十一事务处理)
+* [二十二、字符集](#二十二字符集)
+* [二十三、权限管理](#二十三权限管理)
+* [参考资料](#参考资料)
-# 基础
+# 一、基础
模式定义了数据如何存储、存储什么样的数据以及数据如何分解等信息,数据库和表都有模式。
@@ -53,7 +46,7 @@ FROM mytable; -- 注释
注释2 */
```
-# 创建表
+# 二、创建表
```sql
CREATE TABLE mytable (
@@ -64,16 +57,38 @@ CREATE TABLE mytable (
PRIMARY KEY (`id`));
```
-# 插入
+# 三、修改表
-**普通插入**
+添加列
+
+```sql
+ALTER TABLE mytable
+ADD col CHAR(20);
+```
+
+删除列
+
+```sql
+ALTER TABLE mytable
+DROP COLUMN col;
+```
+
+删除表
+
+```sql
+DROP TABLE mytable;
+```
+
+# 四、插入
+
+普通插入
```sql
INSERT INTO mytable(col1, col2)
VALUES(val1, val2);
```
-**插入检索出来的数据**
+插入检索出来的数据
```sql
INSERT INTO mytable1(col1, col2)
@@ -81,14 +96,14 @@ SELECT col1, col2
FROM mytable2;
```
-**将一个表的内容复制到一个新表**
+将一个表的内容插入到一个新表
```sql
CREATE TABLE newtable AS
SELECT * FROM mytable;
```
-# 更新
+# 五、更新
```sql
UPDATE mytable
@@ -96,7 +111,7 @@ SET col = val
WHERE id = 1;
```
-# 删除
+# 六、删除
```sql
DELETE FROM mytable
@@ -107,31 +122,9 @@ WHERE id = 1;
使用更新和删除操作时一定要用 WHERE 子句,不然会把整张表的数据都破坏。可以先用 SELECT 语句进行测试,防止错误删除。
-# 修改表
+# 七、查询
-**添加列**
-
-```sql
-ALTER TABLE mytable
-ADD col CHAR(20);
-```
-
-**删除列**
-
-```sql
-ALTER TABLE mytable
-DROP COLUMN col;
-```
-
-**删除表**
-
-```sql
-DROP TABLE mytable;
-```
-
-# 查询
-
-**DISTINCT**
+## DISTINCT
相同值只会出现一次。它作用于所有列,也就是说所有列的值都相同才算相同。
@@ -140,7 +133,7 @@ SELECT DISTINCT col1, col2
FROM mytable;
```
-**LIMIT**
+## LIMIT
限制返回的行数。可以有两个参数,第一个参数为起始行,从 0 开始;第二个参数为返回的总行数。
@@ -167,7 +160,7 @@ LIMIT 2, 3;
```
-# 排序
+# 八、排序
- **ASC** :升序(默认)
- **DESC** :降序
@@ -180,9 +173,9 @@ FROM mytable
ORDER BY col1 DESC, col2 ASC;
```
-# 过滤
+# 九、过滤
-不进行过滤的数据非常大,导致通过网络传输了很多多余的数据,从而浪费了网络带宽。因此尽量使用 SQL 语句来过滤不必要的数据,而不是传输所有的数据到客户端中然后由客户端进行过滤。
+不进行过滤的数据非常大,导致通过网络传输了多余的数据,从而浪费了网络带宽。因此尽量使用 SQL 语句来过滤不必要的数据,而不是传输所有的数据到客户端中然后由客户端进行过滤。
```sql
SELECT *
@@ -197,7 +190,7 @@ WHERE col IS NULL;
| = < > | 等于 小于 大于 |
| <> != | 不等于 |
| <= !> | 小于等于 |
-| >= !< | 大于等于 |
+| >= !< | 大于等于 |
| BETWEEN | 在两个值之间 |
| IS NULL | 为NULL值 |
@@ -209,13 +202,13 @@ WHERE col IS NULL;
**NOT** 操作符用于否定一个条件。
-# 通配符
+# 十、通配符
通配符也是用在过滤语句中,但它只能用于文本字段。
-- **%** 匹配 >=0 个任意字符,类似于 \*;
+- **%** 匹配 >=0 个任意字符;
-- **\_** 匹配 ==1 个任意字符,类似于 \.;
+- **\_** 匹配 ==1 个任意字符;
- **[ ]** 可以匹配集合内的字符,例如 [ab] 将匹配字符 a 或者 b。用脱字符 ^ 可以对其进行否定,也就是不匹配集合内的字符。
@@ -228,7 +221,7 @@ WHERE col LIKE '[^AB]%' -- 不以 A 和 B 开头的任意文本
```
不要滥用通配符,通配符位于开头处匹配会非常慢。
-# 计算字段
+# 十一、计算字段
在数据库服务器上完成数据的转换和格式化的工作往往比客户端上快得多,并且转换和格式化后的数据量更少的话可以减少网络通信量。
@@ -239,14 +232,14 @@ SELECT col1*col2 AS alias
FROM mytable
```
-**Concat()** 用于连接两个字段。许多数据库会使用空格把一个值填充为列宽,因此连接的结果会出现一些不必要的空格,使用 **TRIM()** 可以去除首尾空格。
+**CONCAT()** 用于连接两个字段。许多数据库会使用空格把一个值填充为列宽,因此连接的结果会出现一些不必要的空格,使用 **TRIM()** 可以去除首尾空格。
```sql
-SELECT Concat(TRIM(col1), ' (', TRIM(col2), ')')
+SELECT CONCAT(TRIM(col1), ' (', TRIM(col2), ')')
FROM mytable
```
-# 函数
+# 十二、函数
各个 DBMS 的函数都是不相同的,因此不可移植。
@@ -324,20 +317,20 @@ mysql> SELECT NOW();
AVG() 会忽略 NULL 行。
-使用 DISTINCT 可以汇总函数值汇总不同的值。
+使用 DISTINCT 可以让汇总函数值汇总不同的值。
```sql
SELECT AVG(DISTINCT col1) AS avg_col
FROM mytable
```
-# 分组
+# 十三、分组
分组就是把具有相同的数据值的行放在同一组中。
可以对同一分组数据使用汇总函数进行处理,例如求分组数据的平均值等。
-指定的分组字段除了能让数组按该字段进行分组,也可以按该字段进行排序,例如按 col 字段排序并分组数据:
+指定的分组字段除了能按该字段进行分组,也可以按该字段进行排序,例如按 col 字段排序并分组数据:
```sql
SELECT col, COUNT(*) AS num
@@ -345,6 +338,15 @@ FROM mytable
GROUP BY col;
```
+GROUP BY 是按照分组字段进行排序,ORDER BY 也可以以汇总字段来进行排序。
+
+```sql
+SELECT col, COUNT(*) AS num
+FROM mytable
+GROUP BY col
+ORDER BY num;
+```
+
WHERE 过滤行,HAVING 过滤分组。行过滤应当先与分组过滤;
```sql
@@ -355,33 +357,24 @@ GROUP BY col
HAVING COUNT(*) >= 2;
```
-GROUP BY 的排序结果为分组字段,而 ORDER BY 也可以以聚集字段来进行排序。
-
-```sql
-SELECT col, COUNT(*) AS num
-FROM mytable
-GROUP BY col
-ORDER BY num;
-```
-
分组规定:
1. GROUP BY 子句出现在 WHERE 子句之后,ORDER BY 子句之前;
-2. 除了汇总计算语句的字段外,SELECT 语句中的每一字段都必须在 GROUP BY 子句中给出;
+2. 除了汇总字段外,SELECT 语句中的每一字段都必须在 GROUP BY 子句中给出;
3. NULL 的行会单独分为一组;
4. 大多数 SQL 实现不支持 GROUP BY 列具有可变长度的数据类型。
-# 子查询
+# 十四、子查询
子查询中只能返回一个字段的数据。
可以将子查询的结果作为 WHRER 语句的过滤条件:
-```
+```sql
SELECT *
FROM mytable1
WHERE col1 IN (SELECT col2
- FROM mytable2);
+ FROM mytable2);
```
下面的语句可以检索出客户的订单数量,子查询语句会对第一个查询检索出的每个客户执行一次:
@@ -395,9 +388,9 @@ FROM Customers
ORDER BY cust_name;
```
-# 连接
+# 十五、连接
-连接用于连接多个表,使用 JOIN 关键字,并且条件语句使用 ON 而不是 Where。
+连接用于连接多个表,使用 JOIN 关键字,并且条件语句使用 ON 而不是 WHERE。
连接可以替换子查询,并且比子查询的效率一般会更快。
@@ -407,7 +400,7 @@ ORDER BY cust_name;
内连接又称等值连接,使用 INNER JOIN 关键字。
-```
+```sql
select a, b, c
from A inner join B
on A.key = B.key
@@ -415,7 +408,7 @@ on A.key = B.key
可以不明确使用 INNER JOIN,而使用普通查询并在 WHERE 中将两个表中要连接的列用等值方法连接起来。
-```
+```sql
select a, b, c
from A, B
where A.key = B.key
@@ -429,9 +422,9 @@ where A.key = B.key
一张员工表,包含员工姓名和员工所属部门,要找出与 Jim 处在同一部门的所有员工姓名。
-**子查询版本**
+子查询版本
-```
+```sql
select name
from employee
where department = (
@@ -440,10 +433,10 @@ where department = (
where name = "Jim");
```
-**自连接版本**
+自连接版本
-```
-select name
+```sql
+select e2.name
from employee as e1, employee as e2
where e1.department = e2.department
and e1.name = "Jim";
@@ -457,7 +450,7 @@ where e1.department = e2.department
内连接和自然连接的区别:内连接提供连接的列,而自然连接自动连接所有同名列。
-```
+```sql
select *
from employee natural join department;
```
@@ -468,15 +461,15 @@ from employee natural join department;
检索所有顾客的订单信息,包括还没有订单信息的顾客。
-```
+```sql
select Customers.cust_id, Orders.order_num
- from Customers left outer join Orders
- on Customers.cust_id = Orders.curt_id;
+from Customers left outer join Orders
+on Customers.cust_id = Orders.curt_id;
```
如果需要统计顾客的订单数,使用聚集函数。
-```
+```sql
select Customers.cust_id,
COUNT(Orders.order_num) as num_ord
from Customers left outer join Orders
@@ -484,7 +477,7 @@ on Customers.cust_id = Orders.curt_id
group by Customers.cust_id;
```
-# 组合查询
+# 十六、组合查询
使用 **UNION** 来组合两个查询,如果第一个查询返回 M 行,第二个查询返回 N 行,那么组合查询的结果为 M+N 行。
@@ -504,7 +497,7 @@ FROM mytable
WHERE col =2;
```
-# 视图
+# 十七、视图
视图是虚拟的表,本身不包含数据,也就不能对其进行索引操作。对视图的操作和对普通表的操作一样。
@@ -522,17 +515,17 @@ FROM mytable
WHERE col5 = val;
```
-# 存储过程
+# 十八、存储过程
存储过程可以看成是对一系列 SQL 操作的批处理;
-**使用存储过程的好处**
+## 使用存储过程的好处
1. 代码封装,保证了一定的安全性;
2. 代码复用;
3. 由于是预先编译,因此具有很高的性能。
-**创建存储过程**
+## 创建存储过程
命令行中创建存储过程需要自定义分隔符,因为命令行是以 ; 为结束符,而存储过程中也包含了分号,因此会错误把这部分分号当成是结束符,造成语法错误。
@@ -561,13 +554,13 @@ call myprocedure(@ret);
select @ret;
```
-# 游标
+# 十九、游标
在存储过程中使用游标可以对一个结果集进行移动遍历。
游标主要用于交互式应用,其中用户需要对数据集中的任意行进行浏览和修改。
-**使用游标的四个步骤:**
+使用游标的四个步骤:
1. 声明游标,这个过程没有实际检索出数据;
2. 打开游标;
@@ -597,7 +590,7 @@ create procedure myprocedure(out ret int)
delimiter ;
```
-# 触发器
+# 二十、触发器
触发器会在某个表执行以下语句时而自动执行:DELETE、INSERT、UPDATE
@@ -618,16 +611,16 @@ UPDATE 触发器包含一个名为 NEW 和一个名为 OLD 的虚拟表,其中
MySQL 不允许在触发器中使用 CALL 语句 ,也就是不能调用存储过程。
-# 事务处理
+# 二十一、事务处理
-**基本术语**
+基本术语:
1. 事务(transaction)指一组 SQL 语句;
2. 回退(rollback)指撤销指定 SQL 语句的过程;
3. 提交(commit)指将未存储的 SQL 语句结果写入数据库表;
4. 保留点(savepoint)指事务处理中设置的临时占位符(placeholder),你可以对它发布回退(与回退整个事务处理不同)。
-不能回退 SELECT 语句,回退 SELECT 语句也没意义;也不能回退 CRETE 和 DROP 语句。
+不能回退 SELECT 语句,回退 SELECT 语句也没意义;也不能回退 CREATE 和 DROP 语句。
MySQL 的事务提交默认是隐式提交,也就是每执行一条语句就把这条语句当成一个事务然后进行提交。当出现 START TRANSACTION 语句时,会关闭隐式提交;当 COMMIT 或 ROLLBACK 语句执行后,事务会自动关闭,重新恢复隐式提交。
@@ -645,9 +638,9 @@ ROLLBACK TO delete1
COMMIT
```
-# 字符集
+# 二十二、字符集
-**基本术语**
+基本术语:
1. 字符集为字母和符号的集合;
2. 编码为某个字符集成员的内部表示;
@@ -669,7 +662,7 @@ FROM mytable
ORDER BY col COLLATE latin1_general_ci;
```
-# 权限管理
+# 二十三、权限管理
MySQL 的账户信息保存在 mysql 这个数据库中。
@@ -678,7 +671,7 @@ USE mysql;
SELECT user FROM user;
```
-**创建账户**
+## 创建账户
```sql
CREATE USER myuser IDENTIFIED BY 'mypassword';
@@ -686,35 +679,33 @@ CREATE USER myuser IDENTIFIED BY 'mypassword';
新创建的账户没有任何权限。
-**修改账户名**
+## 修改账户名
```sql
RENAME myuser TO newuser;
```
-**删除账户**
+## 删除账户
```sql
DROP USER myuser;
```
-**查看权限**
+## 查看权限
```sql
SHOW GRANTS FOR myuser;
```
-**授予权限**
+## 授予权限
```sql
GRANT SELECT, INSERT ON mydatabase.* TO myuser;
```
-
-
账户用 username@host 的形式定义,username@% 使用的是默认主机名。
-**删除权限**
+## 删除权限
```sql
REVOKE SELECT, INSERT ON mydatabase.* FROM myuser;
@@ -728,7 +719,7 @@ GRANT 和 REVOKE 可在几个层次上控制访问权限:
- 特定的列;
- 特定的存储过程。
-**更改密码**
+## 更改密码
必须使用 Password() 函数
@@ -736,3 +727,6 @@ GRANT 和 REVOKE 可在几个层次上控制访问权限:
SET PASSWROD FOR myuser = Password('newpassword');
```
+# 参考资料
+
+- BenForta. SQL 必知必会 [M]. 人民邮电出版社, 2013.
diff --git a/notes/一致性协议.md b/notes/一致性协议.md
new file mode 100644
index 00000000..ecac4a1e
--- /dev/null
+++ b/notes/一致性协议.md
@@ -0,0 +1,157 @@
+
+* [一、两阶段提交协议](#一两阶段提交协议)
+* [二、Paxos 协议](#二paxos-协议)
+* [三、Raft 协议](#三raft-协议)
+* [四、拜占庭将军问题](#四拜占庭将军问题)
+* [五、参考资料](#五参考资料)
+
+
+
+# 一、两阶段提交协议
+
+Two-phase Commit(2PC)。
+
+可以保证一个事务跨越多个节点时保持 ACID 特性。
+
+两类节点:协调者(Coordinator)和参与者(Participants),协调者只有一个,参与者可以有多个。
+
+## 运行过程
+
+1. 准备阶段:协调者询问参与者事务是否执行成功;
+
+2. 提交阶段:如果事务在每个参与者上都执行成功,协调者发送通知让参与者提交事务;否则,协调者发送通知让参与者回滚事务。
+
+
+
+
+需要注意的是,在准备阶段,参与者执行了事务,但是还未提交。只有在提交阶段接收到协调者发来的通知后,才进行提交或者回滚。
+
+## 存在的问题
+
+- 参与者发生故障。解决方案:可以给事务设置一个超时时间,如果某个参与者一直不响应,那么认为事务执行失败。
+
+- 协调者发生故障。解决方案:将操作日志同步到备用协调者,让备用协调者接替后续工作。
+
+# 二、Paxos 协议
+
+用于达成共识性问题,即对多个节点产生的值,该算法能保证只选出唯一一个值。
+
+主要有三类节点:
+
+1. 提议者(Proposer):提议一个值;
+2. 接受者(Acceptor):对每个提议进行投票;
+3. 告知者(Learner):被告知投票的结果,不参与投票过程。
+
+
+
+## 执行过程
+
+规定一个提议包含两个字段:[n, v],其中 n 为序号(具有唯一性),v 为提议值。
+
+下图演示了两个 Proposer 和三个 Acceptor 的系统中运行该算法的初始过程,每个 Proposer 都会向所有 Acceptor 发送提议请求。
+
+
+
+当 Acceptor 接收到一个提议请求,包含的提议为 [n1, v1],并且之前还未接收过提议请求,那么发送一个提议响应,设置当前接收到的提议为 [n1, v1],并且保证以后不会再接受序号小于 n1 的提议。
+
+如下图,Acceptor X 在收到 [n=2, v=8] 的提议请求时,由于之前没有接收过提议,因此就发送一个 [no previous] 的提议响应,并且设置当前接收到的提议为 [n=2, v=8],并且保证以后不会再接受序号小于 2 的提议。其它的 Acceptor 类似。
+
+
+
+如果 Acceptor 接收到一个提议请求,包含的提议为 [n2, v2],并且之前已经接收过提议 [n1, v1]。如果 n1 > n2,那么就丢弃该提议请求;否则,发送提议响应,该提议响应包含之前已经接收过的提议 [n1, v1],设置当前接收到的提议为 [n2, v2],并且保证以后不会再接受序号小于 n2 的提议。
+
+如下图,Acceptor Z 收到 Proposer A 发来的 [n=2, v=8] 的提议请求,由于之前已经接收过 [n=4, v=5] 的提议,并且 n > 2,因此就抛弃该提议请求;Acceptor X 收到 Proposer B 发来的 [n=4, v=5] 的提议请求,因为之前接收到的提议为 [n=2, v=8],并且 2 <= 4,因此就发送 [n=2, v=8] 的提议响应,设置当前接收到的提议为 [n=4, v=5],并且保证以后不会再接受序号小于 4 的提议。Acceptor Y 类似。
+
+
+
+当一个 Proposer 接收到超过一半 Acceptor 的提议响应时,就可以发送接受请求。
+
+Proposer A 接收到两个提议响应之后,就发送 [n=2, v=8] 接受请求。该接受请求会被所有 Acceptor 丢弃,因为此时所有 Acceptor 都保证不接受序号小于 4 的提议。
+
+Proposer B 过后也收到了两个提议响应,因此也开始发送接受请求。需要注意的是,接受请求的 v 需要取它收到的最大 v 值,也就是 8。因此它发送 [n=4, v=8] 的接受请求。
+
+
+
+Acceptor 接收到接受请求时,如果序号大于等于该 Acceptor 承诺的最小序号,那么就发送通知给所有的 Learner。当 Learner 发现有大多数的 Acceptor 接收了某个提议,那么该提议的提议值就被 Paxos 选择出来。
+
+
+
+## 约束条件
+
+### 1. 正确性
+
+指只有一个提议值会生效。
+
+因为 Paxos 协议要求每个生效的提议被多数 Acceptor 接收,并且 Acceptor 不会接受两个不同的提议,因此可以保证正确性。
+
+### 2. 可终止性
+
+指最后总会有一个提议生效。
+
+Paxos 协议能够让 Proposer 发送的提议朝着能被大多数 Acceptor 接受的那个提议靠拢,因此能够保证可终止性。
+
+# 三、Raft 协议
+
+Raft 和 Paxos 类似,但是更容易理解,也更容易实现。
+
+Raft 主要是用来竞选主节点。
+
+## 单个 Candidate 的竞选
+
+有三种节点:Follower、Candidate 和 Leader。Leader 会周期性的发送心跳包给 Follower。每个 Follower 都设置了一个随机的竞选超时时间,一般为 150ms\~300ms,如果在这个时间内没有收到 Leader 的心跳包,就会变成 Candidate,进入竞选阶段。
+
+① 下图表示一个分布式系统的最初阶段,此时只有 Follower,没有 Leader。Follower A 等待一个随机的竞选超时时间之后,没收到 Leader 发来的心跳包,因此进入竞选阶段。
+
+
+
+② 此时 A 发送投票请求给其它所有节点。
+
+
+
+③ 其它节点会对请求进行回复,如果超过一半的节点回复了,那么该 Candidate 就会变成 Leader。
+
+
+
+④ 之后 Leader 会周期性地发送心跳包给 Follower,Follower 接收到心跳包,会重新开始计时。
+
+
+
+## 多个 Candidate 竞选
+
+① 如果有多个 Follower 成为 Candidate,并且所获得票数相同,那么就需要重新开始投票,例如下图中 Candidate B 和 Candidate D 都获得两票,因此需要重新开始投票。
+
+
+
+② 当重新开始投票时,由于每个节点设置的随机竞选超时时间不同,因此能下一次再次出现多个 Candidate 并获得同样票数的概率很低。
+
+
+
+## 日志复制
+
+① 来自客户端的修改都会被传入 Leader。注意该修改还未被提交,只是写入日志中。
+
+
+
+② Leader 会把修改复制到所有 Follower。
+
+
+
+③ Leader 会等待大多数的 Follower 也进行了修改,然后才将修改提交。
+
+
+
+④ 此时 Leader 会通知的所有 Follower 让它们也提交修改,此时所有节点的值达成一致。
+
+
+
+# 四、拜占庭将军问题
+
+> [拜占庭将军问题深入探讨](http://www.8btc.com/baizhantingjiangjun)
+
+# 五、参考资料
+
+- 杨传辉. 大规模分布式存储系统: 原理解析与架构实战[M]. 机械工业出版社, 2013.
+- [区块链技术指南](https://www.gitbook.com/book/yeasy/blockchain_guide/details)
+- [NEAT ALGORITHMS - PAXOS](http://harry.me/blog/2014/12/27/neat-algorithms-paxos/)
+- [Raft: Understandable Distributed Consensus](http://thesecretlivesofdata.com/raft)
+- [Paxos By Example](https://angus.nyc/2012/paxos-by-example/)
diff --git a/notes/代码可读性.md b/notes/代码可读性.md
index 0c5ab772..744b8bab 100644
--- a/notes/代码可读性.md
+++ b/notes/代码可读性.md
@@ -1,21 +1,22 @@
-* [可读性的重要性](#可读性的重要性)
-* [用名字表达代码含义](#用名字表达代码含义)
-* [名字不能带来歧义](#名字不能带来歧义)
-* [良好的代码风格](#良好的代码风格)
-* [编写注释](#编写注释)
-* [如何编写注释](#如何编写注释)
-* [提高控制流的可读性](#提高控制流的可读性)
-* [拆分长表达式](#拆分长表达式)
-* [变量与可读性](#变量与可读性)
-* [抽取函数](#抽取函数)
-* [一次只做一件事](#一次只做一件事)
-* [用自然语言表述代码](#用自然语言表述代码)
-* [减少代码量](#减少代码量)
+* [一、可读性的重要性](#一可读性的重要性)
+* [二、用名字表达代码含义](#二用名字表达代码含义)
+* [三、名字不能带来歧义](#三名字不能带来歧义)
+* [四、良好的代码风格](#四良好的代码风格)
+* [五、编写注释](#五编写注释)
+* [六、如何编写注释](#六如何编写注释)
+* [七、提高控制流的可读性](#七提高控制流的可读性)
+* [八、拆分长表达式](#八拆分长表达式)
+* [九、变量与可读性](#九变量与可读性)
+* [十、抽取函数](#十抽取函数)
+* [十一、一次只做一件事](#十一一次只做一件事)
+* [十二、用自然语言表述代码](#十二用自然语言表述代码)
+* [十三、减少代码量](#十三减少代码量)
+* [参考资料](#参考资料)
-# 可读性的重要性
+# 一、可读性的重要性
编程有很大一部分时间是在阅读代码,不仅要阅读自己的代码,而且要阅读别人的代码。因此,可读性良好的代码能够大大提高编程效率。
@@ -23,7 +24,7 @@
只有在核心领域为了效率才可以放弃可读性,否则可读性是第一位。
-# 用名字表达代码含义
+# 二、用名字表达代码含义
一些比较有表达力的单词:
@@ -38,7 +39,7 @@
为名字添加形容词等信息能让名字更具有表达力,但是名字也会变长。名字长短的准则是:作用域越大,名字越长。因此只有在短作用域才能使用一些简单名字。
-# 名字不能带来歧义
+# 三、名字不能带来歧义
起完名字要思考一下别人会对这个名字有何解读,会不会误解了原本想表达的含义。
@@ -48,13 +49,13 @@
布尔相关的命名加上 is、can、should、has 等前缀。
-# 良好的代码风格
+# 四、良好的代码风格
适当的空行和缩进。
排列整齐的注释:
-```
+```java
int a = 1; // 注释
int b = 11; // 注释
int c = 111; // 注释
@@ -64,7 +65,7 @@ int c = 111; // 注释
把相关的代码按块组织起来放在一起。
-# 编写注释
+# 五、编写注释
阅读代码首先会注意到注释,如果注释没太大作用,那么就会浪费代码阅读的时间。那些能直接看出含义的代码不需要写注释,特别是并不需要为每个方法都加上注释,比如那些简单的 getter 和 setter 方法,为这些方法写注释反而让代码可读性更差。
@@ -83,24 +84,24 @@ int c = 111; // 注释
|HACH| 粗糙的解决方案 |
|XXX| 危险!这里有重要的问题 |
-# 如何编写注释
+# 六、如何编写注释
尽量简洁明了:
-```
+```java
// The first String is student's name
// The Second Integer is student's score
Map scoreMap = new HashMap<>();
```
-```
+```java
// Student' name -> Student's score
Map scoreMap = new HashMap<>();
```
添加测试用例来说明:
-```
+```java
//...
// Example: add(1, 2), return 3
int add(int x, int y) {
@@ -110,7 +111,7 @@ int add(int x, int y) {
在很复杂的函数调用中对每个参数标上名字:
-```
+```java
int a = 1;
int b = 2;
int num = add(\* x = *\ a, \* y = *\ b);
@@ -118,17 +119,18 @@ int num = add(\* x = *\ a, \* y = *\ b);
使用专业名词来缩短概念上的解释,比如用设计模式名来说明代码。
-# 提高控制流的可读性
+# 七、提高控制流的可读性
条件表达式中,左侧是变量,右侧是常数。比如下面第一个语句正确:
-```
+```java
if(len < 10)
if(10 > len)
```
if / else 条件语句,逻辑的处理顺序为:① 正逻辑;② 关键逻辑;③ 简单逻辑。
-```
+
+```java
if(a == b) {
// 正逻辑
} else{
@@ -144,15 +146,15 @@ do / while 的条件放在后面,不够简单明了,并且会有一些迷惑
在嵌套的循环中,用一些 return 语句往往能减少嵌套的层数。
-# 拆分长表达式
+# 八、拆分长表达式
长表达式的可读性很差,可以引入一些解释变量从而拆分表达式:
-```
+```python
if line.split(':')[0].strip() == "root":
...
```
-```
+```python
username = line.split(':')[0].strip()
if username == "root":
...
@@ -160,22 +162,22 @@ if username == "root":
使用摩根定理简化一些逻辑表达式:
-```
+```java
if(!a && !b) {
...
}
```
-```
-if(a || b) {
+```java
+if(!(a || b)) {
...
}
```
-# 变量与可读性
+# 九、变量与可读性
**去除控制流变量** 。在循环中通过使用 break 或者 return 可以减少控制流变量的使用。
-```
+```java
boolean done = false;
while(/* condition */ && !done) {
...
@@ -198,7 +200,7 @@ while(/* condition */) {
JavaScript 可以用闭包减小作用域。以下代码中 submit_form 是函数变量,submitted 变量控制函数不会被提交两次。第一个实现中 submitted 是全局变量,第二个实现把 submitted 放到匿名函数中,从而限制了起作用域范围。
-```
+```js
submitted = false;
var submit_form = function(form_name) {
if(submitted) {
@@ -208,7 +210,7 @@ var submit_form = function(form_name) {
};
```
-```
+```js
var submit_form = (function() {
var submitted = false;
return function(form_name) {
@@ -228,7 +230,7 @@ JavaScript 中没有用 var 声明的变量都是全局变量,而全局变量
在一个网页中有以下文本输入字段:
-```
+```html
@@ -237,7 +239,7 @@ JavaScript 中没有用 var 声明的变量都是全局变量,而全局变量
现在要接受一个字符串并把它放到第一个空的 input 字段中,初始实现如下:
-```
+```js
var setFirstEmptyInput = function(new_alue) {
var found = false;
var i = 1;
@@ -261,7 +263,7 @@ var setFirstEmptyInput = function(new_alue) {
- elem 作用域过大;
- 可以用 for 循环代替 while 循环;
-```
+```js
var setFirstEmptyInput = function(new_value) {
for(var i = 1; true; i++) {
var elem = document.getElementById('input' + i);
@@ -276,7 +278,7 @@ var setFirstEmptyInput = function(new_value) {
};
```
-# 抽取函数
+# 十、抽取函数
工程学就是把大问题拆分成小问题再把这些问题的解决方案放回一起。
@@ -284,7 +286,7 @@ var setFirstEmptyInput = function(new_value) {
介绍性的代码:
-```
+```java
int findClostElement(int[] arr) {
int clostIdx;
int clostDist = Interger.MAX_VALUE;
@@ -305,7 +307,7 @@ int findClostElement(int[] arr) {
以上代码中循环部分主要计算距离,这部分不属于代码高层次目标,高层次目标是寻找最小距离的值,因此可以把这部分代替提取到独立的函数中。这样做也带来一个额外的好处有:可以单独进行测试、可以快速找到程序错误并修改。
-```
+```java
public int findClostElement(int[] arr) {
int clostIdx;
int clostDist = Interger.MAX_VALUE;
@@ -324,18 +326,22 @@ public int findClostElement(int[] arr) {
函数抽取也用于减小代码的冗余。
-# 一次只做一件事
+# 十一、一次只做一件事
只做一件事的代码很容易让人知道其要做的事;
基本流程:列出代码所做的所有任务;把每个任务拆分到不同的函数,或者不同的段落。
-# 用自然语言表述代码
+# 十二、用自然语言表述代码
先用自然语言书写代码逻辑,也就是伪代码,然后再写代码,这样代码逻辑会更清晰。
-# 减少代码量
+# 十三、减少代码量
不要过度设计,编码过程会有很多变化,过度设计的内容到最后往往是无用的。
多用标准库实现。
+
+# 参考资料
+
+- Dustin, Boswell, Trevor, 等. 编写可读代码的艺术 [M]. 机械工业出版社, 2012.
diff --git a/notes/分布式基础.md b/notes/分布式基础.md
new file mode 100644
index 00000000..f7f77498
--- /dev/null
+++ b/notes/分布式基础.md
@@ -0,0 +1,208 @@
+
+* [一、基本概念](#一基本概念)
+ * [异常](#异常)
+ * [超时](#超时)
+ * [衡量指标](#衡量指标)
+* [二、数据分布](#二数据分布)
+ * [哈希分布](#哈希分布)
+ * [顺序分布](#顺序分布)
+* [三、负载均衡](#三负载均衡)
+* [四、复制](#四复制)
+ * [强同步复制协议](#强同步复制协议)
+ * [异步复制协议](#异步复制协议)
+* [五、CAP](#五cap)
+* [六、BASE](#六base)
+ * [基本可用](#基本可用)
+ * [软状态](#软状态)
+ * [最终一致性](#最终一致性)
+* [七、容错](#七容错)
+ * [故障检测](#故障检测)
+ * [故障恢复](#故障恢复)
+* [八、CDN 架构](#八cdn-架构)
+* [参考资料](#参考资料)
+
+
+
+# 一、基本概念
+
+## 异常
+
+### 1. 服务器宕机
+
+内存错误、服务器停电等都会导致服务器宕机,此时节点无法正常工作,称为不可用。
+
+服务器宕机会导致节点失去所有内存信息,因此需要将内存信息保存到持久化介质上。
+
+### 2. 网络异常
+
+有一种特殊的网络异常称为 **网络分区** ,即集群的所有节点被划分为多个区域,每个区域内部可以通信,但是区域之间无法通信。
+
+### 3. 磁盘故障
+
+磁盘故障是一种发生概率很高的异常。
+
+使用冗余机制,将数据存储到多台服务器。
+
+## 超时
+
+在分布式系统中,一个请求除了成功和失败两种状态,还存在着超时状态。
+
+
+
+可以将服务器的操作设计为具有 **幂等性** ,即执行多次的结果与执行一次的结果相同。如果使用这种方式,当出现超时的时候,可以不断地重新请求直到成功。
+
+## 衡量指标
+
+### 1. 性能
+
+常见的性能指标有:吞吐量、响应时间。
+
+其中,吞吐量指系统在某一段时间可以处理的请求总数,通常为每秒的读操作数或者写操作数;响应时间指从某个请求发出到接收到返回结果消耗的时间。
+
+这两个指标往往是矛盾的,追求高吞吐的系统,往往很难做到低响应时间,解释如下:
+
+- 在无并发的系统中,吞吐量为响应时间的倒数,例如响应时间为 10 ms,那么吞吐量为 100 req/s,因此高吞吐也就意味着低响应时间。
+
+- 但是在并发的系统中,由于一个请求在调用 I/O 资源的时候,需要进行等待。服务器端一般使用的是异步等待方式,即等待的请求被阻塞之后不需要一直占用 CPU 资源。这种方式能大大提高 CPU 资源的利用率,例如上面的例子中,单个请求在无并发的系统中响应时间为 10 ms,如果在并发的系统中,那么吞吐量将大于 100 req/s。因此为了追求高吞吐量,通常会提高并发程度。但是并发程度的增加,会导致请求的平均响应时间也增加,因为请求不能马上被处理,需要和其它请求一起进行并发处理,响应时间自然就会增高。
+
+### 2. 可用性
+
+可用性指系统在面对各种异常时可以提供正常服务的能力。可以用系统可用时间占总时间的比值来衡量,4 个 9 的可用性表示系统 99.99% 的时间是可用的。
+
+### 3. 一致性
+
+可以从两个角度理解一致性:从客户端的角度,读写操作是否满足某种特性;从服务器的角度,多个数据副本之间是否一致。
+
+有以下三种一致性模型:
+
+1. 强一致性:新数据写入之后,在任何数据副本上都能读取到最新值;
+2. 弱一致性:新数据写入之后,不能保证在数据副本上能读取到最新值;
+3. 最终一致性:新数据写入之后,只能保证过了一个时间窗口后才能在数据副本上读取到最新值;
+
+### 4. 可扩展性
+
+指系统通过扩展集群服务器规模来提高性能的能力。理想的分布式系统需要实现“线性可扩展”,即随着集群规模的增加,系统的整体性能也会线性增加。
+
+# 二、数据分布
+
+分布式系统的数据分布在多个节点中,常用的数据分布方式有哈希分布和顺序分布。
+
+## 哈希分布
+
+哈希分布就是将数据计算哈希值之后,按照哈希值分配到不同的节点上。例如有 N 个节点,数据的主键为 key,则将该数据分配的节点序号为:hash(key)%N。
+
+传统的哈希分布算法存在一个问题:当节点数量变化时,也就是 N 值变化,那么几乎所有的数据都需要重新分布,将导致大量的数据迁移。
+
+**一致性哈希**
+
+Distributed Hash Table(DHT):对于哈希空间 [0, 2n -1],将该哈希空间看成一个哈希环,将每个节点都配置到哈希环上。每个数据对象通过哈希取模得到哈希值之后,存放到哈希环中顺时针方向第一个大于等于该哈希值的节点上。
+
+
+
+一致性哈希的优点是在加入或者删除节点时只会影响到哈希环中相邻的节点,例如下图中新增节点 X,只需要将数据对象 C 重新存放到节点 X 上即可,对于节点 A、B、D 都没有影响。
+
+
+
+## 顺序分布
+
+哈希分布式破坏了数据的有序性,顺序分布则不会。
+
+顺序分布的数据划分为多个连续的部分,按一定策略分布到不同节点上。例如下图中,User 表的主键范围为 1 \~ 7000,使用顺序分布可以将其划分成多个子表,对应的主键范围为 1 \~ 1000,1001 \~ 2000,...,6001 \~ 7000。
+
+引入 Meta 表是为了支持更大的集群规模,它将原来的一层索引结分成两层,Meta 维护着 User 子表所在的节点,从而减轻 Root 节点的负担。
+
+
+
+# 三、负载均衡
+
+衡量负载的因素很多,如 CPU、内存、磁盘等资源使用情况、读写请求数等。分布式系统应当能够自动负载均衡,当某个节点的负载较高,将它的部分数据迁移到其它节点。
+
+每个集群都有一个总控节点,其它节点为工作节点,由总控节点根据全局负载信息进行整体调度,工作节点定时发送心跳包(Heartbeat)将节点负载相关的信息发送给总控节点。
+
+一个新上线的工作节点,由于其负载较低,如果不加控制,总控节点会将大量数据同时迁移到该节点上,造成该节点一段时间内无法工作。因此负载均衡操作需要平滑进行,新加入的节点需要较长的一段时间来达到比较均衡的状态。
+
+# 四、复制
+
+复制是保证分布式系统高可用的基础,让一个数据存储多个副本,当某个副本所在的节点出现故障时,能够自动切换到其它副本上,从而实现故障恢复。
+
+多个副本通常有一个为主副本,其它为备副本。主副本用来处理写请求,备副本主要用来处理读请求,实现读写分离。主副本将同步操作日志发送给备副本,备副本通过回放操作日志获取最新修改。
+
+
+
+主备副本之间有两种复制协议,一种是强同步复制协议,一种是异步复制协议。
+
+## 强同步复制协议
+
+要求主副本将同步操作日志发给备副本之后进行等待,要求至少一个备副本返回成功后,才开始修改主副本,修改完成之后通知客户端操作成功。
+
+优点:至少有一个备副本拥有完整的数据,出现故障时可以安全地切换到该备副本,因此一致性好。
+
+缺点:可用性差,因为主副本需要等待,那么整个分布式系统的可用时间就会降低。
+
+## 异步复制协议
+
+主副本将同步操作日志发给备副本之后不需要进行等待,直接修改主副本并通知客户端操作成功。
+
+优点:可用性好。
+
+缺点:一致性差。
+
+# 五、CAP
+
+分布式系统不可能同时满足一致性(C:Consistency)、可用性(A:Availability)和分区容忍性(P:Partition tolerance),最多只能同时满足其中两项。这三个概念上文中已经提到。
+
+在设计分布式系统时,需要根据实际需求弱化某一要求。因此就有了下图中的三种设计:CA、CP 和 AP。
+
+
+
+需要注意的是,分区容忍性必不可少,因为需要总是假设网络是不可靠的。因此实际上设计分布式系统需要在一致性和可用性之间做权衡。
+
+# 六、BASE
+
+BASE 是 Basically Available(基本可用)、Soft State(软状态)和 Eventually Consistent(最终一致性)三个短语的缩写。BASE 理论是对 CAP 中一致性和可用性权衡的结果,是基于 CAP 定理逐步演化而来的。BASE 理论的核心思想是:即使无法做到强一致性,但每个应用都可以根据自身业务特点,采用适当的方式来使系统达到最终一致性。
+
+
+
+## 基本可用
+
+指分布式系统在出现故障的时候,保证核心可用,允许损失部分可用性。
+
+例如,电商在做促销时,服务层可能只提供降级服务,部分用户可能会被引导到降级页面上。
+
+## 软状态
+
+指允许系统存在中间状态,而该中间状态不会影响系统整体可用性,即不同节点的数据副本之间进行同步的过程允许存在延时。
+
+## 最终一致性
+
+指所有的数据副本,在经过一段时间的同步之后,最终都能够达到一致的状态。
+
+强一致性需要保证数据副本实时一致,而最终一致性只需要保证过一段时间是一致的。
+
+ACID 是传统数据库系统常用的设计理论,追求强一致性模型。BASE 常用于大型分布式系统,只需要保证最终一致性。在实际的分布式场景中,不同业务单元和组件对一致性的要求是不同的,因此 ACID 和 BASE 往往会结合在一起使用。
+
+# 七、容错
+
+分布式系统故障发生的概率很大,为了实现高可用以及减少人工运维成本,需要实现自动化容错。
+
+## 故障检测
+
+通过 **租约机制** 来对故障进行检测。假设节点 A 为主控节点,节点 A 向节点 B 发送租约,节点 B 在租约规定的期限内才能提供服务。期限快到达时,节点 B 需要向 A 重新申请租约。
+
+如果过期,那么 B 不再提供服务,并且 A 也能知道 B 此时可能发生故障并已经停止服务。可以看到,通过这种机制,A 和 B 都能对 B 发生故障这一事实达成一致。
+
+## 故障恢复
+
+当某个节点故障时,就将它上面的服务迁移到其它节点。
+
+# 八、CDN 架构
+
+通过将内容发布到靠近用户的边缘节点,使不同地域的用户在访问相同网页时可以就近获取。不仅可以减轻服务器的负担,也可以提高用户的访问速度。
+
+从下图可以看出,DNS 在对域名解析时不再向用户返回源服务器的 IP 地址,而是返回边缘节点的 IP 地址,所以用户最终访问的是边缘节点。边缘节点会先从源服务器中获取用户所需的数据,如果请求成功,边缘节点会将页面缓存下来,下次用户访问时可以直接读取。
+
+
+
+# 参考资料
+
+- 杨传辉. 大规模分布式存储系统: 原理解析与架构实战[M]. 机械工业出版社, 2013.
diff --git a/notes/分布式问题分析.md b/notes/分布式问题分析.md
new file mode 100644
index 00000000..b9064a50
--- /dev/null
+++ b/notes/分布式问题分析.md
@@ -0,0 +1,329 @@
+
+* [一、谈谈业务中使用分布式的场景](#一谈谈业务中使用分布式的场景)
+* [二、分布式事务](#二分布式事务)
+ * [产生原因](#产生原因)
+ * [应用场景](#应用场景)
+ * [解决方案](#解决方案)
+* [三、负载均衡的算法与实现](#三负载均衡的算法与实现)
+ * [算法](#算法)
+ * [实现](#实现)
+* [四、分布式锁](#四分布式锁)
+ * [使用场景](#使用场景)
+ * [实现方式](#实现方式)
+* [五、分布式 Session](#五分布式-session)
+ * [1. Sticky Sessions](#1-sticky-sessions)
+ * [2. Session Replication](#2-session-replication)
+ * [3. Persistent DataStore](#3-persistent-datastore)
+ * [4. In-Memory DataStore](#4-in-memory-datastore)
+* [六、分库与分表带来的分布式困境与应对之策](#六分库与分表带来的分布式困境与应对之策)
+ * [事务问题](#事务问题)
+ * [查询问题](#查询问题)
+ * [ID 唯一性](#id-唯一性)
+* [参考资料](#参考资料)
+
+
+
+# 一、谈谈业务中使用分布式的场景
+
+分布式主要是为了提供可扩展性以及高可用性,业务中使用分布式的场景主要有分布式存储以及分布式计算。
+
+分布式存储中可以将数据分片到多个节点上,不仅可以提高性能(可扩展性),同时也可以使用多个节点对同一份数据进行备份(高可用性)。
+
+至于分布式计算,就是将一个大的计算任务分解成小任务分配到多个节点上去执行,再汇总每个小任务的执行结果得到最终结果。MapReduce 是分布式计算最好的例子。
+
+# 二、分布式事务
+
+指事务的操作位于不同的节点上,需要保证事务的 AICD 特性。
+
+## 产生原因
+
+- 数据库分库分表;
+- SOA 架构,比如一个电商网站将订单业务和库存业务分离出来放到不同的节点上。
+
+## 应用场景
+
+- 下单:减少库存、更新订单状态。库存和订单如果不在同一个数据库,就涉及分布式事务。
+- 支付:买家账户扣款、卖家账户入账。买家和卖家账户信息如果不在同一个数据库,就涉及分布式事务。
+
+## 解决方案
+
+### 1. 两阶段提交协议
+
+> [两阶段提交](https://github.com/CyC2018/Interview-Notebook/blob/master/notes/%E4%B8%80%E8%87%B4%E6%80%A7%E5%8D%8F%E8%AE%AE.md#%E4%B8%A4%E9%98%B6%E6%AE%B5%E6%8F%90%E4%BA%A4%E5%8D%8F%E8%AE%AE)
+
+两阶段提交协议可以很好地解决分布式事务问题。它可以使用 XA 来实现,XA 包含两个部分:事务管理器和本地资源管理器。其中本地资源管理器往往由数据库实现,比如 Oracle、DB2 这些商业数据库都实现了 XA 接口;而事务管理器作为全局的协调者,负责各个本地资源的提交和回滚。
+
+### 2. 消息中间件
+
+消息中间件也可称作消息系统 (MQ),它本质上是一个暂存转发消息的一个中间件。在分布式应用当中,我们可以把一个业务操作转换成一个消息,比如支付宝的余额转入余额宝操作,支付宝系统执行减少余额操作之后向消息系统发送一个消息,余额宝系统订阅这条消息然后进行增加余额宝操作。
+
+#### 2.1 消息处理模型
+
+(一)消息队列
+
+
+
+(二)发布/订阅
+
+
+
+
+#### 2.2 消息的可靠性
+
+(一)发送端的可靠性
+
+发送端完成操作后一定能将消息成功发送到消息系统。
+
+实现方法:在本地数据库建一张消息表,将消息数据与业务数据保存在同一数据库实例里,这样就可以利用本地数据库的事务机制。事务提交成功后,将消息表中的消息转移到消息中间件,若转移消息成功则删除消息表中的数据,否则继续重传。
+
+(二)接收端的可靠性
+
+接收端能够从消息中间件成功消费一次消息。
+
+实现方法:
+
+- 保证接收端处理消息的业务逻辑具有幂等性:只要具有幂等性,那么消费多少次消息,最后处理的结果都是一样的。
+- 保证消息具有唯一编号,并使用一张日志表来记录已经消费的消息编号。
+
+# 三、负载均衡的算法与实现
+
+## 算法
+
+### 1. 轮询(Round Robin)
+
+轮询算法把每个请求轮流发送到每个服务器上。下图中,一共有 6 个客户端产生了 6 个请求,这 6 个请求按 (1, 2, 3, 4, 5, 6) 的顺序发送。最后,(1, 3, 5) 的请求会被发送到服务器 1,(2, 4, 6) 的请求会被发送到服务器 2。
+
+
+
+该算法比较适合每个服务器的性能差不多的场景,如果有性能存在差异的情况下,那么性能较差的服务器可能无法承担多大的负载(下图的 Server 2)。
+
+
+
+### 2. 加权轮询(Weighted Round Robbin)
+
+加权轮询是在轮询的基础上,根据服务器的性能差异,为服务器赋予一定的权值。例如下图中,服务器 1 被赋予的权值为 5,服务器 2 被赋予的权值为 1,那么 (1, 2, 3, 4, 5) 请求会被发送到服务器 1,(6) 请求会被发送到服务器 2。
+
+
+
+### 3. 最少连接(least Connections)
+
+由于每个请求的连接时间不一样,使用轮询或者加权轮询算法的话,可能会让一台服务器当前连接数多大,而另一台服务器的连接多小,造成负载不均衡。例如下图中,(1, 3, 5) 请求会被发送到服务器 1,但是 (1, 3) 很快就断开连接,此时只有 (5) 请求连接服务器 1;(2, 4, 6) 请求被发送到服务器 2,只有 (2) 的连接断开。该系统继续运行时,服务器 2 会承担多大的负载。
+
+
+
+最少连接算法就是将请求发送给当前最少连接数的服务器上。例如下图中,服务器 1 当前连接数最小,那么新到来的请求 6 就会被发送到服务器 1 上。
+
+
+
+### 4. 加权最小连接(Weighted Least Connection)
+
+在最小连接的基础上,根据服务器的性能为每台服务器分配权重,再根据权重计算出每台服务器能处理的连接数。
+
+
+
+### 5. 随机算法(Random)
+
+把请求随机发送到服务器上。和轮询算法类似,该算法比较适合服务器性能差不多的场景。
+
+
+
+### 6. 源地址哈希法 (IP Hash)
+
+源地址哈希通过对客户端 IP 哈希计算得到的一个数值,用该数值对服务器数量进行取模运算,取模结果便是目标服务器的序号。
+
+- 优点:保证同一 IP 的客户端都会被 hash 到同一台服务器上。
+- 缺点:不利于集群扩展,后台服务器数量变更都会影响 hash 结果。可以采用一致性 Hash 改进。
+
+
+
+## 实现
+
+### 1. HTTP 重定向
+
+HTTP 重定向负载均衡服务器收到 HTTP 请求之后会返回服务器的地址,并将该地址写入 HTTP 重定向响应中返回给浏览器,浏览器收到后需要再次发送请求。
+
+缺点:
+
+- 用户访问的延迟会增加;
+- 如果负载均衡器宕机,就无法访问该站点。
+
+
+
+### 2. DNS 重定向
+
+使用 DNS 作为负载均衡器,根据负载情况返回不同服务器的 IP 地址。大型网站基本使用了这种方式做为第一级负载均衡手段,然后在内部使用其它方式做第二级负载均衡。
+
+缺点:
+
+- DNS 查找表可能会被客户端缓存起来,那么之后的所有请求都会被重定向到同一个服务器。
+
+
+
+### 3. 修改 MAC 地址
+
+使用 LVS(Linux Virtual Server)这种链路层负载均衡器,根据负载情况修改请求的 MAC 地址。
+
+
+
+### 4. 修改 IP 地址
+
+在网络层修改请求的目的 IP 地址。
+
+
+
+### 5. 代理自动配置
+
+正向代理与反向代理的区别:
+
+- 正向代理:发生在客户端,是由用户主动发起的。比如翻墙,客户端通过主动访问代理服务器,让代理服务器获得需要的外网数据,然后转发回客户端。
+- 反向代理:发生在服务器端,用户不知道代理的存在。
+
+PAC 服务器是用来判断一个请求是否要经过代理。
+
+
+
+# 四、分布式锁
+
+Java 提供了两种内置的锁的实现,一种是由 JVM 实现的 synchronized 和 JDK 提供的 Lock,对于单机单进程应用,可以使用它们来实现锁。当应用涉及到多机、多进程共同完成时,那么这时候就需要一个全局锁来实现多个进程之间的同步。
+
+## 使用场景
+
+在服务器端使用分布式部署的情况下,一个服务可能分布在不同的节点上,比如订单服务分布在节点 A 和节点 B 上。如果多个客户端同时对一个服务进行请求时,就需要使用分布式锁。例如一个服务可以使用 APP 端或者 Web 端进行访问,如果一个用户同时使用 APP 端和 Web 端访问该服务,并且 APP 端的请求路由到了节点 A,WEB 端的请求被路由到了节点 B,这时候就需要使用分布式锁来进行同步。
+
+## 实现方式
+
+### 1. 数据库分布式锁
+
+**(一)基于 MySQL 锁表**
+
+该实现完全依靠数据库的唯一索引。当想要获得锁时,就向数据库中插入一条记录,释放锁时就删除这条记录。如果记录具有唯一索引,就不会同时插入同一条记录。
+
+这种方式存在以下几个问题:
+
+1. 锁没有失效时间,解锁失败会导致死锁,其他线程无法再获得锁。
+2. 只能是非阻塞锁,插入失败直接就报错了,无法重试。
+3. 不可重入,同一线程在没有释放锁之前无法再获得锁。
+
+**(二)采用乐观锁增加版本号**
+
+根据版本号来判断更新之前有没有其他线程更新过,如果被更新过,则获取锁失败。
+
+### 2. Redis 分布式锁
+
+**(一)基于 SETNX、EXPIRE**
+
+使用 SETNX(set if not exist)命令插入一个键值对时,如果 Key 已经存在,那么会返回 False,否则插入成功并返回 True。因此客户端在尝试获得锁时,先使用 SETNX 向 Redis 中插入一个记录,如果返回 True 表示获得锁,返回 False 表示已经有客户端占用锁。
+
+EXPIRE 可以为一个键值对设置一个过期时间,从而避免了死锁的发生。
+
+**(二)RedLock 算法**
+
+RedLock 算法使用了多个 Redis 实例来实现分布式锁,这是为了保证在发生单点故障时还可用。
+
+1. 尝试从 N 个相互独立 Redis 实例获取锁,如果一个实例不可用,应该尽快尝试下一个。
+2. 计算获取锁消耗的时间,只有当这个时间小于锁的过期时间,并且从大多数(N/2+1)实例上获取了锁,那么就认为锁获取成功了。
+3. 如果锁获取失败,会到每个实例上释放锁。
+
+### 3. Zookeeper 分布式锁
+
+Zookeeper 是一个为分布式应用提供一致性服务的软件,例如配置管理、分布式协同以及命名的中心化等,这些都是分布式系统中非常底层而且是必不可少的基本功能,但是如果自己实现这些功能而且要达到高吞吐、低延迟同时还要保持一致性和可用性,实际上非常困难。
+
+**(一)抽象模型**
+
+Zookeeper 提供了一种树形结构级的命名空间,/app1/p_1 节点表示它的父节点为 /app1。
+
+
+
+**(二)节点类型**
+
+- 永久节点:不会因为会话结束或者超时而消失;
+- 临时节点:如果会话结束或者超时就会消失;
+- 有序节点:会在节点名的后面加一个数字后缀,并且是有序的,例如生成的有序节点为 /lock/node-0000000000,它的下一个有序节点则为 /lock/node-0000000001,依次类推。
+
+**(三)监听器**
+
+为一个节点注册监听器,在节点状态发生改变时,会给客户端发送消息。
+
+**(四)分布式锁实现**
+
+1. 创建一个锁目录 /lock。
+1. 在 /lock 下创建临时的且有序的子节点,第一个客户端对应的子节点为 /lock/lock-0000000000,第二个为 /lock/lock-0000000001,以此类推。
+2. 客户端获取 /lock 下的子节点列表,判断自己创建的子节点是否为当前子节点列表中序号最小的子节点,如果是则认为获得锁,否则监听自己的前一个子节点,获得子节点的变更通知后重复此步骤直至获得锁;
+3. 执行业务代码,完成后,删除对应的子节点。
+
+**(五)会话超时**
+
+如果一个已经获得锁的会话超时了,因为创建的是临时节点,因此该会话对应的临时节点会被删除,其它会话就可以获得锁了。可以看到,Zookeeper 分布式锁不会出现数据库分布式锁的死锁问题。
+
+**(六)羊群效应**
+
+在步骤二,一个节点未获得锁,需要监听监听自己的前一个子节点,这是因为如果监听所有的子节点,那么任意一个子节点状态改变,其它所有子节点都会收到通知(羊群效应),而我们只希望它的后一个子节点收到通知。
+
+# 五、分布式 Session
+
+在分布式场景下,一个用户的 Session 如果只存储在一个服务器上,那么当负载均衡器把用户的下一个请求转发到另一个服务器上,该服务器没有用户的 Session,就可能导致用户需要重新进行登录等操作。
+
+
+
+## 1. Sticky Sessions
+
+需要配置负载均衡器,使得一个用户的所有请求都路由到一个服务器节点上,这样就可以把用户的 Session 存放在该服务器节点中。
+
+缺点:当服务器节点宕机时,将丢失该服务器节点上的所有 Session。
+
+
+
+## 2. Session Replication
+
+在服务器节点之间进行 Session 同步操作,这样的话用户可以访问任何一个服务器节点。
+
+缺点:需要更好的服务器硬件条件;需要对服务器进行配置。
+
+
+
+## 3. Persistent DataStore
+
+将 Session 信息持久化到一个数据库中。
+
+缺点:有可能需要去实现存取 Session 的代码。
+
+
+
+## 4. In-Memory DataStore
+
+可以使用 Redis 和 Memcached 这种内存型数据库对 Session 进行存储,可以大大提高 Session 的读写效率。内存型数据库同样可以持久化数据到磁盘中来保证数据的安全性。
+
+# 六、分库与分表带来的分布式困境与应对之策
+
+
+
+## 事务问题
+
+使用分布式事务。
+
+## 查询问题
+
+使用汇总表。
+
+## ID 唯一性
+
+- 使用全局唯一 ID:GUID。
+- 为每个分片指定一个 ID 范围。
+- 分布式 ID 生成器 (如 Twitter 的 [Snowflake](https://twitter.github.io/twitter-server/) 算法)。
+
+# 参考资料
+
+- [Comparing Load Balancing Algorithms](http://www.jscape.com/blog/load-balancing-algorithms)
+- [负载均衡算法及手段](https://segmentfault.com/a/1190000004492447)
+- [Redirection and Load Balancing](http://slideplayer.com/slide/6599069/#)
+- [Session Management using Spring Session with JDBC DataStore](https://sivalabs.in/2018/02/session-management-using-spring-session-jdbc-datastore/)
+- [Apache Wicket User Guide - Reference Documentation](https://ci.apache.org/projects/wicket/guide/6.x/)
+- [集群/分布式环境下 5 种 Session 处理策略](http://blog.csdn.net/u010028869/article/details/50773174?ref=myread)
+- [浅谈分布式锁](http://www.linkedkeeper.com/detail/blog.action?bid=1023)
+- [深入理解分布式事务](https://juejin.im/entry/577c6f220a2b5800573492be)
+- [分布式系统的事务处理](https://coolshell.cn/articles/10910.html)
+- [关于分布式事务](http://blog.csdn.net/suifeng3051/article/details/52691210)
+- [基于 Zookeeper 的分布式锁](http://www.dengshenyu.com/java/%E5%88%86%E5%B8%83%E5%BC%8F%E7%B3%BB%E7%BB%9F/2017/10/23/zookeeper-distributed-lock.html)
+- [How Sharding Works](https://medium.com/@jeeyoungk/how-sharding-works-b4dec46b3f6)
+- [服务端指南 数据存储篇 | MySQL(09) 分库与分表带来的分布式困境与应对之策](http://blog.720ui.com/2017/mysql_core_09_multi_db_table2/ "服务端指南 数据存储篇 | MySQL(09) 分库与分表带来的分布式困境与应对之策")
+- [How to create unique row ID in sharded databases?](https://stackoverflow.com/questions/788829/how-to-create-unique-row-id-in-sharded-databases)
diff --git a/notes/剑指 offer 题解.md b/notes/剑指 offer 题解.md
index 018069e5..4a6c375c 100644
--- a/notes/剑指 offer 题解.md
+++ b/notes/剑指 offer 题解.md
@@ -1,206 +1,227 @@
-* [第二章 面试需要的基础知识](#第二章-面试需要的基础知识)
- * [2. 实现 Singleton](#2-实现-singleton)
- * [3. 数组中重复的数字](#3-数组中重复的数字)
- * [4. 二维数组中的查找](#4-二维数组中的查找)
- * [5. 替换空格](#5-替换空格)
- * [6. 从尾到头打印链表](#6-从尾到头打印链表)
- * [7. 重建二叉树](#7-重建二叉树)
- * [8. 二叉树的下一个结点](#8-二叉树的下一个结点)
- * [9. 用两个栈实现队列](#9-用两个栈实现队列)
- * [10.1 斐波那契数列](#101-斐波那契数列)
- * [10.2 跳台阶](#102-跳台阶)
- * [10.3 变态跳台阶](#103-变态跳台阶)
- * [10.4 矩形覆盖](#104-矩形覆盖)
- * [11. 旋转数组的最小数字](#11-旋转数组的最小数字)
- * [12. 矩阵中的路径](#12-矩阵中的路径)
- * [13. 机器人的运动范围](#13-机器人的运动范围)
- * [14. 剪绳子](#14-剪绳子)
- * [15. 二进制中 1 的个数](#15-二进制中-1-的个数)
-* [第三章 高质量的代码](#第三章-高质量的代码)
- * [16. 数值的整数次方](#16-数值的整数次方)
- * [17. 打印从 1 到最大的 n 位数](#17-打印从-1-到最大的-n-位数)
- * [18.1 在 O(1) 时间内删除链表节点](#181-在-o1-时间内删除链表节点)
- * [18.2 删除链表中重复的结点](#182-删除链表中重复的结点)
- * [19. 正则表达式匹配](#19-正则表达式匹配)
- * [20. 表示数值的字符串](#20-表示数值的字符串)
- * [21. 调整数组顺序使奇数位于偶数前面](#21-调整数组顺序使奇数位于偶数前面)
- * [22. 链表中倒数第 k 个结点](#22-链表中倒数第-k-个结点)
- * [23. 链表中环的入口结点](#23-链表中环的入口结点)
- * [24. 反转链表](#24-反转链表)
- * [25. 合并两个排序的链表](#25-合并两个排序的链表)
- * [26. 树的子结构](#26-树的子结构)
-* [第四章 解决面试题的思路](#第四章-解决面试题的思路)
- * [27. 二叉树的镜像](#27-二叉树的镜像)
- * [28.1 对称的二叉树](#281-对称的二叉树)
- * [28.2 平衡二叉树](#282-平衡二叉树)
- * [29. 顺时针打印矩阵](#29-顺时针打印矩阵)
- * [30. 包含 min 函数的栈](#30-包含-min-函数的栈)
- * [31. 栈的压入、弹出序列](#31-栈的压入弹出序列)
- * [32.1 从上往下打印二叉树](#321-从上往下打印二叉树)
- * [32.3 把二叉树打印成多行](#323--把二叉树打印成多行)
- * [32.3 按之字形顺序打印二叉树](#323-按之字形顺序打印二叉树)
- * [33. 二叉搜索树的后序遍历序列](#33-二叉搜索树的后序遍历序列)
- * [34. 二叉树中和为某一值的路径](#34-二叉树中和为某一值的路径)
- * [35. 复杂链表的复制](#35-复杂链表的复制)
- * [36. 二叉搜索树与双向链表](#36-二叉搜索树与双向链表)
- * [37. 序列化二叉树](#37-序列化二叉树)
- * [38. 字符串的排列](#38-字符串的排列)
-* [第五章 优化时间和空间效率](#第五章-优化时间和空间效率)
- * [39. 数组中出现次数超过一半的数字](#39-数组中出现次数超过一半的数字)
- * [40. 最小的 K 个数](#40-最小的-k-个数)
- * [41.1 数据流中的中位数](#411-数据流中的中位数)
- * [41.2 字符流中第一个不重复的字符](#412-字符流中第一个不重复的字符)
- * [42. 连续子数组的最大和](#42-连续子数组的最大和)
- * [43. 从 1 到 n 整数中 1 出现的次数](#43-从-1-到-n-整数中-1-出现的次数)
- * [44. 数字序列中的某一位数字](#44-数字序列中的某一位数字)
- * [45. 把数组排成最小的数](#45-把数组排成最小的数)
- * [46. 把数字翻译成字符串](#46-把数字翻译成字符串)
- * [47. 礼物的最大价值](#47-礼物的最大价值)
- * [48. 最长不含重复字符的子字符串](#48-最长不含重复字符的子字符串)
- * [49. 丑数](#49-丑数)
- * [50. 第一个只出现一次的字符位置](#50-第一个只出现一次的字符位置)
- * [51. 数组中的逆序对](#51-数组中的逆序对)
- * [52. 两个链表的第一个公共结点](#52-两个链表的第一个公共结点)
-* [第六章 面试中的各项能力](#第六章-面试中的各项能力)
- * [53 数字在排序数组中出现的次数](#53-数字在排序数组中出现的次数)
- * [54. 二叉搜索树的第 k 个结点](#54-二叉搜索树的第-k-个结点)
- * [55 二叉树的深度](#55-二叉树的深度)
- * [56. 数组中只出现一次的数字](#56-数组中只出现一次的数字)
- * [57.1 和为 S 的两个数字](#571-和为-s-的两个数字)
- * [57.2 和为 S 的连续正数序列](#572-和为-s-的连续正数序列)
- * [58.1 翻转单词顺序列](#581-翻转单词顺序列)
- * [58.2 左旋转字符串](#582-左旋转字符串)
- * [59. 滑动窗口的最大值](#59-滑动窗口的最大值)
- * [60. n 个骰子的点数](#60-n-个骰子的点数)
- * [61. 扑克牌顺子](#61-扑克牌顺子)
- * [62. 圆圈中最后剩下的数](#62-圆圈中最后剩下的数)
- * [63. 股票的最大利润](#63-股票的最大利润)
- * [64. 求 1+2+3+...+n](#64-求-1+2+3++n)
- * [65. 不用加减乘除做加法](#65-不用加减乘除做加法)
- * [66. 构建乘积数组](#66-构建乘积数组)
-* [第七章 两个面试案例](#第七章-两个面试案例)
- * [67. 把字符串转换成整数](#67-把字符串转换成整数)
- * [68. 树中两个节点的最低公共祖先](#68-树中两个节点的最低公共祖先)
+* [2. 实现 Singleton](#2-实现-singleton)
+* [3. 数组中重复的数字](#3-数组中重复的数字)
+* [4. 二维数组中的查找](#4-二维数组中的查找)
+* [5. 替换空格](#5-替换空格)
+* [6. 从尾到头打印链表](#6-从尾到头打印链表)
+* [7. 重建二叉树](#7-重建二叉树)
+* [8. 二叉树的下一个结点](#8-二叉树的下一个结点)
+* [9. 用两个栈实现队列](#9-用两个栈实现队列)
+* [10.1 斐波那契数列](#101-斐波那契数列)
+* [10.2 跳台阶](#102-跳台阶)
+* [10.3 变态跳台阶](#103-变态跳台阶)
+* [10.4 矩形覆盖](#104-矩形覆盖)
+* [11. 旋转数组的最小数字](#11-旋转数组的最小数字)
+* [12. 矩阵中的路径](#12-矩阵中的路径)
+* [13. 机器人的运动范围](#13-机器人的运动范围)
+* [14. 剪绳子](#14-剪绳子)
+* [15. 二进制中 1 的个数](#15-二进制中-1-的个数)
+* [16. 数值的整数次方](#16-数值的整数次方)
+* [17. 打印从 1 到最大的 n 位数](#17-打印从-1-到最大的-n-位数)
+* [18.1 在 O(1) 时间内删除链表节点](#181-在-o1-时间内删除链表节点)
+* [18.2 删除链表中重复的结点](#182-删除链表中重复的结点)
+* [19. 正则表达式匹配](#19-正则表达式匹配)
+* [20. 表示数值的字符串](#20-表示数值的字符串)
+* [21. 调整数组顺序使奇数位于偶数前面](#21-调整数组顺序使奇数位于偶数前面)
+* [22. 链表中倒数第 K 个结点](#22-链表中倒数第-k-个结点)
+* [23. 链表中环的入口结点](#23-链表中环的入口结点)
+* [24. 反转链表](#24-反转链表)
+* [25. 合并两个排序的链表](#25-合并两个排序的链表)
+* [26. 树的子结构](#26-树的子结构)
+* [27. 二叉树的镜像](#27-二叉树的镜像)
+* [28 对称的二叉树](#28-对称的二叉树)
+* [29. 顺时针打印矩阵](#29-顺时针打印矩阵)
+* [30. 包含 min 函数的栈](#30-包含-min-函数的栈)
+* [31. 栈的压入、弹出序列](#31-栈的压入弹出序列)
+* [32.1 从上往下打印二叉树](#321-从上往下打印二叉树)
+* [32.2 把二叉树打印成多行](#322-把二叉树打印成多行)
+* [32.3 按之字形顺序打印二叉树](#323-按之字形顺序打印二叉树)
+* [33. 二叉搜索树的后序遍历序列](#33-二叉搜索树的后序遍历序列)
+* [34. 二叉树中和为某一值的路径](#34-二叉树中和为某一值的路径)
+* [35. 复杂链表的复制](#35-复杂链表的复制)
+* [36. 二叉搜索树与双向链表](#36-二叉搜索树与双向链表)
+* [37. 序列化二叉树](#37-序列化二叉树)
+* [38. 字符串的排列](#38-字符串的排列)
+* [39. 数组中出现次数超过一半的数字](#39-数组中出现次数超过一半的数字)
+* [40. 最小的 K 个数](#40-最小的-k-个数)
+* [41.1 数据流中的中位数](#411-数据流中的中位数)
+* [41.2 字符流中第一个不重复的字符](#412-字符流中第一个不重复的字符)
+* [42. 连续子数组的最大和](#42-连续子数组的最大和)
+* [43. 从 1 到 n 整数中 1 出现的次数](#43-从-1-到-n-整数中-1-出现的次数)
+* [44. 数字序列中的某一位数字](#44-数字序列中的某一位数字)
+* [45. 把数组排成最小的数](#45-把数组排成最小的数)
+* [46. 把数字翻译成字符串](#46-把数字翻译成字符串)
+* [47. 礼物的最大价值](#47-礼物的最大价值)
+* [48. 最长不含重复字符的子字符串](#48-最长不含重复字符的子字符串)
+* [49. 丑数](#49-丑数)
+* [50. 第一个只出现一次的字符位置](#50-第一个只出现一次的字符位置)
+* [51. 数组中的逆序对](#51-数组中的逆序对)
+* [52. 两个链表的第一个公共结点](#52-两个链表的第一个公共结点)
+* [53 数字在排序数组中出现的次数](#53-数字在排序数组中出现的次数)
+* [54. 二叉搜索树的第 K 个结点](#54-二叉搜索树的第-k-个结点)
+* [55.1 二叉树的深度](#551-二叉树的深度)
+* [55.2 平衡二叉树](#552-平衡二叉树)
+* [56. 数组中只出现一次的数字](#56-数组中只出现一次的数字)
+* [57.1 和为 S 的两个数字](#571-和为-s-的两个数字)
+* [57.2 和为 S 的连续正数序列](#572-和为-s-的连续正数序列)
+* [58.1 翻转单词顺序列](#581-翻转单词顺序列)
+* [58.2 左旋转字符串](#582-左旋转字符串)
+* [59. 滑动窗口的最大值](#59-滑动窗口的最大值)
+* [60. n 个骰子的点数](#60-n-个骰子的点数)
+* [61. 扑克牌顺子](#61-扑克牌顺子)
+* [62. 圆圈中最后剩下的数](#62-圆圈中最后剩下的数)
+* [63. 股票的最大利润](#63-股票的最大利润)
+* [64. 求 1+2+3+...+n](#64-求-1+2+3++n)
+* [65. 不用加减乘除做加法](#65-不用加减乘除做加法)
+* [66. 构建乘积数组](#66-构建乘积数组)
+* [67. 把字符串转换成整数](#67-把字符串转换成整数)
+* [68. 树中两个节点的最低公共祖先](#68-树中两个节点的最低公共祖先)
+* [参考文献](#参考文献)
-# 第二章 面试需要的基础知识
+# 2. 实现 Singleton
-## 2. 实现 Singleton
+> [单例模式](https://github.com/CyC2018/Interview-Notebook/blob/master/notes/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F.md)
-[ 单例模式 ](https://github.com/CyC2018/InterviewNotes/blob/master/notes/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F.md#%E7%AC%AC-5-%E7%AB%A0-%E5%8D%95%E4%BB%B6%E6%A8%A1%E5%BC%8F)
+# 3. 数组中重复的数字
-## 3. 数组中重复的数字
-
-**题目描述**
+## 题目描述
在一个长度为 n 的数组里的所有数字都在 0 到 n-1 的范围内。数组中某些数字是重复的,但不知道有几个数字是重复的。也不知道每个数字重复几次。请找出数组中任意一个重复的数字。例如,如果输入长度为 7 的数组 {2, 3, 1, 0, 2, 5, 3},那么对应的输出是第一个重复的数字 2。
-**解题思路**
+要求复杂度为 O(N) + O(1),时间复杂度 O(N),空间复杂度 O(1)。因此不能使用排序的方法,也不能使用额外的标记数组。
+
+## 解题思路
这种数组元素在 [0, n-1] 范围内的问题,可以将值为 i 的元素放到第 i 个位置上。
+以 (2, 3, 1, 0, 2, 5) 为例:
+
+```html
+position-0 : (2,3,1,0,2,5) // 2 <-> 1
+ (1,3,2,0,2,5) // 1 <-> 3
+ (3,1,1,0,2,5) // 3 <-> 0
+ (0,1,1,3,2,5) // already in position
+position-1 : (0,1,1,3,2,5) // already in position
+position-2 : (0,1,1,3,2,5) // nums[i] == nums[nums[i]], exit
+```
+
+遍历到位置 2 时,该位置上的数为 1,但是第 1 个位置上已经有一个 1 的值了,因此可以知道 1 重复。
+
+复杂度:O(N) + O(1)
+
```java
-public boolean duplicate(int[] numbers, int length, int[] duplication) {
- if(numbers == null || length <= 0) return false;
+public boolean duplicate(int[] nums, int length, int[] duplication) {
+ if (nums == null || length <= 0) return false;
for (int i = 0; i < length; i++) {
- while (numbers[i] != i && numbers[i] != numbers[numbers[i]]) {
- swap(numbers, i, numbers[i]);
+ while (nums[i] != i && nums[i] != nums[nums[i]]) {
+ swap(nums, i, nums[i]);
}
- if (numbers[i] != i && numbers[i] == numbers[numbers[i]]) {
- duplication[0] = numbers[i];
+ if (nums[i] != i && nums[i] == nums[nums[i]]) {
+ duplication[0] = nums[i];
return true;
}
}
return false;
}
-private void swap(int[] numbers, int i, int j) {
- int t = numbers[i];
- numbers[i] = numbers[j];
- numbers[j] = t;
+private void swap(int[] nums, int i, int j) {
+ int t = nums[i]; nums[i] = nums[j]; nums[j] = t;
}
```
-## 4. 二维数组中的查找
+# 4. 二维数组中的查找
-**题目描述**
+## 题目描述
在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。
-**解题思路**
+```html
+Consider the following matrix:
+[
+ [1, 4, 7, 11, 15],
+ [2, 5, 8, 12, 19],
+ [3, 6, 9, 16, 22],
+ [10, 13, 14, 17, 24],
+ [18, 21, 23, 26, 30]
+]
-从右上角开始查找。因为矩阵中的一个数,它左边的数都比它来的小,下边的数都比它来的大。因此,从右上角开始查找,就可以根据 target 和当前元素的大小关系来改变行和列的下标,从而缩小查找区间。
+Given target = 5, return true.
+Given target = 20, return false.
+```
+
+## 解题思路
+
+从右上角开始查找。因为矩阵中的一个数,它左边的数都比它小,下边的数都比它大。因此,从右上角开始查找,就可以根据 target 和当前元素的大小关系来缩小查找区间。
+
+复杂度:O(M+N) + O(1)
```java
-public boolean Find(int target, int [][] array) {
- if (array == null || array.length == 0 || array[0].length == 0) return false;
- int m = array.length, n = array[0].length;
- int row = 0, col = n - 1;
- while (row < m && col >= 0) {
- if (target == array[row][col]) return true;
- else if (target < array[row][col]) col--;
- else row++;
+public boolean Find(int target, int[][] matrix) {
+ if (matrix == null || matrix.length == 0 || matrix[0].length == 0) return false;
+ int m = matrix.length, n = matrix[0].length;
+ int r = 0, c = n - 1; // 从右上角开始
+ while (r <= m - 1 && c >= 0) {
+ if (target == matrix[r][c]) return true;
+ else if (target > matrix[r][c]) r++;
+ else c--;
}
return false;
}
```
-## 5. 替换空格
+# 5. 替换空格
-**题目描述**
+## 题目描述
请实现一个函数,将一个字符串中的空格替换成“%20”。例如,当字符串为 We Are Happy. 则经过替换之后的字符串为 We%20Are%20Happy。
-**题目要求**
+## 解题思路
-以 O(1) 的空间复杂度和 O(n) 的空间复杂度来求解。
+在字符串尾部填充任意字符,使得字符串的长度等于字符串替换之后的长度。因为一个空格要替换成三个字符(%20),因此当遍历到一个空格时,需要在尾部填充两个任意字符。
-**解题思路**
+令 P1 指向字符串原来的末尾位置,P2 指向字符串现在的末尾位置。P1 和 P2 从后向前遍历,当 P1 遍历到一个空格时,就需要令 P2 指向的位置依次填充 02%(注意是逆序的),否则就填充上 P1 指向字符的值。
-从后向前改变字符串。
+从后向前遍是为了在改变 P2 所指向的内容时,不会影响到 P1 遍历原来字符串的内容。
+
+复杂度:O(N) + O(1)
```java
public String replaceSpace(StringBuffer str) {
- int n = str.length();
- for (int i = 0; i < n; i++) {
- if (str.charAt(i) == ' ') str.append(" "); // 尾部填充两个
+ int oldLen = str.length();
+ for (int i = 0; i < oldLen; i++) {
+ if (str.charAt(i) == ' ') {
+ str.append(" ");
+ }
}
-
- int idxOfOriginal = n - 1;
+ int idxOfOld = oldLen - 1;
int idxOfNew = str.length() - 1;
- while (idxOfOriginal >= 0 && idxOfNew > idxOfOriginal) {
- if (str.charAt(idxOfOriginal) == ' ') {
+ while (idxOfOld >= 0 && idxOfNew > idxOfOld) {
+ char c = str.charAt(idxOfOld--);
+ if (c == ' ') {
str.setCharAt(idxOfNew--, '0');
str.setCharAt(idxOfNew--, '2');
str.setCharAt(idxOfNew--, '%');
} else {
- str.setCharAt(idxOfNew--, str.charAt(idxOfOriginal));
+ str.setCharAt(idxOfNew--, c);
}
- idxOfOriginal--;
}
return str.toString();
}
```
-## 6. 从尾到头打印链表
+# 6. 从尾到头打印链表
-正向遍历然后调用 Collections.reverse()。
+## 题目描述
-```java
-public ArrayList printListFromTailToHead(ListNode listNode) {
- ArrayList ret = new ArrayList<>();
- while (listNode != null) {
- ret.add(listNode.val);
- listNode = listNode.next;
- }
- Collections.reverse(ret);
- return ret;
-}
-```
+输入链表的第一个节点,从尾到头反过来打印出每个结点的值。
-使用 Stack
+
+
+## 解题思路
+
+### 使用栈
```java
public ArrayList printListFromTailToHead(ListNode listNode) {
@@ -217,7 +238,7 @@ public ArrayList printListFromTailToHead(ListNode listNode) {
}
```
-递归
+### 使用递归
```java
public ArrayList printListFromTailToHead(ListNode listNode) {
@@ -230,18 +251,37 @@ public ArrayList printListFromTailToHead(ListNode listNode) {
}
```
-不使用库函数,并且不使用递归的迭代实现,利用链表的头插法为逆序的特性。
+### 使用 Collections.reverse()
```java
public ArrayList printListFromTailToHead(ListNode listNode) {
- ListNode head = new ListNode(-1); // 头结点
- ListNode cur = listNode;
- while (cur != null) {
- ListNode next = cur.next;
- cur.next = head.next;
- head.next = cur;
- cur = next;
+ ArrayList ret = new ArrayList<>();
+ while (listNode != null) {
+ ret.add(listNode.val);
+ listNode = listNode.next;
}
+ Collections.reverse(ret);
+ return ret;
+}
+```
+
+### 使用头插法
+
+利用链表头插法为逆序的特点。
+
+头结点和第一个节点的区别:头结点是在头插法中使用的一个额外节点,这个节点不存储值;第一个节点就是链表的第一个真正存储值的节点。
+
+```java
+public ArrayList printListFromTailToHead(ListNode listNode) {
+ // 头插法构建逆序链表
+ ListNode head = new ListNode(-1);
+ while (listNode != null) {
+ ListNode memo = listNode.next;
+ listNode.next = head.next;
+ head.next = listNode;
+ listNode = memo;
+ }
+ // 构建 ArrayList
ArrayList ret = new ArrayList<>();
head = head.next;
while (head != null) {
@@ -252,14 +292,30 @@ public ArrayList printListFromTailToHead(ListNode listNode) {
}
```
-## 7. 重建二叉树
+# 7. 重建二叉树
-**题目描述**
+## 题目描述
-根据二叉树的前序遍历和中序遍历的结果,重建出该二叉树。
+根据二叉树的前序遍历和中序遍历的结果,重建出该二叉树。假设输入的前序遍历和中序遍历的结果中都不含重复的数字。
+
+```html
+preorder = [3,9,20,15,7]
+inorder = [9,3,15,20,7]
+```
+
+
+
+## 解题思路
+
+前序遍历的第一个值为根节点的值,使用这个值将中序遍历结果分成两部分,左部分为树的左子树中序遍历结果,右部分为树的右子树中序遍历的结果。
```java
+private Map inOrderNumsIdx = new HashMap<>(); // 缓存中序遍历数组的每个值对应的索引
+
public TreeNode reConstructBinaryTree(int[] pre, int[] in) {
+ for (int i = 0; i < in.length; i++) {
+ inOrderNumsIdx.put(in[i], i);
+ }
return reConstructBinaryTree(pre, 0, pre.length - 1, in, 0, in.length - 1);
}
@@ -267,243 +323,387 @@ private TreeNode reConstructBinaryTree(int[] pre, int preL, int preR, int[] in,
if (preL == preR) return new TreeNode(pre[preL]);
if (preL > preR || inL > inR) return null;
TreeNode root = new TreeNode(pre[preL]);
- int midIdx = inL;
- while (midIdx <= inR && in[midIdx] != root.val) midIdx++;
- int leftTreeSize = midIdx - inL;
+ int inIdx = inOrderNumsIdx.get(root.val);
+ int leftTreeSize = inIdx - inL;
root.left = reConstructBinaryTree(pre, preL + 1, preL + leftTreeSize, in, inL, inL + leftTreeSize - 1);
root.right = reConstructBinaryTree(pre, preL + leftTreeSize + 1, preR, in, inL + leftTreeSize + 1, inR);
return root;
}
```
-## 8. 二叉树的下一个结点
+# 8. 二叉树的下一个结点
-**题目描述**
+## 题目描述
给定一个二叉树和其中的一个结点,请找出中序遍历顺序的下一个结点并且返回。注意,树中的结点不仅包含左右子结点,同时包含指向父结点的指针。
-**解题思路**
+## 解题思路
-- 如果一个节点有右子树不为空,那么该节点的下一个节点是右子树的最左节点;
-- 否则,向上找第一个左链接指向的树包含该节点的祖先节点。
+① 如果一个节点的右子树不为空,那么该节点的下一个节点是右子树的最左节点;
-
+
+
+② 否则,向上找第一个左链接指向的树包含该节点的祖先节点。
+
+
+
+```java
+public class TreeLinkNode {
+ int val;
+ TreeLinkNode left = null;
+ TreeLinkNode right = null;
+ TreeLinkNode next = null;
+
+ TreeLinkNode(int val) {
+ this.val = val;
+ }
+}
+```
```java
public TreeLinkNode GetNext(TreeLinkNode pNode) {
- if (pNode == null) return null;
if (pNode.right != null) {
- pNode = pNode.right;
- while (pNode.left != null) pNode = pNode.left;
- return pNode;
+ TreeLinkNode node = pNode.right;
+ while (node.left != null) node = node.left;
+ return node;
} else {
- TreeLinkNode parent = pNode.next;
- while (parent != null) {
+ while (pNode.next != null) {
+ TreeLinkNode parent = pNode.next;
if (parent.left == pNode) return parent;
pNode = pNode.next;
- parent = pNode.next;
}
}
return null;
}
```
-## 9. 用两个栈实现队列
+# 9. 用两个栈实现队列
-**解题思路**
+## 解题思路
-添加到栈中的序列顺序会被反转,如果进行两次反转,那么得到的序列依然是正向的。因此,添加的数据需要同时压入两个栈之后才能出栈,这样就能保证出栈的顺序为先进先出。
+in 栈用来处理入栈(push)操作,out 栈用来处理出栈(pop)操作。一个元素进入 in 栈之后,出栈的顺序被反转。当元素要出栈时,需要先进入 out 栈,此时元素出栈顺序再一次被反转,因此出栈顺序就和最开始入栈顺序是相同的,此时先进入的元素先退出,这就是队列的顺序。
+
+
```java
-Stack stack1 = new Stack();
-Stack stack2 = new Stack();
+Stack in = new Stack();
+Stack out = new Stack();
public void push(int node) {
- stack1.push(node);
+ in.push(node);
}
-public int pop() {
- if (stack2.isEmpty()) {
- while (!stack1.isEmpty()) {
- stack2.push(stack1.pop());
+public int pop() throws Exception {
+ if (out.isEmpty()) {
+ while (!in.isEmpty()) {
+ out.push(in.pop());
}
}
- return stack2.pop();
+ if (out.isEmpty()) {
+ throw new Exception("queue is empty");
+ }
+ return out.pop();
}
```
-## 10.1 斐波那契数列
+# 10.1 斐波那契数列
+
+## 题目描述
+
+求菲波那契数列的第 n 项。
+
+
+
+## 解题思路
+
+如果使用递归求解,会重复计算一些子问题。例如,计算 f(10) 需要计算 f(9) 和 f(8),计算 f(9) 需要计算 f(8) 和 f(7),可以看到 f(8) 被重复计算了。
+
+
+
+递归方法是将一个问题划分成多个子问题求解,动态规划也是如此,但是动态规划会把子问题的解缓存起来,避免重复求解子问题。
```java
-private int[] fib = new int[40];
-
-public Solution() {
+public int Fibonacci(int n) {
+ if(n <= 1) return n;
+ int[] fib = new int[n + 1];
fib[1] = 1;
- fib[2] = 2;
- for (int i = 2; i < fib.length; i++) {
+ for (int i = 2; i <= n; i++) {
fib[i] = fib[i - 1] + fib[i - 2];
}
-}
-
-public int Fibonacci(int n) {
return fib[n];
}
```
-## 10.2 跳台阶
+考虑到第 i 项只与第 i-1 和第 i-2 项有关,因此只需要存储前两项的值就能求解第 i 项,从而将空间复杂度由 O(N) 降低为 O(1)。
```java
-public int JumpFloor(int target) {
- if (target == 1) return 1;
- int[] dp = new int[target];
- dp[0] = 1;
- dp[1] = 2;
- for (int i = 2; i < dp.length; i++) {
- dp[i] = dp[i - 1] + dp[i - 2];
+public int Fibonacci(int n) {
+ if(n <= 1) return n;
+ int pre2 = 0, pre1 = 1;
+ int fib = 0;
+ for (int i = 2; i <= n; i++) {
+ fib = pre2 + pre1;
+ pre2 = pre1;
+ pre1 = fib;
}
- return dp[target - 1];
+ return fib;
}
```
-## 10.3 变态跳台阶
+由于待求解的 n 小于 40,因此可以将前 40 项的结果先进行计算,之后就能以 O(1) 时间复杂度得到第 n 项的值了。
```java
-public int JumpFloorII(int target) {
- int[] dp = new int[target];
+public class Solution {
+ private int[] fib = new int[40];
+ public Solution() {
+ fib[1] = 1;
+ fib[2] = 2;
+ for(int i = 2; i < fib.length; i++) {
+ fib[i] = fib[i - 1] + fib[i - 2];
+ }
+ }
+ public int Fibonacci(int n) {
+ return fib[n];
+ }
+}
+```
+
+# 10.2 跳台阶
+
+## 题目描述
+
+一只青蛙一次可以跳上 1 级台阶,也可以跳上 2 级。求该青蛙跳上一个 n 级的台阶总共有多少种跳法。
+
+## 解题思路
+
+复杂度:O(N) + O(N)
+
+```java
+public int JumpFloor(int n) {
+ if (n == 1) return 1;
+ int[] dp = new int[n];
+ dp[0] = 1;
+ dp[1] = 2;
+ for (int i = 2; i < n; i++) {
+ dp[i] = dp[i - 1] + dp[i - 2];
+ }
+ return dp[n - 1];
+}
+```
+
+复杂度:O(N) + O(1)
+
+```java
+public int JumpFloor(int n) {
+ if (n <= 1) return n;
+ int pre2 = 0, pre1 = 1;
+ int result = 0;
+ for (int i = 1; i <= n; i++) {
+ result = pre2 + pre1;
+ pre2 = pre1;
+ pre1 = result;
+ }
+ return result;
+}
+```
+
+# 10.3 变态跳台阶
+
+## 题目描述
+
+一只青蛙一次可以跳上 1 级台阶,也可以跳上 2 级……它也可以跳上 n 级。求该青蛙跳上一个 n 级的台阶总共有多少种跳法。
+
+## 解题思路
+
+```java
+public int JumpFloorII(int n) {
+ int[] dp = new int[n];
Arrays.fill(dp, 1);
- for (int i = 1; i < target; i++) {
- for (int j = 0; j < i; j++) {
+ for(int i = 1; i < n; i++) {
+ for(int j = 0; j < i; j++) {
dp[i] += dp[j];
}
}
- return dp[target - 1];
+ return dp[n - 1];
}
```
-## 10.4 矩形覆盖
+# 10.4 矩形覆盖
-**题目描述**
+## 题目描述
我们可以用 2\*1 的小矩形横着或者竖着去覆盖更大的矩形。请问用 n 个 2\*1 的小矩形无重叠地覆盖一个 2\*n 的大矩形,总共有多少种方法?
+## 解题思路
+
+复杂度:O(N) + O(N)
+
```java
-public int RectCover(int target) {
- if (target <= 2) return target;
- return RectCover(target - 1) + RectCover(target - 2);
+public int RectCover(int n) {
+ if (n <= 2) return n;
+ int[] dp = new int[n];
+ dp[0] = 1;
+ dp[1] = 2;
+ for (int i = 2; i < n; i++) {
+ dp[i] = dp[i - 1] + dp[i - 2];
+ }
+ return dp[n - 1];
}
```
-## 11. 旋转数组的最小数字
+复杂度:O(N) + O(1)
-**题目描述**
+```java
+public int RectCover(int n) {
+ if (n <= 2) return n;
+ int pre2 = 1, pre1 = 2;
+ int result = 0;
+ for (int i = 3; i <= n; i++) {
+ result = pre2 + pre1;
+ pre2 = pre1;
+ pre1 = result;
+ }
+ return result;
+}
+```
+
+# 11. 旋转数组的最小数字
+
+## 题目描述
把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。输入一个非递减排序的数组的一个旋转,输出旋转数组的最小元素。例如数组 {3, 4, 5, 1, 2} 为 {1, 2, 3, 4, 5} 的一个旋转,该数组的最小值为 1。NOTE:给出的所有元素都大于 0,若数组大小为 0,请返回 0。
+## 解题思路
-O(N) 时间复杂度解法:
+### 分治
+
+复杂度:O(logN) + O(1),其实空间复杂度不止 O(1),因为分治使用了递归栈,用到了额外的空间,如果对空间有要求就不能用这种方法。
```java
-public int minNumberInRotateArray(int[] array) {
- if (array.length == 0) return 0;
- for (int i = 0; i < array.length - 1; i++) {
- if (array[i] > array[i + 1]) return array[i + 1];
- }
- return 0;
+public int minNumberInRotateArray(int[] nums) {
+ return minNumberInRotateArray(nums, 0, nums.length - 1);
+}
+
+private int minNumberInRotateArray(int[] nums, int first, int last) {
+ if (nums[first] < nums[last]) return nums[first];
+ if (first == last) return nums[first];
+ int mid = first + (last - first) / 2;
+ return Math.min(minNumberInRotateArray(nums, first, mid), minNumberInRotateArray(nums, mid + 1, last));
}
```
-O(lgN) 时间复杂度解法:
+### 二分查找
+
+复杂度:O(logN) + O(1)
```java
-public int minNumberInRotateArray(int[] array) {
- if (array.length == 0) return 0;
- int l = 0, r = array.length - 1;
- int mid = -1;
- while (array[l] >= array[r]) {
- if (r - l == 1) return array[r];
- mid = l + (r - l) / 2;
- if (array[mid] >= array[l]) l = mid;
- else if (array[mid] <= array[r]) r = mid;
+public int minNumberInRotateArray(int[] nums) {
+ if (nums.length == 0) return 0;
+ int l = 0, h = nums.length - 1;
+ while (nums[l] >= nums[h]) {
+ if (h - l == 1) return nums[h];
+ int mid = l + (h - l) / 2;
+ if (nums[mid] >= nums[l]) l = mid;
+ else h = mid;
}
- return array[mid];
+ return nums[l];
}
```
-## 12. 矩阵中的路径
+# 12. 矩阵中的路径
-**题目描述**
+## 题目描述
-请设计一个函数,用来判断在一个矩阵中是否存在一条包含某字符串所有字符的路径。路径可以从矩阵中的任意一个格子开始,每一步可以在矩阵中向左,向右,向上,向下移动一个格子。如果一条路径经过了矩阵中的某一个格子,则该路径不能再进入该格子。例如 a b c e s f c s a d e e 矩阵中包含一条字符串 "bcced" 的路径,但是矩阵中不包含 "abcb" 路径,因为字符串的第一个字符 b 占据了矩阵中的第一行第二个格子之后,路径不能再次进入该格子。
+请设计一个函数,用来判断在一个矩阵中是否存在一条包含某字符串所有字符的路径。路径可以从矩阵中的任意一个格子开始,每一步可以在矩阵中向左,向右,向上,向下移动一个格子。如果一条路径经过了矩阵中的某一个格子,则该路径不能再进入该格子。
+
+例如下面的矩阵包含了一条 bfce 路径。
+
+
+
+## 解题思路
```java
private int[][] next = {{0, -1}, {0, 1}, {-1, 0}, {1, 0}};
private int rows;
private int cols;
-public boolean hasPath(char[] matrix, int rows, int cols, char[] str) {
+public boolean hasPath(char[] array, int rows, int cols, char[] str) {
if (rows == 0 || cols == 0) return false;
this.rows = rows;
this.cols = cols;
- // 一维数组重建二维矩阵
- char[][] newMatrix = new char[rows][cols];
- for (int i = 0, idx = 0; i < rows; i++) {
- for (int j = 0; j < cols; j++) {
- newMatrix[i][j] = matrix[idx++];
- }
- }
+ boolean[][] hasUsed = new boolean[rows][cols];
+ char[][] matrix = buildMatrix(array);
for (int i = 0; i < rows; i++) {
for (int j = 0; j < cols; j++) {
- if (backtracking(newMatrix, str, new boolean[rows][cols], 0, i, j)) return true;
+ if (backtracking(matrix, str, hasUsed, 0, i, j)) return true;
}
}
return false;
}
-private boolean backtracking(char[][] matrix, char[] str, boolean[][] used, int pathLen, int curR, int curC) {
+private boolean backtracking(char[][] matrix, char[] str, boolean[][] hasUsed, int pathLen, int row, int col) {
if (pathLen == str.length) return true;
- if (curR < 0 || curR >= rows || curC < 0 || curC >= cols) return false;
- if (matrix[curR][curC] != str[pathLen]) return false;
- if (used[curR][curC]) return false;
- used[curR][curC] = true;
+ if (row < 0 || row >= rows || col < 0 || col >= cols) return false;
+ if (matrix[row][col] != str[pathLen]) return false;
+ if (hasUsed[row][col]) return false;
+ hasUsed[row][col] = true;
for (int i = 0; i < next.length; i++) {
- if (backtracking(matrix, str, used, pathLen + 1, curR + next[i][0], curC + next[i][1]))
+ if (backtracking(matrix, str, hasUsed, pathLen + 1, row + next[i][0], col + next[i][1]))
return true;
}
- used[curR][curC] = false;
+ hasUsed[row][col] = false;
return false;
}
+
+private char[][] buildMatrix(char[] array) {
+ char[][] matrix = new char[rows][cols];
+ for (int i = 0, idx = 0; i < rows; i++) {
+ for (int j = 0; j < cols; j++) {
+ matrix[i][j] = array[idx++];
+ }
+ }
+ return matrix;
+}
```
+# 13. 机器人的运动范围
-## 13. 机器人的运动范围
-
-**题目描述**
+## 题目描述
地上有一个 m 行和 n 列的方格。一个机器人从坐标 (0, 0) 的格子开始移动,每一次只能向左右上下四个方向移动一格,但是不能进入行坐标和列坐标的数位之和大于 k 的格子。例如,当 k 为 18 时,机器人能够进入方格(35, 37),因为 3+5+3+7=18。但是,它不能进入方格(35, 38),因为 3+5+3+8=19。请问该机器人能够达到多少个格子?
+## 解题思路
+
```java
private int cnt = 0;
private int[][] next = {{0, -1}, {0, 1}, {-1, 0}, {1, 0}};
+private int rows;
+private int cols;
+private int threshold;
private int[][] digitSum;
public int movingCount(int threshold, int rows, int cols) {
- initDigitSum(rows, cols);
- dfs(new boolean[rows][cols], threshold, rows, cols, 0, 0);
+ this.rows = rows;
+ this.cols = cols;
+ this.threshold = threshold;
+ initDigitSum();
+ boolean[][] hasVisited = new boolean[rows][cols];
+ dfs(hasVisited, 0, 0);
return cnt;
}
-private void dfs(boolean[][] visited, int threshold, int rows, int cols, int r, int c) {
- if (r < 0 || r >= rows || c < 0 || c >= cols) return;
- if (visited[r][c]) return;
- visited[r][c] = true;
- if (this.digitSum[r][c] > threshold) return;
+private void dfs(boolean[][] hasVisited, int r, int c) {
+ if (r < 0 || r >= this.rows || c < 0 || c >= this.cols) return;
+ if (hasVisited[r][c]) return;
+ hasVisited[r][c] = true;
+ if (this.digitSum[r][c] > this.threshold) return;
this.cnt++;
for (int i = 0; i < this.next.length; i++) {
- dfs(visited, threshold, rows, cols, r + next[i][0], c + next[i][1]);
+ dfs(hasVisited, r + next[i][0], c + next[i][1]);
}
}
-private void initDigitSum(int rows, int cols) {
+private void initDigitSum() {
int[] digitSumOne = new int[Math.max(rows, cols)];
for (int i = 0; i < digitSumOne.length; i++) {
int n = i;
@@ -513,43 +713,62 @@ private void initDigitSum(int rows, int cols) {
}
}
this.digitSum = new int[rows][cols];
- for (int i = 0; i < rows; i++) {
- for (int j = 0; j < cols; j++) {
+ for (int i = 0; i < this.rows; i++) {
+ for (int j = 0; j < this.cols; j++) {
this.digitSum[i][j] = digitSumOne[i] + digitSumOne[j];
}
}
}
```
-## 14. 剪绳子
+# 14. 剪绳子
-**题目描述**
+## 题目描述
把一根绳子剪成多段,并且使得每段的长度乘积最大。
-**动态规划解法**
+## 解题思路
-[ 分割整数 ](https://github.com/CyC2018/InterviewNotes/blob/master/notes/Leetcode%20%E9%A2%98%E8%A7%A3.md#%E5%88%86%E5%89%B2%E6%95%B4%E6%95%B0)
-
-**贪心解法**
-
-尽可能多得剪长度为 3 的绳子,并且不允许有长度为 1 的绳子出现,如果出现了,就从已经切好长度为 3 的绳子中拿出一段与长度为 1 的绳子重新组合,把它们切成两段长度为 2 的绳子。
+### 动态规划解法
```java
-int maxProductAfterCuttin(int length) {
- if (length < 2) return 0;
- if (length == 2) return 1;
- if (length == 3) return 2;
- int timesOf3 = length / 3;
- if (length - timesOf3 * 3 == 1) timesOf3--;
- int timesOf2 = (length - timesOf3 * 3) / 2;
+public int maxProductAfterCutting(int n) {
+ int[] dp = new int[n + 1];
+ dp[1] = 1;
+ for (int i = 2; i <= n; i++) {
+ for (int j = 1; j < i; j++) {
+ dp[i] = Math.max(dp[i], Math.max(j * (i - j), dp[j] * (i - j)));
+ }
+ }
+ return dp[n];
+}
+```
+
+### 贪心解法
+
+尽可能多剪长度为 3 的绳子,并且不允许有长度为 1 的绳子出现,如果出现了,就从已经切好长度为 3 的绳子中拿出一段与长度为 1 的绳子重新组合,把它们切成两段长度为 2 的绳子。
+
+证明:当 n >= 5 时,3(n - 3) - 2(n - 2) = n - 5 >= 0。因此把长度大于 5 的绳子切成两段,令其中一段长度为 3 可以使得两段的乘积最大。
+
+```java
+public int maxProductAfterCutting(int n) {
+ if (n < 2) return 0;
+ if (n == 2) return 1;
+ if (n == 3) return 2;
+ int timesOf3 = n / 3;
+ if (n - timesOf3 * 3 == 1) timesOf3--;
+ int timesOf2 = (n - timesOf3 * 3) / 2;
return (int) (Math.pow(3, timesOf3)) * (int) (Math.pow(2, timesOf2));
}
```
-## 15. 二进制中 1 的个数
+# 15. 二进制中 1 的个数
-使用库函数:
+## 题目描述
+
+输入一个整数,输出该数二进制表示中 1 的个数。
+
+### Integer.bitCount()
```java
public int NumberOf1(int n) {
@@ -557,9 +776,17 @@ public int NumberOf1(int n) {
}
```
-O(lgM) 时间复杂度解法,其中 M 表示 1 的个数:
+### n&(n-1)
-n&(n-1) 该位运算是去除 n 的位级表示中最低的那一位。例如对于二进制表示 10110100,减去 1 得到 10110011,这两个数相与得到 10110000。
+O(logM) 时间复杂度解法,其中 M 表示 1 的个数。
+
+该位运算是去除 n 的位级表示中最低的那一位。
+
+```
+n : 10110100
+n-1 : 10110011
+n&(n-1) : 10110000
+```
```java
public int NumberOf1(int n) {
@@ -572,9 +799,19 @@ public int NumberOf1(int n) {
}
```
-# 第三章 高质量的代码
+# 16. 数值的整数次方
-## 16. 数值的整数次方
+## 题目描述
+
+给定一个 double 类型的浮点数 base 和 int 类型的整数 exponent。求 base 的 exponent 次方。
+
+## 解题思路
+
+下面的讨论中 x 代表 base,n 代表 exponent。
+
+
+
+因为 (x\*x)n/2 可以通过递归求解,并且每递归一次,n 都减小一半,因此整个算法的时间复杂度为 O(logn)。
```java
public double Power(double base, int exponent) {
@@ -587,11 +824,21 @@ public double Power(double base, int exponent) {
}
double pow = Power(base * base, exponent / 2);
if (exponent % 2 != 0) pow = pow * base;
- return isNegative ? 1 / pow : pow;
+ return isNegative ? (1 / pow) : pow;
}
```
-## 17. 打印从 1 到最大的 n 位数
+# 17. 打印从 1 到最大的 n 位数
+
+## 题目描述
+
+输入数字 n,按顺序打印出从 1 最大的 n 位十进制数。比如输入 3,则打印出 1、2、3 一直到最大的 3 位数即 999。
+
+## 解题思路
+
+由于 n 可能会非常大,因此不能直接用 int 表示数字,而是用 char 数组进行存储。
+
+使用回溯法得到所有的数。
```java
public void print1ToMaxOfNDigits(int n) {
@@ -600,28 +847,38 @@ public void print1ToMaxOfNDigits(int n) {
print1ToMaxOfNDigits(number, -1);
}
-private void print1ToMaxOfNDigits(char[] number, int idx) {
- if (idx == number.length - 1) {
+private void print1ToMaxOfNDigits(char[] number, int digit) {
+ if (digit == number.length - 1) {
printNumber(number);
return;
}
for (int i = 0; i < 10; i++) {
- number[idx + 1] = (char) (i + '0');
- print1ToMaxOfNDigits(number, idx + 1);
+ number[digit + 1] = (char) (i + '0');
+ print1ToMaxOfNDigits(number, digit + 1);
}
}
private void printNumber(char[] number) {
- boolean isBeginWith0 = true;
- for (char c : number) {
- if (isBeginWith0 && c != '0') isBeginWith0 = false;
- if(!isBeginWith0) System.out.print(c);
- }
+ int index = 0;
+ while (index < number.length && number[index] == '0') index++;
+ while (index < number.length) System.out.print(number[index++]);
System.out.println();
}
```
-## 18.1 在 O(1) 时间内删除链表节点
+# 18.1 在 O(1) 时间内删除链表节点
+
+## 解题思路
+
+① 如果该节点不是尾节点,那么可以直接将下一个节点的值赋给该节点,令该节点指向下下个节点,然后删除下一个节点,时间复杂度为 O(1)。
+
+
+
+② 否则,就需要先遍历链表,找到节点的前一个节点,然后让前一个节点指向 null,时间复杂度为 O(N)。
+
+
+
+综上,如果进行 N 次操作,那么大约需要操作节点的次数为 N-1+N=2N-1,其中 N-1 表示 N-1 个不是尾节点的每个节点以 O(1) 的时间复杂度操作节点的总次数,N 表示 1 个尾节点以 O(N) 的时间复杂度操作节点的总次数。(2N-1)/N \~ 2,因此该算法的平均时间复杂度为 O(1)。
```java
public ListNode deleteNode(ListNode head, ListNode tobeDelete) {
@@ -640,126 +897,167 @@ public ListNode deleteNode(ListNode head, ListNode tobeDelete) {
}
```
-## 18.2 删除链表中重复的结点
+# 18.2 删除链表中重复的结点
+
+## 题目描述
+
+
+
+## 解题描述
```java
public ListNode deleteDuplication(ListNode pHead) {
if (pHead == null) return null;
- if (pHead.next == null) return pHead;
- if (pHead.val == pHead.next.val) {
- ListNode next = pHead.next;
- while (next != null && pHead.val == next.val) {
- next = next.next;
- }
+ ListNode next = pHead.next;
+ if (next == null) return pHead;
+ if (pHead.val == next.val) {
+ while (next != null && pHead.val == next.val) next = next.next;
return deleteDuplication(next);
- } else {
- pHead.next = deleteDuplication(pHead.next);
- return pHead;
}
+ pHead.next = deleteDuplication(pHead.next);
+ return pHead;
}
```
-## 19. 正则表达式匹配
+# 19. 正则表达式匹配
-**题目描述**
+## 题目描述
-请实现一个函数用来匹配包括 '.' 和 '\*' 的正则表达式。模式中的字符 '.' 表示任意一个字符,而 '\*' 表示它前面的字符可以出现任意次(包含 0 次)。 在本题中,匹配是指字符串的所有字符匹配整个模式。例如,字符串 "aaa" 与模式 "a.a" 和 "ab\*ac\*a" 匹配,但是与 "aa.a" 和 "ab\*a" 均不匹配
+请实现一个函数用来匹配包括 '.' 和 '\*' 的正则表达式。模式中的字符 '.' 表示任意一个字符,而 '\*' 表示它前面的字符可以出现任意次(包含 0 次)。 在本题中,匹配是指字符串的所有字符匹配整个模式。例如,字符串 "aaa" 与模式 "a.a" 和 "ab\*ac\*a" 匹配,但是与 "aa.a" 和 "ab\*a" 均不匹配。
+
+## 解题思路
+
+应该注意到,'.' 是用来当做一个任意字符,而 '\*' 是用来重复前面的字符。这两个的作用不同,不能把 '.' 的作用和 '\*' 进行类比,从而把它当成重复前面字符一次。
+
+```html
+p.charAt(j) == s.charAt(i) : dp[i][j] = dp[i-1][j-1];
+p.charAt(j) == '.' : dp[i][j] = dp[i-1][j-1];
+p.charAt(j) == '*' :
+ p.charAt(j-1) != s.charAt(i) : dp[i][j] = dp[i][j-2] //a* only counts as empty
+ p.charAt(j-1) == s.charAt(i) or p.charAt(i-1) == '.':
+ dp[i][j] = dp[i-1][j] // a* counts as multiple a
+ or dp[i][j] = dp[i][j-1] // a* counts as single a
+ or dp[i][j] = dp[i][j-2] // a* counts as empty
+```
```java
public boolean match(char[] str, char[] pattern) {
- int n = str.length, m = pattern.length;
- boolean[][] dp = new boolean[n + 1][m + 1];
+ int m = str.length, n = pattern.length;
+ boolean[][] dp = new boolean[m + 1][n + 1];
dp[0][0] = true;
- for (int i = 1; i <= m; i++) {
- if (pattern[i - 1] == '*') dp[0][i] = dp[0][i - 2];
- }
for (int i = 1; i <= n; i++) {
- for (int j = 1; j <= m; j++) {
- if (str[i - 1] == pattern[j - 1] || pattern[j - 1] == '.') dp[i][j] = dp[i - 1][j - 1];
- else if (pattern[j - 1] == '*') {
- if (pattern[j - 2] != str[i - 1] && pattern[j - 2] != '.') dp[i][j] = dp[i][j - 2];
- else dp[i][j] = dp[i][j - 1] || dp[i][j - 2] || dp[i - 1][j];
+ if (pattern[i - 1] == '*') {
+ dp[0][i] = dp[0][i - 2];
+ }
+ }
+ for (int i = 1; i <= m; i++) {
+ for (int j = 1; j <= n; j++) {
+ if (str[i - 1] == pattern[j - 1] || pattern[j - 1] == '.') {
+ dp[i][j] = dp[i - 1][j - 1];
+ } else if (pattern[j - 1] == '*') {
+ if (pattern[j - 2] == str[i - 1] || pattern[j - 2] == '.') {
+ dp[i][j] = dp[i][j - 1] || dp[i][j - 2] || dp[i - 1][j];
+ } else {
+ dp[i][j] = dp[i][j - 2];
+ }
}
}
}
- return dp[n][m];
+ return dp[m][n];
}
```
-## 20. 表示数值的字符串
+# 20. 表示数值的字符串
-**题目描述**
+## 题目描述
请实现一个函数用来判断字符串是否表示数值(包括整数和小数)。例如,字符串 "+100","5e2","-123","3.1416" 和 "-1E-16" 都表示数值。 但是 "12e","1a3.14","1.2.3","+-5" 和 "12e+4.3" 都不是。
+## 解题思路
+
```java
public boolean isNumeric(char[] str) {
- String string = String.valueOf(str);
- return string.matches("[\\+-]?[0-9]*(\\.[0-9]*)?([eE][\\+-]?[0-9]+)?");
+ return new String(str).matches("[+-]?\\d*(\\.\\d+)?([eE][+-]?\\d+)?");
}
```
-## 21. 调整数组顺序使奇数位于偶数前面
+# 21. 调整数组顺序使奇数位于偶数前面
-**题目要求**
+## 题目描述
保证奇数和奇数,偶数和偶数之间的相对位置不变,这和书本不太一样。
-时间复杂度 : O(n2 )
-空间复杂度 : O(1)
+## 解题思路
+
+复杂度:O(N2 ) + O(1)
```java
-public void reOrderArray(int[] array) {
- int n = array.length;
+public void reOrderArray(int[] nums) {
+ int n = nums.length;
for (int i = 0; i < n; i++) {
- if (array[i] % 2 == 0) {
+ if (nums[i] % 2 == 0) {
int nextOddIdx = i + 1;
- while (nextOddIdx < n && array[nextOddIdx] % 2 == 0) nextOddIdx++;
+ while (nextOddIdx < n && nums[nextOddIdx] % 2 == 0) nextOddIdx++;
if (nextOddIdx == n) break;
- int nextOddVal = array[nextOddIdx];
+ int nextOddVal = nums[nextOddIdx];
for (int j = nextOddIdx; j > i; j--) {
- array[j] = array[j - 1];
+ nums[j] = nums[j - 1];
}
- array[i] = nextOddVal;
+ nums[i] = nextOddVal;
}
}
}
```
-时间复杂度 : O(n)
-空间复杂度 : O(n)
+复杂度:O(N) + O(N)
```java
-public void reOrderArray(int[] array) {
+public void reOrderArray(int[] nums) {
int oddCnt = 0;
- for (int num : array) if (num % 2 == 1) oddCnt++;
- int[] copy = array.clone();
+ for (int val : nums) if (val % 2 == 1) oddCnt++;
+ int[] copy = nums.clone();
int i = 0, j = oddCnt;
for (int num : copy) {
- if (num % 2 == 1) array[i++] = num;
- else array[j++] = num;
+ if (num % 2 == 1) nums[i++] = num;
+ else nums[j++] = num;
}
}
```
-## 22. 链表中倒数第 k 个结点
+# 22. 链表中倒数第 K 个结点
+
+## 解题思路
+
+设链表的长度为 N。设两个指针 P1 和 P2,先让 P1 移动 K 个节点,则还有 N - K 个节点可以移动。此时让 P1 和 P2 同时移动,可以知道当 P1 移动到链表结尾时,P2 移动到 N - K 个节点处,该位置就是倒数第 K 个节点。
+
+## 解题思路
+
+
```java
public ListNode FindKthToTail(ListNode head, int k) {
if (head == null) return null;
- ListNode fast, slow;
- fast = slow = head;
- while (fast != null && k-- > 0) fast = fast.next;
+ ListNode P1, P2;
+ P1 = P2 = head;
+ while (P1 != null && k-- > 0) P1 = P1.next;
if (k > 0) return null;
- while (fast != null) {
- fast = fast.next;
- slow = slow.next;
+ while (P1 != null) {
+ P1 = P1.next;
+ P2 = P2.next;
}
- return slow;
+ return P2;
}
```
-## 23. 链表中环的入口结点
+# 23. 链表中环的入口结点
+
+## 解题思路
+
+使用双指针,一个指针 fast 每次移动两个节点,一个指针 slow 每次移动一个节点。因为存在环,所以两个指针必定相遇在环中的某个节点上。此时 fast 移动的节点数为 x+2y+z,slow 为 x+y,由于 fast 速度比 slow 快一倍,因此 x+2y+z=2(x+y),得到 x=z。
+
+在相遇点,slow 要到环的入口点还需要移动 z 个节点,如果让 fast 重新从头开始移动,并且速度变为每次移动一个节点,那么它到环入口点还需要移动 x 个节点。在上面已经推导出 x=z,因此 fast 和 slow 将在环入口点相遇。
+
+
```java
public ListNode EntryNodeOfLoop(ListNode pHead) {
@@ -781,7 +1079,24 @@ public ListNode EntryNodeOfLoop(ListNode pHead) {
}
```
-## 24. 反转链表
+# 24. 反转链表
+
+## 解题思路
+
+### 递归
+
+```java
+public ListNode ReverseList(ListNode head) {
+ if (head == null || head.next == null) return head;
+ ListNode next = head.next;
+ head.next = null;
+ ListNode newHead = ReverseList(next);
+ next.next = head;
+ return newHead;
+}
+```
+
+### 迭代
```java
public ListNode ReverseList(ListNode head) {
@@ -796,14 +1111,38 @@ public ListNode ReverseList(ListNode head) {
}
```
-## 25. 合并两个排序的链表
+# 25. 合并两个排序的链表
+
+## 题目描述
+
+
+
+## 解题思路
+
+### 递归
+
+```java
+public ListNode Merge(ListNode list1, ListNode list2) {
+ if (list1 == null) return list2;
+ if (list2 == null) return list1;
+ if (list1.val <= list2.val) {
+ list1.next = Merge(list1.next, list2);
+ return list1;
+ } else {
+ list2.next = Merge(list1, list2.next);
+ return list2;
+ }
+}
+```
+
+### 迭代
```java
public ListNode Merge(ListNode list1, ListNode list2) {
ListNode head = new ListNode(-1);
ListNode cur = head;
while (list1 != null && list2 != null) {
- if (list1.val < list2.val) {
+ if (list1.val <= list2.val) {
cur.next = list1;
list1 = list1.next;
} else {
@@ -818,7 +1157,13 @@ public ListNode Merge(ListNode list1, ListNode list2) {
}
```
-## 26. 树的子结构
+# 26. 树的子结构
+
+## 题目描述
+
+
+
+## 解题思路
```java
public boolean HasSubtree(TreeNode root1, TreeNode root2) {
@@ -835,22 +1180,36 @@ private boolean isSubtree(TreeNode root1, TreeNode root2) {
}
```
-# 第四章 解决面试题的思路
+# 27. 二叉树的镜像
-## 27. 二叉树的镜像
+## 题目描述
+
+
+
+## 解题思路
```java
public void Mirror(TreeNode root) {
if (root == null) return;
- TreeNode t = root.left;
- root.left = root.right;
- root.right = t;
+ swap(root);
Mirror(root.left);
Mirror(root.right);
}
+
+private void swap(TreeNode root) {
+ TreeNode t = root.left;
+ root.left = root.right;
+ root.right = t;
+}
```
-## 28.1 对称的二叉树
+# 28 对称的二叉树
+
+## 题目描述
+
+
+
+## 解题思路
```java
boolean isSymmetrical(TreeNode pRoot) {
@@ -866,26 +1225,15 @@ boolean isSymmetrical(TreeNode t1, TreeNode t2) {
}
```
-## 28.2 平衡二叉树
+# 29. 顺时针打印矩阵
-```java
-private boolean isBalanced = true;
+## 题目描述
-public boolean IsBalanced_Solution(TreeNode root) {
- height(root);
- return isBalanced;
-}
+下图的矩阵顺时针打印结果为:1, 2, 3, 4, 8, 12, 16, 15, 14, 13, 9, 5, 6, 7, 11, 10
-private int height(TreeNode root) {
- if (root == null) return 0;
- int left = height(root.left);
- int right = height(root.right);
- if (Math.abs(left - right) > 1) isBalanced = false;
- return 1 + Math.max(left, right);
-}
-```
+
-## 29. 顺时针打印矩阵
+## 解题思路
```java
public ArrayList printMatrix(int[][] matrix) {
@@ -902,7 +1250,13 @@ public ArrayList printMatrix(int[][] matrix) {
}
```
-## 30. 包含 min 函数的栈
+# 30. 包含 min 函数的栈
+
+## 题目描述
+
+定义栈的数据结构,请在该类型中实现一个能够得到栈最小元素的 min 函数。
+
+## 解题思路
```java
private Stack stack = new Stack<>();
@@ -930,24 +1284,46 @@ public int min() {
}
```
-## 31. 栈的压入、弹出序列
+# 31. 栈的压入、弹出序列
+
+## 题目描述
+
+输入两个整数序列,第一个序列表示栈的压入顺序,请判断第二个序列是否为该栈的弹出顺序。假设压入栈的所有数字均不相等。例如序列 1,2,3,4,5 是某栈的压入顺序,序列 4,5,3,2,1 是该压栈序列对应的一个弹出序列,但 4,3,5,1,2 就不可能是该压栈序列的弹出序列。
+
+## 解题思路
+
+使用一个栈来模拟压入弹出操作。
```java
public boolean IsPopOrder(int[] pushA, int[] popA) {
int n = pushA.length;
Stack stack = new Stack<>();
- for (int i = 0, j = 0; i < n; i++) {
- stack.push(pushA[i]);
- while (j < n && stack.peek() == popA[j]) {
+ for (int pushIndex = 0, popIndex = 0; pushIndex < n; pushIndex++) {
+ stack.push(pushA[pushIndex]);
+ while (popIndex < n && stack.peek() == popA[popIndex]) {
stack.pop();
- j++;
+ popIndex++;
}
}
return stack.isEmpty();
}
```
-## 32.1 从上往下打印二叉树
+# 32.1 从上往下打印二叉树
+
+## 题目描述
+
+从上往下打印出二叉树的每个节点,同层节点从左至右打印。
+
+例如,以下二叉树层次遍历的结果为:1,2,3,4,5,6,7
+
+
+
+## 解题思路
+
+使用队列来进行层次遍历。
+
+不需要使用两个队列分别存储当前层的节点和下一层的节点,因为在开始遍历一层的节点时,当前队列中的节点数就是当前层的节点数,只要控制遍历这么多节点数,就能保证这次遍历的都是当前层的节点。
```java
public ArrayList PrintFromTopToBottom(TreeNode root) {
@@ -968,7 +1344,13 @@ public ArrayList PrintFromTopToBottom(TreeNode root) {
}
```
-## 32.3 把二叉树打印成多行
+# 32.2 把二叉树打印成多行
+
+## 题目描述
+
+和上题几乎一样。
+
+## 解题思路
```java
ArrayList> Print(TreeNode pRoot) {
@@ -991,7 +1373,13 @@ ArrayList> Print(TreeNode pRoot) {
}
```
-## 32.3 按之字形顺序打印二叉树
+# 32.3 按之字形顺序打印二叉树
+
+## 题目描述
+
+请实现一个函数按照之字形打印二叉树,即第一行按照从左到右的顺序打印,第二层按照从右至左的顺序打印,第三行按照从左到右的顺序打印,其他行以此类推。
+
+## 解题思路
```java
public ArrayList> Print(TreeNode pRoot) {
@@ -1009,95 +1397,112 @@ public ArrayList> Print(TreeNode pRoot) {
if (node.left != null) queue.add(node.left);
if (node.right != null) queue.add(node.right);
}
- if (reverse) {
- Collections.reverse(list);
- reverse = false;
- } else {
- reverse = true;
- }
+ if (reverse) Collections.reverse(list);
+ reverse = !reverse;
ret.add(list);
}
return ret;
}
```
+# 33. 二叉搜索树的后序遍历序列
-## 33. 二叉搜索树的后序遍历序列
+## 题目描述
+
+输入一个整数数组,判断该数组是不是某二叉搜索树的后序遍历的结果。假设输入的数组的任意两个数字都互不相同。
+
+例如,下图是后序遍历序列 3,1,2 所对应的二叉搜索树。
+
+
+
+## 解题思路
```java
public boolean VerifySquenceOfBST(int[] sequence) {
- if (sequence.length == 0) return false;
+ if (sequence == null || sequence.length == 0) return false;
return verify(sequence, 0, sequence.length - 1);
}
-private boolean verify(int[] sequence, int start, int end) {
- if (end - start <= 1) return true;
- int rootVal = sequence[end];
- int cutIdx = start;
- while (cutIdx < end) {
- if (sequence[cutIdx] > rootVal) break;
- cutIdx++;
- }
- for (int i = cutIdx + 1; i < end; i++) {
+private boolean verify(int[] sequence, int first, int last) {
+ if (last - first <= 1) return true;
+ int rootVal = sequence[last];
+ int cutIndex = first;
+ while (cutIndex < last && sequence[cutIndex] <= rootVal) cutIndex++;
+ for (int i = cutIndex + 1; i < last; i++) {
if (sequence[i] < rootVal) return false;
}
- return verify(sequence, start, cutIdx - 1) && verify(sequence, cutIdx, end - 1);
+ return verify(sequence, first, cutIndex - 1) && verify(sequence, cutIndex, last - 1);
}
```
-## 34. 二叉树中和为某一值的路径
+# 34. 二叉树中和为某一值的路径
+
+## 题目描述
+
+输入一颗二叉树和一个整数,打印出二叉树中结点值的和为输入整数的所有路径。路径定义为从树的根结点开始往下一直到叶结点所经过的结点形成一条路径。
+
+下图的二叉树有两条和为 22 的路径:10, 5, 7 和 10, 12
+
+
+
+## 解题思路
```java
private ArrayList> ret = new ArrayList<>();
public ArrayList> FindPath(TreeNode root, int target) {
- dfs(root, target, 0, new ArrayList<>());
+ dfs(root, target, new ArrayList<>());
return ret;
}
-private void dfs(TreeNode node, int target, int curSum, ArrayList path) {
+private void dfs(TreeNode node, int target, ArrayList path) {
if (node == null) return;
- curSum += node.val;
path.add(node.val);
- if (curSum == target && node.left == null && node.right == null) {
+ target -= node.val;
+ if (target == 0 && node.left == null && node.right == null) {
ret.add(new ArrayList(path));
} else {
- dfs(node.left, target, curSum, path);
- dfs(node.right, target, curSum, path);
+ dfs(node.left, target, path);
+ dfs(node.right, target, path);
}
path.remove(path.size() - 1);
}
```
-## 35. 复杂链表的复制
+# 35. 复杂链表的复制
-**题目描述**
+## 题目描述
-输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点),返回结果为复制后复杂链表的 head。(注意,输出结果中请不要返回参数中的节点引用,否则判题程序会直接返回空)
+输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点),返回结果为复制后复杂链表的 head。
+
+
+
+## 解题思路
第一步,在每个节点的后面插入复制的节点。
-
+
第二步,对复制节点的 random 链接进行赋值。
-
+
第三步,拆分。
-
-
+
```java
public RandomListNode Clone(RandomListNode pHead) {
- if (pHead == null) return null;
+ if (pHead == null) {
+ return null;
+ }
// 插入新节点
RandomListNode cur = pHead;
while (cur != null) {
- RandomListNode node = new RandomListNode(cur.label);
- node.next = cur.next;
- cur.next = node;
- cur = node.next;
+ RandomListNode clone = new RandomListNode(cur.label);
+ clone.next = cur.next;
+ cur.next = clone;
+ cur = clone.next;
}
// 建立 random 链接
cur = pHead;
@@ -1109,291 +1514,321 @@ public RandomListNode Clone(RandomListNode pHead) {
cur = clone.next;
}
// 拆分
- RandomListNode pCloneHead = pHead.next;
cur = pHead;
+ RandomListNode pCloneHead = pHead.next;
while (cur.next != null) {
- RandomListNode t = cur.next;
- cur.next = t.next;
- cur = t;
+ RandomListNode next = cur.next;
+ cur.next = next.next;
+ cur = next;
}
return pCloneHead;
}
```
-## 36. 二叉搜索树与双向链表
+# 36. 二叉搜索树与双向链表
-**题目描述**
+## 题目描述
输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表。要求不能创建任何新的结点,只能调整树中结点指针的指向。
+
+
+## 解题思路
+
```java
private TreeNode pre = null;
-public TreeNode Convert(TreeNode pRootOfTree) {
- if(pRootOfTree == null) return null;
- inOrder(pRootOfTree);
- while(pRootOfTree.left != null) pRootOfTree = pRootOfTree.left;
- return pRootOfTree;
+private TreeNode head = null;
+
+public TreeNode Convert(TreeNode root) {
+ if (root == null) return null;
+ inOrder(root);
+ return head;
}
private void inOrder(TreeNode node) {
- if(node == null) return;
+ if (node == null) return;
inOrder(node.left);
node.left = pre;
- if(pre != null) pre.right = node;
+ if (pre != null) pre.right = node;
pre = node;
+ if (head == null) head = node;
inOrder(node.right);
}
```
-## 37. 序列化二叉树
+# 37. 序列化二叉树
+
+## 题目描述
+
+请实现两个函数,分别用来序列化和反序列化二叉树。
+
+## 解题思路
```java
-private String serizeString = "";
+public class Solution {
-String Serialize(TreeNode root) {
- if (root == null) return "#";
- return root.val + " " + Serialize(root.left) + " "
- + Serialize(root.right);
-}
+ private String deserializeStr;
-TreeNode Deserialize(String str) {
- this.serizeString = str;
- return Deserialize();
-}
-
-private TreeNode Deserialize() {
- if (this.serizeString.length() == 0) return null;
- int idx = this.serizeString.indexOf(" ");
- if (idx == -1) return null;
- String sub = this.serizeString.substring(0, idx);
- this.serizeString = this.serizeString.substring(idx + 1);
- if (sub.equals("#")) {
- return null;
+ public String Serialize(TreeNode root) {
+ if (root == null) return "#";
+ return root.val + " " + Serialize(root.left) + " " + Serialize(root.right);
+ }
+
+ public TreeNode Deserialize(String str) {
+ deserializeStr = str;
+ return Deserialize();
+ }
+
+ private TreeNode Deserialize() {
+ if (deserializeStr.length() == 0) return null;
+ int index = deserializeStr.indexOf(" ");
+ String node = index == -1 ? deserializeStr : deserializeStr.substring(0, index);
+ deserializeStr = index == -1 ? "" : deserializeStr.substring(index + 1);
+ if (node.equals("#")) return null;
+ int val = Integer.valueOf(node);
+ TreeNode t = new TreeNode(val);
+ t.left = Deserialize();
+ t.right = Deserialize();
+ return t;
}
- int val = Integer.valueOf(sub);
- TreeNode t = new TreeNode(val);
- t.left = Deserialize();
- t.right = Deserialize();
- return t;
}
```
-## 38. 字符串的排列
+# 38. 字符串的排列
-**题目描述**
+## 题目描述
-输入一个字符串 , 按字典序打印出该字符串中字符的所有排列。例如输入字符串 abc, 则打印出由字符 a, b, c 所能排列出来的所有字符串 abc, acb, bac, bca, cab 和 cba。
+输入一个字符串,按字典序打印出该字符串中字符的所有排列。例如输入字符串 abc,则打印出由字符 a, b, c 所能排列出来的所有字符串 abc, acb, bac, bca, cab 和 cba。
+
+## 解题思路
```java
private ArrayList ret = new ArrayList<>();
public ArrayList Permutation(String str) {
- if (str.length() == 0) return new ArrayList<>();
+ if (str.length() == 0) return ret;
char[] chars = str.toCharArray();
Arrays.sort(chars);
- backtracking(chars, new boolean[chars.length], "");
+ backtracking(chars, new boolean[chars.length], new StringBuffer());
return ret;
}
-private void backtracking(char[] chars, boolean[] used, String s) {
+private void backtracking(char[] chars, boolean[] hasUsed, StringBuffer s) {
if (s.length() == chars.length) {
- ret.add(s);
+ ret.add(s.toString());
return;
}
for (int i = 0; i < chars.length; i++) {
- if (used[i]) continue;
- if (i != 0 && chars[i] == chars[i - 1] && !used[i - 1]) continue; // 保证不重复
- used[i] = true;
- backtracking(chars, used, s + chars[i]);
- used[i] = false;
+ if (hasUsed[i]) continue;
+ if (i != 0 && chars[i] == chars[i - 1] && !hasUsed[i - 1]) continue; // 保证不重复
+ hasUsed[i] = true;
+ s.append(chars[i]);
+ backtracking(chars, hasUsed, s);
+ s.deleteCharAt(s.length() - 1);
+ hasUsed[i] = false;
}
}
```
-# 第五章 优化时间和空间效率
+# 39. 数组中出现次数超过一半的数字
-## 39. 数组中出现次数超过一半的数字
+## 解题思路
+
+多数投票问题,可以利用 Boyer-Moore Majority Vote Algorithm 来解决这个问题,使得时间复杂度为 O(N)。
+
+使用 cnt 来统计一个元素出现的次数,当遍历到的元素和统计元素不相等时,令 cnt--。如果前面查找了 i 个元素,且 cnt == 0 ,说明前 i 个元素没有 majority,或者有 majority,但是出现的次数少于 i / 2 ,因为如果多于 i / 2 的话 cnt 就一定不会为 0 。此时剩下的 n - i 个元素中,majority 的数目依然多于 (n - i) / 2,因此继续查找就能找出 majority。
```java
-public int MoreThanHalfNum_Solution(int[] array) {
- int cnt = 1, num = array[0];
- for (int i = 1; i < array.length; i++) {
- if (array[i] == num) cnt++;
- else cnt--;
+public int MoreThanHalfNum_Solution(int[] nums) {
+ int majority = nums[0];
+ for (int i = 1, cnt = 1; i < nums.length; i++) {
+ cnt = nums[i] == majority ? cnt + 1 : cnt - 1;
if (cnt == 0) {
- num = array[i];
+ majority = nums[i];
cnt = 1;
}
}
- cnt = 0;
- for (int i = 0; i < array.length; i++) {
- if (num == array[i]) cnt++;
- }
- return cnt > array.length / 2 ? num : 0;
+ int cnt = 0;
+ for (int val : nums) if (val == majority) cnt++;
+ return cnt > nums.length / 2 ? majority : 0;
}
```
+# 40. 最小的 K 个数
-## 40. 最小的 K 个数
+## 解题思路
-构建大小为 k 的小顶堆。
+### 快速选择
-时间复杂度:O(nlgk)
-空间复杂度:O(k)
+- 复杂度:O(N) + O(1)
+- 只有当允许修改数组元素时才可以使用
+
+快速排序的 partition() 方法,会返回一个整数 j 使得 a[l..j-1] 小于等于 a[j],且 a[j+1..h] 大于等于 a[j],此时 a[j] 就是数组的第 j 大元素。可以利用这个特性找出数组的第 K 个元素,这种找第 K 个元素的算法称为快速选择算法。
+
+找到第 K 个元素之后,就可以再遍历一次数组,所有小于等于该元素的数组元素都是最小的 K 个数。
```java
-public ArrayList GetLeastNumbers_Solution(int[] input, int k) {
- if (k > input.length || k <= 0) return new ArrayList<>();
- PriorityQueue pq = new PriorityQueue<>((o1, o2) -> o2 - o1);
- for (int num : input) {
- pq.add(num);
- if (pq.size() > k) {
- pq.poll();
- }
- }
- ArrayList ret = new ArrayList<>(pq);
- return ret;
-}
-```
-
-利用快速选择
-
-时间复杂度:O(n)
-空间复杂度:O(1)
-
-```java
-public ArrayList GetLeastNumbers_Solution(int[] input, int k) {
- if (k > input.length || k <= 0) return new ArrayList<>();
- int kthSmallest = findKthSmallest(input, k - 1);
+public ArrayList GetLeastNumbers_Solution(int[] nums, int k) {
+ if (k > nums.length || k <= 0) return new ArrayList<>();
+ int kthSmallest = findKthSmallest(nums, k - 1);
ArrayList ret = new ArrayList<>();
- for (int num : input) {
- if(num <= kthSmallest && ret.size() < k) ret.add(num);
+ for (int val : nums) {
+ if (val <= kthSmallest && ret.size() < k) {
+ ret.add(val);
+ }
}
return ret;
}
public int findKthSmallest(int[] nums, int k) {
- int lo = 0;
- int hi = nums.length - 1;
- while (lo < hi) {
- int j = partition(nums, lo, hi);
- if (j < k) {
- lo = j + 1;
- } else if (j > k) {
- hi = j - 1;
- } else {
- break;
- }
+ int l = 0, h = nums.length - 1;
+ while (l < h) {
+ int j = partition(nums, l, h);
+ if (j == k) break;
+ if (j > k) h = j - 1;
+ else l = j + 1;
}
return nums[k];
}
-private int partition(int[] a, int lo, int hi) {
- int i = lo;
- int j = hi + 1;
+private int partition(int[] nums, int l, int h) {
+ int i = l, j = h + 1;
while (true) {
- while (i < hi && less(a[++i], a[lo])) ;
- while (j > lo && less(a[lo], a[--j])) ;
- if (i >= j) {
- break;
- }
- exch(a, i, j);
+ while (i < h && nums[++i] < nums[l]) ;
+ while (j > l && nums[l] < nums[--j]) ;
+ if (i >= j) break;
+ swap(nums, i, j);
}
- exch(a, lo, j);
+ swap(nums, l, j);
return j;
}
-private void exch(int[] a, int i, int j) {
- final int tmp = a[i];
- a[i] = a[j];
- a[j] = tmp;
-}
-
-private boolean less(int v, int w) {
- return v < w;
+private void swap(int[] nums, int i, int j) {
+ int t = nums[i]; nums[i] = nums[j]; nums[j] = t;
}
```
-## 41.1 数据流中的中位数
+### 大小为 K 的最小堆
+- 复杂度:O(NlogK) + O(K)
+- 特别适合处理海量数据
-**题目描述**
+应该使用大顶堆来维护最小堆,而不能直接创建一个小顶堆并设置一个大小,企图让小顶堆中的元素都是最小元素。
+
+维护一个大小为 K 的最小堆过程如下:在添加一个元素之后,如果大顶堆的大小大于 K,那么需要将大顶堆的堆顶元素去除。
+
+```java
+public ArrayList GetLeastNumbers_Solution(int[] nums, int k) {
+ if (k > nums.length || k <= 0) return new ArrayList<>();
+ PriorityQueue maxHeap = new PriorityQueue<>((o1, o2) -> o2 - o1);
+ for (int num : nums) {
+ maxHeap.add(num);
+ if (maxHeap.size() > k) {
+ maxHeap.poll();
+ }
+ }
+ ArrayList ret = new ArrayList<>(maxHeap);
+ return ret;
+}
+```
+
+# 41.1 数据流中的中位数
+
+## 题目描述
如何得到一个数据流中的中位数?如果从数据流中读出奇数个数值,那么中位数就是所有数值排序之后位于中间的数值。如果从数据流中读出偶数个数值,那么中位数就是所有数值排序之后中间两个数的平均值。
+## 解题思路
+
```java
-private PriorityQueue maxHeap = new PriorityQueue<>((o1, o2) -> o2-o1); // 实现左边部分
-private PriorityQueue minHeep = new PriorityQueue<>(); // 实现右边部分,右边部分所有元素大于左边部分
-private int cnt = 0;
+public class Solution {
+ // 大顶堆,存储左半边元素
+ private PriorityQueue left = new PriorityQueue<>((o1, o2) -> o2 - o1);
+ // 小顶堆,存储右半边元素,并且右半边元素都大于左半边
+ private PriorityQueue right = new PriorityQueue<>();
+ // 当前数据流读入的元素个数
+ private int N = 0;
-public void Insert(Integer num) {
- // 插入要保证两个堆存于平衡状态
- if(cnt % 2 == 0) {
- // 为偶数的情况下插入到最小堆,先经过最大堆筛选,这样就能保证最大堆中的元素都小于最小堆中的元素
- maxHeap.add(num);
- minHeep.add(maxHeap.poll());
- } else {
- minHeep.add(num);
- maxHeap.add(minHeep.poll());
+ public void Insert(Integer val) {
+ // 插入要保证两个堆存于平衡状态
+ if (N % 2 == 0) {
+ // N 为偶数的情况下插入到右半边。
+ // 因为右半边元素都要大于左半边,但是新插入的元素不一定比左半边元素来的大,
+ // 因此需要先将元素插入左半边,然后利用左半边为大顶堆的特点,取出堆顶元素即为最大元素,此时插入右半边
+ left.add(val);
+ right.add(left.poll());
+ } else {
+ right.add(val);
+ left.add(right.poll());
+ }
+ N++;
}
- cnt++;
-}
-public Double GetMedian() {
- if(cnt % 2 == 0) {
- return (maxHeap.peek() + minHeep.peek()) / 2.0;
- } else {
- return (double) minHeep.peek();
+ public Double GetMedian() {
+ if (N % 2 == 0) {
+ return (left.peek() + right.peek()) / 2.0;
+ } else {
+ return (double) right.peek();
+ }
}
}
```
-## 41.2 字符流中第一个不重复的字符
+# 41.2 字符流中第一个不重复的字符
-**题目描述**
+## 题目描述
请实现一个函数用来找出字符流中第一个只出现一次的字符。例如,当从字符流中只读出前两个字符 "go" 时,第一个只出现一次的字符是 "g"。当从该字符流中读出前六个字符“google" 时,第一个只出现一次的字符是 "l"。
+## 解题思路
+
```java
-//Insert one char from stringstream
-private int[] cnts = new int[256];
-private Queue queue = new LinkedList<>();
+public class Solution {
+ private int[] cnts = new int[256];
+ private Queue queue = new LinkedList<>();
-public void Insert(char ch) {
- cnts[ch]++;
- queue.add(ch);
- while (!queue.isEmpty() && cnts[queue.peek()] > 1) {
- queue.poll();
+ public void Insert(char ch) {
+ cnts[ch]++;
+ queue.add(ch);
+ while (!queue.isEmpty() && cnts[queue.peek()] > 1) {
+ queue.poll();
+ }
}
-}
-//return the first appearence once char in current stringstream
-public char FirstAppearingOnce() {
- if (queue.isEmpty()) return '#';
- return queue.peek();
+ public char FirstAppearingOnce() {
+ if (queue.isEmpty()) return '#';
+ return queue.peek();
+ }
}
```
+# 42. 连续子数组的最大和
-## 42. 连续子数组的最大和
+## 题目描述
+
+{6,-3,-2,7,-15,1,2,2},连续子数组的最大和为 8(从第 0 个开始,到第 3 个为止)。
+
+## 解题思路
```java
-public int FindGreatestSumOfSubArray(int[] array) {
- if(array.length == 0) return 0;
+public int FindGreatestSumOfSubArray(int[] nums) {
+ if (nums.length == 0) return 0;
int ret = Integer.MIN_VALUE;
int sum = 0;
- for(int num : array) {
- if(sum <= 0) sum = num;
- else sum += num;
+ for (int val : nums) {
+ if (sum <= 0) sum = val;
+ else sum += val;
ret = Math.max(ret, sum);
}
return ret;
}
```
-## 43. 从 1 到 n 整数中 1 出现的次数
+# 43. 从 1 到 n 整数中 1 出现的次数
-解题参考:[Leetcode : 233. Number of Digit One](https://leetcode.com/problems/number-of-digit-one/discuss/64381/4+-lines-O(log-n)-C++JavaPython)
+## 解题思路
+
+> [Leetcode : 233. Number of Digit One](https://leetcode.com/problems/number-of-digit-one/discuss/64381/4+-lines-O(log-n)-C++JavaPython)
```java
public int NumberOf1Between1AndN_Solution(int n) {
@@ -1406,53 +1841,67 @@ public int NumberOf1Between1AndN_Solution(int n) {
}
```
-## 44. 数字序列中的某一位数字
+# 44. 数字序列中的某一位数字
-**题目描述**
+## 题目描述
数字以 0123456789101112131415... 的格式序列化到一个字符串中,求这个字符串的第 index 位。
+## 解题思路
+
```java
-int digitAtIndex(int index) {
+public int digitAtIndex(int index) {
if (index < 0) return -1;
int digit = 1;
while (true) {
int amount = getAmountOfDigit(digit);
int totalAmount = amount * digit;
- if (index < totalAmount) return digitAtIndex(index, digit);
+ if (index < totalAmount) {
+ return digitAtIndex(index, digit);
+ }
index -= totalAmount;
digit++;
}
}
+/**
+ * digit 位数的数字组成的字符串长度
+ * 例如 digit = 2,return 90
+ */
private int getAmountOfDigit(int digit) {
if (digit == 1) return 10;
- return (int) Math.pow(10, digit - 1);
+ return (int) Math.pow(10, digit - 1) * 9;
}
-private int digitAtIndex(int index, int digits) {
- int number = beginNumber(digits) + index / digits;
- int remain = index % digits;
+/**
+ * 在 digit 位数组成的字符串中,第 index 个数
+ */
+private int digitAtIndex(int index, int digit) {
+ int number = beginNumber(digit) + index / digit;
+ int remain = index % digit;
return (number + "").charAt(remain) - '0';
}
-private int beginNumber(int digits) {
- if (digits == 1) return 0;
- return (int) Math.pow(10, digits - 1);
-}
-
-public static void main(String[] args) {
- Solution solution = new Solution();
- System.out.println(solution.digitAtIndex(1001));
+/**
+ * digit 位数的起始数字
+ * 例如 digit = 2 return 10
+ */
+private int beginNumber(int digit) {
+ if (digit == 1) return 0;
+ return (int) Math.pow(10, digit - 1);
}
```
-## 45. 把数组排成最小的数
+# 45. 把数组排成最小的数
-**题目描述**
+## 题目描述
输入一个正整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个。例如输入数组 {3,32,321},则打印出这三个数字能排成的最小数字为 321323。
+## 解题思路
+
+可以看成是一个排序问题,在比较两个字符串 S1 和 S2 的大小时,应该比较的是 S1+S2 和 S2+S1 的大小,如果 S1+S2 < S2+S1,那么应该把 S1 排在前面,否则应该把 S2 排在前面。
+
```java
public String PrintMinNumber(int[] numbers) {
int n = numbers.length;
@@ -1465,11 +1914,13 @@ public String PrintMinNumber(int[] numbers) {
}
```
-## 46. 把数字翻译成字符串
+# 46. 把数字翻译成字符串
-**题目描述**
+## 题目描述
-给定一个数字,按照如下规则翻译成字符串:0 翻译成“a”,1 翻译成“b”...25 翻译成“z”。一个数字有多种翻译可能,例如 12258 一共有 5 种,分别是 bccfi,bwfi,bczi,mcfi,mzi。实现一个函数,用来计算一个数字有多少种不同的翻译方法。
+给定一个数字,按照如下规则翻译成字符串:0 翻译成“a”,1 翻译成“b”... 25 翻译成“z”。一个数字有多种翻译可能,例如 12258 一共有 5 种,分别是 bccfi,bwfi,bczi,mcfi,mzi。实现一个函数,用来计算一个数字有多少种不同的翻译方法。
+
+## 解题思路
```java
public int getTranslationCount(String number) {
@@ -1487,11 +1938,11 @@ public int getTranslationCount(String number) {
}
```
-## 47. 礼物的最大价值
+# 47. 礼物的最大价值
-**题目描述**
+## 题目描述
-在一个 m * n 的棋盘的每一个格都放有一个礼物,每个礼物都有一定价值(大于 0)。从左上角开始拿礼物,每次向右或向下移动一格,直到右下角结束。给定一个棋盘,求拿到礼物的最大价值。例如,对于如下棋盘
+在一个 m\*n 的棋盘的每一个格都放有一个礼物,每个礼物都有一定价值(大于 0)。从左上角开始拿礼物,每次向右或向下移动一格,直到右下角结束。给定一个棋盘,求拿到礼物的最大价值。例如,对于如下棋盘
```
1 10 3 8
@@ -1502,7 +1953,7 @@ public int getTranslationCount(String number) {
礼物的最大价值为 1+12+5+7+7+16+5=53。
-**解题思路**
+## 解题思路
应该用动态规划求解,而不是深度优先搜索,深度优先搜索过于复杂,不是最优解。
@@ -1522,58 +1973,69 @@ public int getMaxValue(int[][] values) {
}
```
-## 48. 最长不含重复字符的子字符串
+# 48. 最长不含重复字符的子字符串
-**题目描述**
+## 题目描述
输入一个字符串(只包含 a\~z 的字符),求其最长不含重复字符的子字符串的长度。例如对于 arabcacfr,最长不含重复字符的子字符串为 acfr,长度为 4。
+## 解题思路
+
```java
public int longestSubStringWithoutDuplication(String str) {
int curLen = 0;
int maxLen = 0;
- int[] position = new int[26];
+ int[] indexs = new int[26];
+ Arrays.fill(indexs, -1);
for (int i = 0; i < str.length(); i++) {
int c = str.charAt(i) - 'a';
- int preIndex = position[c];
+ int preIndex = indexs[c];
if (preIndex == -1 || i - preIndex > curLen) curLen++;
else {
maxLen = Math.max(maxLen, curLen);
curLen = i - preIndex;
}
- position[c] = i;
+ indexs[c] = i;
}
maxLen = Math.max(maxLen, curLen);
return maxLen;
}
```
-## 49. 丑数
+# 49. 丑数
-**题目描述**
+## 题目描述
把只包含因子 2、3 和 5 的数称作丑数(Ugly Number)。例如 6、8 都是丑数,但 14 不是,因为它包含因子 7。 习惯上我们把 1 当做是第一个丑数。求按从小到大的顺序的第 N 个丑数。
+## 解题思路
+
```java
public int GetUglyNumber_Solution(int index) {
if (index <= 6) return index;
int i2 = 0, i3 = 0, i5 = 0;
- int cnt = 1;
int[] dp = new int[index];
dp[0] = 1;
- while (cnt < index) {
+ for (int i = 1; i < index; i++) {
int n2 = dp[i2] * 2, n3 = dp[i3] * 3, n5 = dp[i5] * 5;
- int tmp = Math.min(n2, Math.min(n3, n5));
- dp[cnt++] = tmp;
- if (tmp == n2) i2++;
- if (tmp == n3) i3++;
- if (tmp == n5) i5++;
+ dp[i] = Math.min(n2, Math.min(n3, n5));
+ if (dp[i] == n2) i2++;
+ if (dp[i] == n3) i3++;
+ if (dp[i] == n5) i5++;
}
return dp[index - 1];
}
```
-## 50. 第一个只出现一次的字符位置
+# 50. 第一个只出现一次的字符位置
+
+## 题目描述
+
+在一个字符串 (1<=字符串长度 <=10000,全部由字母组成) 中找到第一个只出现一次的字符,并返回它的位置。
+
+## 解题思路
+
+最直观的解法是使用 HashMap 对出现次数进行统计,但是考虑到要统计的字符范围有限,因此可以使用整型数组代替 HashMap。
```java
public int FirstNotRepeatingChar(String str) {
@@ -1584,99 +2046,174 @@ public int FirstNotRepeatingChar(String str) {
}
```
-## 51. 数组中的逆序对
+以上实现的空间复杂度还不是最优的。考虑到只需要找到只出现一次的字符,那么我们只需要统计的次数信息只有 0,1,更大,那么使用两个比特位就能存储这些信息。
+
+```java
+public int FirstNotRepeatingChar(String str) {
+ BitSet bs1 = new BitSet(256);
+ BitSet bs2 = new BitSet(256);
+ for (char c : str.toCharArray()) {
+ if (!bs1.get(c) && !bs2.get(c)) bs1.set(c); // 0 0
+ else if (bs1.get(c) && !bs2.get(c)) bs2.set(c); // 0 1
+ }
+ for (int i = 0; i < str.length(); i++) {
+ char c = str.charAt(i);
+ if (bs1.get(c) && !bs2.get(c)) return i;
+ }
+ return -1;
+}
+```
+
+# 51. 数组中的逆序对
+
+## 题目描述
+
+在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数 P。
+
+## 解题思路
```java
private long cnt = 0;
+private int[] tmp; // 在这里创建辅助数组,而不是在 merge() 递归函数中创建
-public int InversePairs(int[] array) {
- mergeSortUp2Down(array, 0, array.length - 1);
+public int InversePairs(int[] nums) {
+ tmp = new int[nums.length];
+ mergeSortUp2Down(nums, 0, nums.length - 1);
return (int) (cnt % 1000000007);
}
-private void mergeSortUp2Down(int[] a, int start, int end) {
- if (end - start < 1) return;
- int mid = start + (end - start) / 2;
- mergeSortUp2Down(a, start, mid);
- mergeSortUp2Down(a, mid + 1, end);
- merge(a, start, mid, end);
+private void mergeSortUp2Down(int[] nums, int first, int last) {
+ if (last - first < 1) return;
+ int mid = first + (last - first) / 2;
+ mergeSortUp2Down(nums, first, mid);
+ mergeSortUp2Down(nums, mid + 1, last);
+ merge(nums, first, mid, last);
}
-private void merge(int[] a, int start, int mid, int end) {
- int[] tmp = new int[end - start + 1];
- int i = start, j = mid + 1, k = 0;
- while (i <= mid || j <= end) {
- if (i > mid) tmp[k] = a[j++];
- else if (j > end) tmp[k] = a[i++];
- else if (a[i] < a[j]) tmp[k] = a[i++];
+private void merge(int[] nums, int first, int mid, int last) {
+ int i = first, j = mid + 1, k = first;
+ while (i <= mid || j <= last) {
+ if (i > mid) tmp[k] = nums[j++];
+ else if (j > last) tmp[k] = nums[i++];
+ else if (nums[i] < nums[j]) tmp[k] = nums[i++];
else {
- tmp[k] = a[j++];
- this.cnt += mid - i + 1; // a[i] > a[j] ,说明 a[i...mid] 都大于 a[j]
+ tmp[k] = nums[j++];
+ this.cnt += mid - i + 1; // nums[i] > nums[j],说明 nums[i...mid] 都大于 nums[j]
}
k++;
}
-
- for (k = 0; k < tmp.length; k++) {
- a[start + k] = tmp[k];
+ for (k = first; k <= last; k++) {
+ nums[k] = tmp[k];
}
}
```
-## 52. 两个链表的第一个公共结点
+# 52. 两个链表的第一个公共结点
+
+## 题目描述
+
+
+
+## 解题思路
+
+设 A 的长度为 a + c,B 的长度为 b + c,其中 c 为尾部公共部分长度,可知 a + c + b = b + c + a。
+
+当访问 A 链表的指针访问到链表尾部时,令它从链表 B 的头部重新开始访问链表 B;同样地,当访问 B 链表的指针访问到链表尾部时,令它从链表 A 的头部重新开始访问链表 A。这样就能控制访问 A 和 B 两个链表的指针能同时访问到交点。
```java
public ListNode FindFirstCommonNode(ListNode pHead1, ListNode pHead2) {
ListNode l1 = pHead1, l2 = pHead2;
while (l1 != l2) {
- if (l1 == null) l1 = pHead2;
- else l1 = l1.next;
- if (l2 == null) l2 = pHead1;
- else l2 = l2.next;
+ l1 = (l1 == null) ? pHead2 : l1.next;
+ l2 = (l2 == null) ? pHead1 : l2.next;
}
return l1;
}
```
-# 第六章 面试中的各项能力
+# 53 数字在排序数组中出现的次数
-## 53 数字在排序数组中出现的次数
+## 题目描述
+
+```html
+Input:
+1, 2, 3, 3, 3, 3, 4, 6
+3
+Output:
+4
+```
+
+## 解题思路
+
+可以用二分查找找出数字在数组的最左端和最右端,找最左端和最右端在方法实现上的区别主要在于对 nums[m] == K 的处理:
+
+- 找最左端令 h = m - 1
+- 找最右端令 l = m + 1
```java
-public int GetNumberOfK(int[] array, int k) {
- int l = 0, h = array.length - 1;
+public int GetNumberOfK(int[] nums, int K) {
+ int first = getFirstK(nums, K);
+ int last = getLastK(nums, K);
+ return first == -1 || last == -1 ? 0 : last - first + 1;
+}
+
+private int getFirstK(int[] nums, int K) {
+ int l = 0, h = nums.length - 1;
while (l <= h) {
int m = l + (h - l) / 2;
- if (array[m] >= k) h = m - 1;
+ if (nums[m] >= K) h = m - 1;
else l = m + 1;
}
- int cnt = 0;
- while (l < array.length && array[l++] == k) cnt++;
- return cnt;
+ if (l > nums.length - 1 || nums[l] != K) return -1;
+ return l;
+}
+
+private int getLastK(int[] nums, int K) {
+ int l = 0, h = nums.length - 1;
+ while (l <= h) {
+ int m = l + (h - l) / 2;
+ if (nums[m] > K) h = m - 1;
+ else l = m + 1;
+ }
+ if (h < 0 || nums[h] != K) return -1;
+ return h;
}
```
-## 54. 二叉搜索树的第 k 个结点
+# 54. 二叉搜索树的第 K 个结点
+
+## 解题思路
+
+利用二叉搜索数中序遍历有序的特点。
```java
-TreeNode ret;
-int cnt = 0;
+private TreeNode ret;
+private int cnt = 0;
-TreeNode KthNode(TreeNode pRoot, int k) {
- inorder(pRoot, k);
+public TreeNode KthNode(TreeNode pRoot, int k) {
+ inOrder(pRoot, k);
return ret;
}
-private void inorder(TreeNode root, int k) {
+private void inOrder(TreeNode root, int k) {
if (root == null) return;
if (cnt > k) return;
- inorder(root.left, k);
+ inOrder(root.left, k);
cnt++;
if (cnt == k) ret = root;
- inorder(root.right, k);
+ inOrder(root.right, k);
}
```
-## 55 二叉树的深度
+# 55.1 二叉树的深度
+
+## 题目描述
+
+从根结点到叶结点依次经过的结点(含根、叶结点)形成树的一条路径,最长路径的长度为树的深度。
+
+
+
+## 解题思路
```java
public int TreeDepth(TreeNode root) {
@@ -1685,13 +2222,41 @@ public int TreeDepth(TreeNode root) {
}
```
-## 56. 数组中只出现一次的数字
+# 55.2 平衡二叉树
-**题目描述**
+## 题目描述
+
+平衡二叉树左右子树高度差不超过 1。
+
+
+
+## 解题思路
+
+```java
+private boolean isBalanced = true;
+
+public boolean IsBalanced_Solution(TreeNode root) {
+ height(root);
+ return isBalanced;
+}
+
+private int height(TreeNode root) {
+ if (root == null) return 0;
+ int left = height(root.left);
+ int right = height(root.right);
+ if (Math.abs(left - right) > 1) isBalanced = false;
+ return 1 + Math.max(left, right);
+}
+```
+
+
+# 56. 数组中只出现一次的数字
+
+## 题目描述
一个整型数组里除了两个数字之外,其他的数字都出现了两次,找出这两个数。
-**解题思路**
+## 解题思路
两个不相等的元素在位级表示上必定会有一位存在不同。
@@ -1712,11 +2277,17 @@ public void FindNumsAppearOnce(int[] array, int num1[], int num2[]) {
}
```
-## 57.1 和为 S 的两个数字
+# 57.1 和为 S 的两个数字
-**题目描述**
+## 题目描述
-输入一个递增排序的数组和一个数字 S,在数组中查找两个数,是的他们的和正好是 S,如果有多对数字的和等于 S,输出两个数的乘积最小的。
+输入一个递增排序的数组和一个数字 S,在数组中查找两个数,使得他们的和正好是 S,如果有多对数字的和等于 S,输出两个数的乘积最小的。
+
+## 解题思路
+
+使用双指针,一个指针指向元素较小的值,一个指针指向元素较大的值。指向较小元素的指针从头向尾遍历,指向较大元素的指针从尾向头遍历。
+
+如果两个指针指向元素的和 sum == target,那么得到要求的结果;如果 sum > target,移动较大的元素,使 sum 变小一些;如果 sum < target,移动较小的元素,使 sum 变大一些。
```java
public ArrayList FindNumbersWithSum(int[] array, int sum) {
@@ -1731,86 +2302,90 @@ public ArrayList FindNumbersWithSum(int[] array, int sum) {
}
```
-## 57.2 和为 S 的连续正数序列
+# 57.2 和为 S 的连续正数序列
-**题目描述**
+## 题目描述
-和为 100 的连续序列有 18, 19, 20, 21, 22
+和为 100 的连续序列有 18, 19, 20, 21, 22。
+
+## 解题思路
```java
public ArrayList> FindContinuousSequence(int sum) {
ArrayList> ret = new ArrayList<>();
- int start = 1, end = 2;
- int mid = sum / 2;
+ int first = 1, last = 2;
int curSum = 3;
- while (start <= mid && end < sum) {
+ while (first <= sum / 2 && last < sum) {
if (curSum > sum) {
- curSum -= start;
- start++;
+ curSum -= first;
+ first++;
} else if (curSum < sum) {
- end++;
- curSum += end;
+ last++;
+ curSum += last;
} else {
ArrayList list = new ArrayList<>();
- for (int i = start; i <= end; i++) {
+ for (int i = first; i <= last; i++) {
list.add(i);
}
ret.add(list);
- curSum -= start;
- start++;
- end++;
- curSum += end;
+ curSum -= first;
+ first++;
+ last++;
+ curSum += last;
}
}
return ret;
}
```
-## 58.1 翻转单词顺序列
+# 58.1 翻转单词顺序列
-**题目描述**
+## 题目描述
输入:"I am a student."
输出:"student. a am I"
+## 解题思路
+
+题目应该有一个隐含条件,就是不能用额外的空间。虽然 Java 的题目输入参数为 String 类型,需要先创建一个字符数组使得空间复杂度为 O(n),但是正确的参数类型应该和原书一样,为字符数组,并且只能使用该字符数组的空间。任何使用了额外空间的解法在面试时都会大打折扣,包括递归解法。正确的解法应该是和书上一样,先旋转每个单词,再旋转整个字符串。
+
```java
public String ReverseSentence(String str) {
if (str.length() == 0) return str;
int n = str.length();
char[] chars = str.toCharArray();
- int start = 0, end = 0;
- while (end <= n) {
- if (end == n || chars[end] == ' ') {
- reverse(chars, start, end - 1);
- start = end + 1;
+ int i = 0, j = 0;
+ while (j <= n) {
+ if (j == n || chars[j] == ' ') {
+ reverse(chars, i, j - 1);
+ i = j + 1;
}
- end++;
+ j++;
}
reverse(chars, 0, n - 1);
return new String(chars);
}
-private void reverse(char[] c, int start, int end) {
- while (start < end) {
- char t = c[start];
- c[start] = c[end];
- c[end] = t;
- start++;
- end--;
+private void reverse(char[] c, int i, int j) {
+ while(i < j) {
+ char t = c[i]; c[i] = c[j]; c[j] = t;
+ i++; j--;
}
}
```
-## 58.2 左旋转字符串
+# 58.2 左旋转字符串
-**题目描述**
+## 题目描述
对于一个给定的字符序列 S,请你把其循环左移 K 位后的序列输出。例如,字符序列 S=”abcXYZdef”, 要求输出循环左移 3 位后的结果,即“XYZdefabc”。
+## 解题思路
+
```java
public String LeftRotateString(String str, int n) {
- if (str.length() == 0) return "";
+ if(str.length() == 0) return "";
char[] c = str.toCharArray();
reverse(c, 0, n - 1);
reverse(c, n, c.length - 1);
@@ -1819,26 +2394,26 @@ public String LeftRotateString(String str, int n) {
}
private void reverse(char[] c, int i, int j) {
- while (i < j) {
- char t = c[i];
- c[i] = c[j];
- c[j] = t;
- i++;
- j--;
+ while(i < j) {
+ char t = c[i]; c[i] = c[j]; c[j] = t;
+ i++; j--;
}
}
```
-## 59. 滑动窗口的最大值
+# 59. 滑动窗口的最大值
-**题目描述**
+## 题目描述
-给定一个数组和滑动窗口的大小,找出所有滑动窗口里数值的最大值。例如,如果输入数组 {2, 3, 4, 2, 6, 2, 5, 1} 及滑动窗口的大小 3,那么一共存在 6 个滑动窗口,他们的最大值分别为 {4, 4, 6, 6, 6, 5};
+给定一个数组和滑动窗口的大小,找出所有滑动窗口里数值的最大值。例如,如果输入数组 {2, 3, 4, 2, 6, 2, 5, 1} 及滑动窗口的大小 3,那么一共存在 6 个滑动窗口,他们的最大值分别为 {4, 4, 6, 6, 6, 5}。
+
+## 解题思路
```java
public ArrayList maxInWindows(int[] num, int size) {
ArrayList ret = new ArrayList<>();
if (size > num.length || size < 1) return ret;
+ // 构建最大堆,即堆顶元素是堆的最大值。
PriorityQueue heap = new PriorityQueue((o1, o2) -> o2 - o1);
for (int i = 0; i < size; i++) heap.add(num[i]);
ret.add(heap.peek());
@@ -1851,13 +2426,17 @@ public ArrayList maxInWindows(int[] num, int size) {
}
```
-## 60. n 个骰子的点数
+# 60. n 个骰子的点数
-**题目描述**
+## 题目描述
把 n 个骰子仍在地上,求点数和为 s 的概率。
-最直观的动态规划解法,O(n2 ) 的空间复杂度。
+## 解题思路
+
+### 动态规划解法
+
+空间复杂度:O(N2 )
```java
private static int face = 6;
@@ -1872,18 +2451,20 @@ public double countProbability(int n, int s) {
for (int i = 1; i < n; i++) {
for (int j = i; j < pointNum; j++) { // 使用 i 个骰子最小点数为 i
for (int k = 1; k <= face; k++) {
- if (j - k < 0) continue;
- dp[i][j] += dp[i - 1][j - k];
+ if (j - k >= 0) {
+ dp[i][j] += dp[i - 1][j - k];
+ }
}
}
}
-
int totalNum = (int) Math.pow(6, n);
return (double) dp[n - 1][s - 1] / totalNum;
}
```
-使用旋转数组将空间复杂度降低为 O(n)
+### 动态规划解法 + 旋转数组
+
+空间复杂度:O(N)
```java
private static int face = 6;
@@ -1899,48 +2480,51 @@ public double countProbability(int n, int s) {
for (int i = 1; i < n; i++) {
for (int j = i; j < pointNum; j++) { // 使用 i 个骰子最小点数为 i
for (int k = 1; k <= face; k++) {
- if (j - k < 0) continue;
- dp[flag][j] += dp[1 - flag][j - k];
+ if (j - k >= 0) {
+ dp[flag][j] += dp[1 - flag][j - k];
+ }
}
}
+ flag = 1 - flag;
}
-
int totalNum = (int) Math.pow(6, n);
- return (double) dp[n - 1][s - 1] / totalNum;
+ return (double) dp[flag][s - 1] / totalNum;
}
```
-## 61. 扑克牌顺子
+# 61. 扑克牌顺子
-**题目描述**
+## 题目描述
五张牌,其中大小鬼为癞子,牌面大小为 0。判断是否能组成顺子。
+## 解题思路
+
```java
-public boolean isContinuous(int[] numbers) {
- if (numbers.length < 5) return false;
- Arrays.sort(numbers);
+public boolean isContinuous(int[] nums) {
+ if (nums.length < 5) return false;
+ Arrays.sort(nums);
int cnt = 0;
- for (int num : numbers) if (num == 0) cnt++;
- for (int i = cnt; i < numbers.length - 1; i++) {
- if (numbers[i + 1] == numbers[i]) return false;
- int cut = numbers[i + 1] - numbers[i] - 1;
- if (cut > cnt) return false;
- cnt -= cut;
+ for (int num : nums) if (num == 0) cnt++;
+ for (int i = cnt; i < nums.length - 1; i++) {
+ if (nums[i + 1] == nums[i]) return false;
+ int interval = nums[i + 1] - nums[i] - 1;
+ if (interval > cnt) return false;
+ cnt -= interval;
}
return true;
}
```
-## 62. 圆圈中最后剩下的数
+# 62. 圆圈中最后剩下的数
-**题目描述**
+## 题目描述
-让小朋友们围成一个大圈。然后 , 他随机指定一个数 m, 让编号为 0 的小朋友开始报数。每次喊到 m-1 的那个小朋友要出列唱首歌 , 然后可以在礼品箱中任意的挑选礼物 , 并且不再回到圈中 , 从他的下一个小朋友开始 , 继续 0...m-1 报数 .... 这样下去 .... 直到剩下最后一个小朋友 , 可以不用表演。
+让小朋友们围成一个大圈。然后,他随机指定一个数 m,让编号为 0 的小朋友开始报数。每次喊到 m-1 的那个小朋友要出列唱首歌,然后可以在礼品箱中任意的挑选礼物,并且不再回到圈中,从他的下一个小朋友开始,继续 0...m-1 报数 .... 这样下去 .... 直到剩下最后一个小朋友,可以不用表演。
-**解题思路**
+## 解题思路
-约瑟夫环
+约瑟夫环,圆圈长度为 n 的解可以看成长度为 n-1 的解再加上报数的长度 m。因为是圆圈,所以最后需要对 n 取余。
```java
public int LastRemaining_Solution(int n, int m) {
@@ -1950,12 +2534,16 @@ public int LastRemaining_Solution(int n, int m) {
}
```
-## 63. 股票的最大利润
+# 63. 股票的最大利润
-**题目描述**
+## 题目描述
可以有一次买入和一次卖出,买入必须在前。求最大收益。
+## 解题思路
+
+使用贪心策略,假设第 i 轮进行卖出操作,买入操作价格应该是 i 之前并且价格最低。
+
```java
public int maxProfit(int[] prices) {
int n = prices.length;
@@ -1970,11 +2558,13 @@ public int maxProfit(int[] prices) {
}
```
-## 64. 求 1+2+3+...+n
+# 64. 求 1+2+3+...+n
-**题目描述**
+## 题目描述
-求 1+2+3+...+n,要求不能使用乘除法、for、while、if、else、switch、case 等关键字及条件判断语句(A?B:C)
+求 1+2+3+...+n,要求不能使用乘除法、for、while、if、else、switch、case 等关键字及条件判断语句(A?B:C)。
+
+## 解题思路
```java
public int Sum_Solution(int n) {
@@ -1984,9 +2574,13 @@ public int Sum_Solution(int n) {
}
```
-## 65. 不用加减乘除做加法
+# 65. 不用加减乘除做加法
-a ^ b 表示没有考虑进位的情况下两数的和,(a & b) << 1 就是进位。递归会终止的原因是 (a & b) << 1 最右边会多一个 0,那么继续递归,进位最右边的 0 会慢慢增多,最后进位会变为 0,递归终止。
+## 解题思路
+
+a ^ b 表示没有考虑进位的情况下两数的和,(a & b) << 1 就是进位。
+
+递归会终止的原因是 (a & b) << 1 最右边会多一个 0,那么继续递归,进位最右边的 0 会慢慢增多,最后进位会变为 0,递归终止。
```java
public int Add(int num1, int num2) {
@@ -1995,38 +2589,31 @@ public int Add(int num1, int num2) {
}
```
-## 66. 构建乘积数组
+# 66. 构建乘积数组
-**题目描述**
+## 题目描述
给定一个数组 A[0, 1,..., n-1], 请构建一个数组 B[0, 1,..., n-1], 其中 B 中的元素 B[i]=A[0]\*A[1]\*...\*A[i-1]\*A[i+1]\*...\*A[n-1]。不能使用除法。
+## 解题思路
+
```java
public int[] multiply(int[] A) {
int n = A.length;
- int[][] dp = new int[n][n];
- for (int i = 0; i < n; i++) {
- dp[i][i] = A[i];
- }
- for (int i = 0; i < n; i++) {
- for (int j = i + 1; j < n; j++) {
- dp[i][j] = dp[i][j - 1] * A[j];
- }
- }
-
int[] B = new int[n];
- Arrays.fill(B, 1);
- for (int i = 0; i < n; i++) {
- if (i != 0) B[i] *= dp[0][i - 1];
- if (i != n - 1) B[i] *= dp[i + 1][n - 1];
+ for (int i = 0, product = 1; i < n; product *= A[i], i++) {
+ B[i] = product;
+ }
+ for (int i = n - 1, product = 1; i >= 0; product *= A[i], i--) {
+ B[i] *= product;
}
return B;
}
```
-# 第七章 两个面试案例
+# 67. 把字符串转换成整数
-## 67. 把字符串转换成整数
+## 解题思路
```java
public int StrToInt(String str) {
@@ -2036,16 +2623,22 @@ public int StrToInt(String str) {
int ret = 0;
for (int i = 0; i < chars.length; i++) {
if (i == 0 && (chars[i] == '+' || chars[i] == '-')) continue;
- if (chars[i] < '0' || chars[i] > '9') return 0;
+ if (chars[i] < '0' || chars[i] > '9') return 0; // 非法输入
ret = ret * 10 + (chars[i] - '0');
}
return isNegative ? -ret : ret;
}
```
-## 68. 树中两个节点的最低公共祖先
+# 68. 树中两个节点的最低公共祖先
-树是二叉查找树的最低公共祖先问题:
+## 解题思路
+
+### 二叉查找树
+
+
+
+二叉查找树中,两个节点 p, q 的公共祖先 root 满足 p.val <= root.val && root.val <= q.val,只要找到满足这个条件的最低层节点即可。换句话说,应该先考虑子树的解而不是根节点的解,二叉树的后序遍历操作满足这个特性。在本题中我们可以利用后序遍历的特性,先在左右子树中查找解,最后再考虑根节点的解。
```java
public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
@@ -2054,3 +2647,22 @@ public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
return root;
}
```
+
+### 普通二叉树
+
+
+
+在左右子树中查找两个节点的最低公共祖先,如果在其中一颗子树中查找到,那么就返回这个解,否则可以认为根节点就是最低公共祖先。
+
+```java
+public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
+ if (root == null || root == p || root == q) return root;
+ TreeNode left = lowestCommonAncestor(root.left, p, q);
+ TreeNode right = lowestCommonAncestor(root.right, p, q);
+ return left == null ? right : right == null ? left : root;
+}
+```
+
+# 参考文献
+
+- 何海涛. 剑指 Offer[M]. 电子工业出版社, 2012.
diff --git a/notes/数据库系统原理.md b/notes/数据库系统原理.md
index d4e9586e..6ebc72a8 100644
--- a/notes/数据库系统原理.md
+++ b/notes/数据库系统原理.md
@@ -1,330 +1,574 @@
-* [事务四大特性](#事务四大特性)
- * [原子性](#原子性)
- * [一致性](#一致性)
- * [隔离性](#隔离性)
- * [持久性](#持久性)
-* [数据不一致](#数据不一致)
- * [丢失修改](#丢失修改)
- * [读脏数据](#读脏数据)
- * [不可重复读](#不可重复读)
-* [隔离级别](#隔离级别)
- * [未提交读(READ UNCOMMITTED)](#未提交读read-uncommitted)
- * [提交读(READ COMMITTED)](#提交读read-committed)
- * [可重复读(REPEATABLE READ)](#可重复读repeatable-read)
- * [可串行化(SERIALIXABLE)](#可串行化serialixable)
-* [可串行化调度](#可串行化调度)
-* [封锁类型](#封锁类型)
-* [封锁粒度](#封锁粒度)
-* [封锁协议](#封锁协议)
- * [三级封锁协议](#三级封锁协议)
- * [两段锁协议](#两段锁协议)
-* [乐观锁和悲观锁](#乐观锁和悲观锁)
- * [悲观锁](#悲观锁)
- * [乐观锁](#乐观锁)
- * [MySQL 隐式和显示锁定](#mysql-隐式和显示锁定)
-* [范式](#范式)
- * [第一范式 (1NF)](#第一范式-1nf)
- * [第二范式 (2NF)](#第二范式-2nf)
- * [第三范式 (3NF)](#第三范式-3nf)
- * [BC 范式(BCNF)](#bc-范式bcnf)
-* [约束](#约束)
- * [键码](#键码)
- * [单值约束](#单值约束)
- * [引用完整性约束](#引用完整性约束)
- * [域约束](#域约束)
- * [一般约束](#一般约束)
-* [数据库的三层模式和两层映像](#数据库的三层模式和两层映像)
- * [外模式](#外模式)
- * [模式](#模式)
- * [内模式](#内模式)
- * [外模式/模式映像](#外模式模式映像)
- * [模式/内模式映像](#模式内模式映像)
-* [ER 图](#er-图)
- * [实体的三种联系](#实体的三种联系)
- * [表示出现多次的关系](#表示出现多次的关系)
- * [联系的多向性](#联系的多向性)
- * [表示子类](#表示子类)
-* [一些概念](#一些概念)
+* [一、事务](#一事务)
+ * [概念](#概念)
+ * [四大特性](#四大特性)
+* [二、并发一致性问题](#二并发一致性问题)
+ * [问题](#问题)
+ * [解决方法](#解决方法)
+* [三、封锁](#三封锁)
+ * [封锁粒度](#封锁粒度)
+ * [封锁类型](#封锁类型)
+ * [封锁协议](#封锁协议)
+* [四、隔离级别](#四隔离级别)
+* [五、多版本并发控制](#五多版本并发控制)
+ * [版本号](#版本号)
+ * [Undo 日志](#undo-日志)
+ * [实现过程](#实现过程)
+ * [快照读与当前读](#快照读与当前读)
+* [六、Next-Key Locks](#六next-key-locks)
+ * [Record Locks](#record-locks)
+ * [Grap Locks](#grap-locks)
+ * [Next-Key Locks](#next-key-locks)
+* [七、关系数据库设计理论](#七关系数据库设计理论)
+ * [函数依赖](#函数依赖)
+ * [异常](#异常)
+ * [范式](#范式)
+* [八、数据库系统概述](#八数据库系统概述)
+ * [基本术语](#基本术语)
+ * [数据库的三层模式和两层映像](#数据库的三层模式和两层映像)
+* [九、关系数据库建模](#九关系数据库建模)
+ * [ER 图](#er-图)
+* [十、约束](#十约束)
+ * [1. 键码](#1-键码)
+ * [2. 单值约束](#2-单值约束)
+ * [3. 引用完整性约束](#3-引用完整性约束)
+ * [4. 域约束](#4-域约束)
+ * [5. 一般约束](#5-一般约束)
* [参考资料](#参考资料)
-# 事务四大特性
+# 一、事务
-## 原子性
+## 概念
-要么都执行,要么都不执行。
+
-## 一致性
+事务指的是满足 ACID 特性的一系列操作。在数据库中,可以通过 Commit 提交一个事务,也可以使用 Rollback 进行回滚。
+
+## 四大特性
+
+
+
+### 1. 原子性(Atomicity)
+
+事务被视为不可分割的最小单元,要么全部提交成功,要么全部失败回滚。
+
+### 2. 一致性(Consistency)
事务执行前后都保持一致性状态。在一致性状态下,所有事务对一个数据的读取结果都是相同的。
-## 隔离性
+### 3. 隔离性(Isolation)
-多个事务单独执行,互不影响。
+一个事务所做的修改在最终提交以前,对其它事务是不可见的。
-## 持久性
+### 4. 持久性(Durability)
-即使系统发生故障,事务执行的结果也不能丢失。持久性通过数据库备份和恢复来保证。
+一旦事务提交,则其所做的修改将会永远保存到数据库中。即使系统发生崩溃,事务执行的结果也不能丢失。可以通过数据库备份和恢复来保证持久性。
-# 数据不一致
+# 二、并发一致性问题
-## 丢失修改
+在并发环境下,一个事务如果受到另一个事务的影响,那么事务操作就无法满足一致性条件。
-T1 和 T2 两个事务同时对一个数据进行修改,T1 先修改,T2 随后修改,T2 的修改覆盖了 T1 的修改。
+## 问题
-## 读脏数据
+### 1. 丢失修改
-T1 做修改后写入数据库,T2 读取这个修改后的数据,但是如果 T1 撤销了这次修改,使得 T2 读取的数据是脏数据。
+T1 和 T2 两个事务都对一个数据进行修改,T1 先修改,T2 随后修改,T2 的修改覆盖了 T1 的修改。
-## 不可重复读
+
-T1 读入某个数据,T2 对该数据做了修改,如果 T1 再读这个数据,该数据已经改变,和最开始读入的是不一样的。
+### 2. 读脏数据
-# 隔离级别
+T1 修改一个数据,T2 随后读取这个数据。如果 T1 撤销了这次修改,那么 T2 读取的数据是脏数据。
-数据库管理系统需要防止数据出现不一致,并且有多种级别可以实现,这些级别称为隔离级别。
+
-## 未提交读(READ UNCOMMITTED)
+### 3. 不可重复读
-一个事务可以读取自己的未提交数据,也被称为脏读。
+T2 读取一个数据,T1 对该数据做了修改。如果 T2 再次读取这个数据,此时读取的结果和和第一次读取的结果不同。
-## 提交读(READ COMMITTED)
+
-一个事务可以读取自己的已提交数据,但是该数据可能过后就会被其它事务改变,因此也称为不可重复读。
+### 4. 幻影读
-## 可重复读(REPEATABLE READ)
+T1 读取某个范围的数据,T2 在这个范围内插入新的数据,T1 再次读取这个范围的数据,此时读取的结果和和第一次读取的结果不同。
-保证在同一个事务中多次读取同样的记录结果是一致的。但是会出现幻读的问题,所谓幻读,指的是某个事务在读取某个范围内的记录时,其它事务会在范围内插入数据,产生幻行。
+
-## 可串行化(SERIALIXABLE)
+## 解决方法
-强制事务串行执行,避免幻读。
+产生并发不一致性问题主要原因是破坏了事务的隔离性,解决方法是通过并发控制来保证隔离性。
-# 可串行化调度
+在没有并发的情况下,事务以串行的方式执行,互不干扰,因此可以保证隔离性。在并发的情况下,如果能通过并发控制,让事务的执行结果和某一个串行执行的结果相同,就认为事务的执行结果满足隔离性要求,也就是说是正确的。把这种事务执行方式称为 **可串行化调度** 。
-如果并行的事务的执行结果和某一个串行的方式执行的结果一样,那么可以认为结果是正确的。
+**并发控制可以通过封锁来实现,但是封锁操作需要用户自己控制,相当复杂。数据库管理系统提供了事务的隔离级别,让用户以一种更轻松的方式处理并发一致性问题。**
-# 封锁类型
+# 三、封锁
-排它锁 (X 锁),共享锁 (S 锁)
+## 封锁粒度
-一个事务 T 对数据对象 A 加了 X 锁,T 就可以对 A 进行读取和更新。加锁期间其它事务不能对数据对象 A 加任何其它锁;
+
-一个事务 T 对数据对象加了 S 锁,T 可以对 A 进行读取操作,但是不能进行更新操作。加锁期间其它事务能对数据对象 A 加 S 锁,但是不能加 X 锁。
+应该尽量只锁定需要修改的那部分数据,而不是所有的资源。锁定的数据量越少,发生锁争用的可能就越小,系统的并发程度就越高。
-# 封锁粒度
+但是加锁需要消耗资源,锁的各种操作,包括获取锁,检查锁是否已经解除、释放锁,都会增加系统开销。因此封锁粒度越小,系统开销就越大。需要在锁开销以及数据安全性之间做一个权衡。
-粒度可以是整个数据库,也可以是表,行,或者分量。
+MySQL 中提供了两种封锁粒度:行级锁以及表级锁。
-粒度越小,开销越大。
+## 封锁类型
-# 封锁协议
+### 1. 排它锁与共享锁
-## 三级封锁协议
+- 排它锁(Exclusive),简写为 X 锁,又称写锁。
+- 共享锁(Shared),简写为 S 锁,又称读锁。
-
+有以下两个规定:
-**1 级封锁协议**
+1. 一个事务对数据对象 A 加了 X 锁,就可以对 A 进行读取和更新。加锁期间其它事务不能对 A 加任何锁。
+2. 一个事务对数据对象 A 加了 S 锁,可以对 A 进行读取操作,但是不能进行更新操作。加锁期间其它事务能对 A 加 S 锁,但是不能加 X 锁。
-事务 T 要修改数据 A 时必须加 X 锁,直到事务结束才释放锁。
+锁的兼容关系如下:
-可以解决丢失修改问题;
+| - | X | S |
+| :--: | :--: | :--: |
+|X|No|No|
+|S|No|Yes|
-**2 级封锁协议**
+### 2. 意向锁
-在 1 级的基础上,要求读取数据 A 时必须加 S 锁,读取完马上释放 S 锁。
+意向锁(Intention Locks)可以支持多粒度封锁。它本身是一个表锁,通过在原来的 X/S 锁之上引入了 IX/IS,用来表示一个事务想要在某个数据行上加 X 锁或 S 锁。
+
+有以下两个规定:
+
+1. 一个事务在获得某个数据行对象的 S 锁之前,必须先获得 IS 锁或者更强的锁;
+2. 一个事务在获得某个数据行对象的 X 锁之前,必须先获得 IX 锁。
+
+各种锁的兼容关系如下:
+
+| - | X | IX | S | IS |
+| :--: | :--: | :--: | :--: | :--: |
+|X |No |No |No | No|
+|IX |No |Yes|No | Yes|
+|S |No |No |Yes| Yes|
+|IS |No |Yes|Yes| Yes|
+
+## 封锁协议
+
+### 1. 三级封锁协议
+
+**一级封锁协议**
+
+事务 T 要修改数据 A 时必须加 X 锁,直到 T 结束才释放锁。
+
+可以解决丢失修改问题,因为不能同时有两个事务对同一个数据进行修改,那么一个事务的修改就不会被覆盖。
+
+| T1 | T1 |
+| :--: | :--: |
+| lock-x(A) | |
+| read A=20 | |
+| | lock-x(A) |
+| | wait |
+| write A=19 | |
+| commit | |
+| unlock-x(A) | |
+| | obtain |
+| | read A=19 |
+| | write A=21 |
+| | commit |
+| | unlock-x(A)|
+
+**二级封锁协议**
+
+在一级的基础上,要求读取数据 A 时必须加 S 锁,读取完马上释放 S 锁。
可以解决读脏数据问题,因为如果一个事务在对数据 A 进行修改,根据 1 级封锁协议,会加 X 锁,那么就不能再加 S 锁了,也就是不会读入数据。
-**3 级封锁协议**
+| T1 | T1 |
+| :--: | :--: |
+| lock-x(A) | |
+| read A=20 | |
+| write A=19 | |
+| | lock-s(A) |
+| | wait |
+| rollback | |
+| A=20 | |
+| unlock-x(A) | |
+| | obtain |
+| | read A=20 |
+| | commit |
+| | unlock-s(A)|
-在 2 级的基础上,要求读取数据 A 时必须加 S 锁,直到事务结束了才能释放 S 锁。
+**三级封锁协议**
+
+在二级的基础上,要求读取数据 A 时必须加 S 锁,直到事务结束了才能释放 S 锁。
可以解决不可重复读的问题,因为读 A 时,其它事务不能对 A 加 X 锁,从而避免了在读的期间数据发生改变。
-## 两段锁协议
+| T1 | T1 |
+| :--: | :--: |
+| lock-s(A) | |
+| read A=20 | |
+| |lock-x(A) |
+| | wait |
+| read A=20| |
+| commit | |
+| unlock-s(A) | |
+| | obtain |
+| | read A=20 |
+| | write A=19|
+| | commit |
+| | unlock-X(A)|
-加锁和解锁分为两个阶段进行。两段锁是并行事务可串行化的充分条件,但不是必要条件。
+### 2. 两段锁协议
+
+加锁和解锁分为两个阶段进行,事务 T 对数据 A 进行读或者写操作之前,必须先获得对 A 的封锁,并且在释放一个封锁之后,T 不能再获得任何的其它锁。
+
+事务遵循两段锁协议是保证并发操作可串行化调度的充分条件。例如以下操作满足两段锁协议,它是可串行化调度。
```html
lock-x(A)...lock-s(B)...lock-s(c)...unlock(A)...unlock(C)...unlock(B)
```
-# 乐观锁和悲观锁
+但不是必要条件,例如以下操作不满足两段锁协议,但是它还是可串行化调度。
-## 悲观锁
+```html
+lock-x(A)...unlock(A)...lock-s(B)...unlock(B)...lock-s(c)...unlock(C)...
+```
-假定会发生并发冲突,屏蔽一切可能违反数据完整性的操作。
+# 四、隔离级别
-Java synchronized 就属于悲观锁的一种实现,每次线程要修改数据时都先获得锁,保证同一时刻只有一个线程能操作数据,其他线程则会被阻塞。
+ **1. 未提交读(READ UNCOMMITTED)**
-## 乐观锁
+事务中的修改,即使没有提交,对其它事务也是可见的。
-假设不会发生并发冲突,只在提交操作时检查是否违反数据完整性。
+ **2. 提交读(READ COMMITTED)**
-Java JUC 中的 Atomic 包就是乐观锁的一种实现,AtomicInteger 通过 CAS(Compare And Set)操作实现线程安全的自增操作。
+一个事务只能读取已经提交的事务所做的修改。换句话说,一个事务所做的修改在提交之前对其它事务是不可见的。
-乐观锁有两种实现方式,数据版本和时间戳。它们都需要在数据库表中增加一个字段,使用这个字段来判断数据是否过期。例如,数据版本实现方式中,需要在数据库表中增加一个数字类型的 version 字段,当读取数据时,将 version 字段的值一同读出。随后数据每更新一次,对此 version 值加 1。当提交更新的时候,判断读出的 version 和数据库表中的 version 是否一致,如果一致,则予以更新;否则认为是过期数据。
+ **3. 可重复读(REPEATABLE READ)**
-## MySQL 隐式和显示锁定
+保证在同一个事务中多次读取同样数据的结果是一样的。
-MySQL InnoDB 采用的是两阶段锁协议。在事务执行过程中,随时都可以执行锁定,锁只有在执行 COMMIT 或者 ROLLBACK 的时候才会释放,并且所有的锁是在同一时刻被释放。前面描述的锁定都是隐式锁定,InnoDB 会根据事务隔离级别在需要的时候自动加锁。
+ **4. 可串行化(SERIALIXABLE)**
-另外,InnoDB 也支持通过特定的语句进行显示锁定,这些语句不属于 SQL 规范:
+强制事务串行执行。
-- SELECT ... LOCK IN SHARE MODE
-- SELECT ... FOR UPDATE
+ **四个隔离级别的对比**
-# 范式
+| 隔离级别 | 脏读 | 不可重复读 | 幻影读 |
+| :---: | :---: | :---:| :---: |
+| 未提交读 | YES | YES | YES |
+| 提交读 | NO | YES | YES |
+| 可重复读 | NO | NO | YES |
+| 可串行化 | NO | NO | NO |
-记 A->B 表示 A 函数决定于 B,也可以说 B 函数依赖于 A。
+# 五、多版本并发控制
+
+(Multi-Version Concurrency Control, MVCC)是 MySQL 的 InnoDB 存储引擎实现隔离级别的一种具体方式,用于实现提交读和可重复读这两种隔离级别。而未提交读隔离级别总是读取最新的数据行,无需使用 MVCC;可串行化隔离级别需要对所有读取的行都加锁,单纯使用 MVCC 无法实现。
+
+## 版本号
+
+- 系统版本号:是一个递增的数字,每开始一个新的事务,系统版本号就会自动递增。
+- 事务版本号:事务开始时的系统版本号。
+
+InooDB 的 MVCC 在每行记录后面都保存着两个隐藏的列,用来存储两个版本号:
+
+- 创建版本号:指示创建一个数据行的快照时的系统版本号;
+- 删除版本号:如果该快照的删除版本号大于当前事务版本号表示该快照有效,否则表示该快照已经被删除了。
+
+## Undo 日志
+
+InnoDB 的 MVCC 使用到的快照存储在 Undo 日志中,该日志通过回滚指针把一个数据行(Record)的所有快照连接起来。
+
+
+
+## 实现过程
+
+以下过程针对可重复读隔离级别。
+
+### 1. SELECT
+
+该操作必须保证多个事务读取到同一个数据行的快照,这个快照是最近的一个有效快照。但是也有例外,如果有一个事务正在修改该数据行,那么它可以读取事务本身所做的修改,而不用和其它事务的读取结果一致。
+
+当开始新一个事务时,该事务的版本号肯定会大于当前所有数据行快照的创建版本号,理解这一点很关键。
+
+把没对一个数据行做修改的事务称为 T,T 所要读取的数据行快照的创建版本号必须小于 T 的版本号,因为如果大于或者等于 T 的版本号,那么表示该数据行快照是其它事务的最新修改,因此不能去读取它。
+
+除了上面的要求,T 所要读取的数据行快照的删除版本号必须大于 T 的版本号,因为如果小于等于 T 的版本号,那么表示该数据行快照是已经被删除的,不应该去读取它。
+
+### 2. INSERT
+
+将系统版本号作为数据行快照的创建版本号。
+
+### 3. DELETE
+
+将系统版本号作为数据行快照的删除版本号。
+
+### 4. UPDATE
+
+将系统版本号作为更新后的数据行快照的创建版本号,同时将系统版本号作为更新前的数据行快照的删除版本号。可以理解为先执行 DELETE 后执行 INSERT。
+
+## 快照读与当前读
+
+### 1. 快照读
+
+读取快照中的数据。
+
+引入快照读的目的主要是为了免去加锁操作带来的性能开销,但是当前读需要加锁。
+
+```sql
+select * from table ....;
+```
+
+### 2. 当前读
+
+读取最新的数据。
+
+需要加锁,以下第一个语句加 S 锁,其它都加 X 锁。
+
+```sql
+select * from table where ? lock in share mode;
+select * from table where ? for update;
+insert;
+update ;
+delete;
+```
+
+# 六、Next-Key Locks
+
+Next-Key Locks 也是 MySQL 的 InnoDB 存储引擎的一种锁实现。MVCC 不能解决幻读的问题,Next-Key Locks 就是为了解决这个问题而存在的。在可重复读隔离级别下,MVCC + Next-Key Locks,就可以防止幻读的出现。
+
+## Record Locks
+
+锁定的对象是索引,而不是数据。如果表没有设置索引,InnoDB 会自动在主键上创建隐藏的聚集索引,因此 Record Locks 依然可以使用。
+
+## Grap Locks
+
+锁定一个范围内的索引,例如当一个事务执行以下语句,其它事务就不能在 t.c 中插入 15。
+
+```sql
+SELECT c FROM t WHERE c BETWEEN 10 and 20 FOR UPDATE;
+```
+
+## Next-Key Locks
+
+它是 Record Locks 和 Gap Locks 的结合。在 user 中有以下记录:
+
+```sql
+| id | last_name | first_name | age |
+|------|-------------|--------------|-------|
+| 4 | stark | tony | 21 |
+| 1 | tom | hiddleston | 30 |
+| 3 | morgan | freeman | 40 |
+| 5 | jeff | dean | 50 |
+| 2 | donald | trump | 80 |
++------|-------------|--------------|-------+
+```
+
+那么就需要锁定以下范围:
+
+```sql
+(-∞, 21]
+(21, 30]
+(30, 40]
+(40, 50]
+(50, 80]
+(80, ∞)
+```
+
+# 七、关系数据库设计理论
+
+## 函数依赖
+
+记 A->B 表示 A 函数决定 B,也可以说 B 函数依赖于 A。
如果 {A1,A2,... ,An} 是关系的一个或多个属性的集合,该集合决定了关系的其它所有属性并且是最小的,那么该集合就称为键码。
-对于函数依赖 W->A,如果能找到 W 的真子集使得 A 依赖于这个真子集,那么就是部分依赖,否则就是完全依赖;
+对于 W->A,如果能找到 W 的真子集 W',使得 W'-> A,那么 W->A 就是部分函数依赖,否则就是完全函数依赖;
-以下关系中,Sno 表示学号,Sname 表示学生姓名,Sdept 表示学院,Cname 表示课程名,Mname 表示院长姓名。函数依赖为 (Sno, Cname) -> (Sname, Sdept, Mname)。注:实际开发过程中,不会出现这种表,而是每个实体都放在单独一张表中,然后实体之间的联系表用实体 id 来表示。
+## 异常
-
+以下的学生课程关系的函数依赖为 Sno, Cname -> Sname, Sdept, Mname, Grade,键码为 {Sno, Cname}。也就是说,确定学生和课程之后,就能确定其它信息。
-不符合范式的关系,会产生很多异常。主要有以下四种异常:
+| Sno | Sname | Sdept | Mname | Cname | Grade |
+| :---: | :---: | :---: | :---: | :---: |:---:|
+| 1 | 学生-1 | 学院-1 | 院长-1 | 课程-1 | 90 |
+| 2 | 学生-2 | 学院-2 | 院长-2 | 课程-2 | 80 |
+| 2 | 学生-2 | 学院-2 | 院长-2 | 课程-1 | 100 |
-1. 冗余数据
-2. 修改异常
-3. 删除异常
-4. 插入异常,比如如果新插入一个学生的信息,而这个学生还没选课,那么就无法插入该学生。
+不符合范式的关系,会产生很多异常,主要有以下四种异常:
-关系数据库的范式理论就是是为了解决这四种异常。
+1. 冗余数据,例如学生-2 出现了两次。
+2. 修改异常,修改了一个记录中的信息,但是另一个记录中相同的信息却没有被修改。
+3. 删除异常,删除一个信息,那么也会丢失其它信息。例如如果删除了课程-1,需要删除第一行和第三行,那么学生-1 的信息就会丢失。
+4. 插入异常,例如想要插入一个学生的信息,如果这个学生还没选课,那么就无法插入。
-高级别范式的依赖基于低级别的范式。
+## 范式
-## 第一范式 (1NF)
+范式理论是为了解决以上提到四种异常。高级别范式的依赖于低级别的范式。
-属性不可分。
+
-## 第二范式 (2NF)
+### 1. 第一范式 (1NF)
+
+属性不可分;
+
+### 2. 第二范式 (2NF)
每个非主属性完全函数依赖于键码。
可以通过分解来满足。
-**分解前**
+ **分解前**
-
+| Sno | Sname | Sdept | Mname | Cname | Grade |
+| :---: | :---: | :---: | :---: | :---: |:---:|
+| 1 | 学生-1 | 学院-1 | 院长-1 | 课程-1 | 90 |
+| 2 | 学生-2 | 学院-2 | 院长-2 | 课程-2 | 80 |
+| 2 | 学生-2 | 学院-2 | 院长-2 | 课程-1 | 100 |
-
+以上学生课程关系中,{Sno, Cname} 为键码,有如下函数依赖:
-**分解后**
+- Sno, Cname -> Sname, Sdept, Mname
+- Sno -> Sname, Sdept
+- Sdept -> Mname
+- Sno -> Mname
+- Sno, Cname-> Grade
-
+Grade 完全函数依赖于键码,它没有任何冗余数据,每个学生的每门课都有特定的成绩。
-
+Sname, Sdept 和 Mname 都函数依赖于 Sno,而部分依赖于键码。当一个学生选修了多门课时,这些数据就会出现多次,造成大量冗余数据。
-
+ **分解后**
-
+关系-1
-
+| Sno | Sname | Sdept | Mname |
+| :---: | :---: | :---: | :---: |
+| 1 | 学生-1 | 学院-1 | 院长-1 |
+| 2 | 学生-2 | 学院-2 | 院长-2 |
-
+有以下函数依赖:
-
+- Sno -> Sname, Sdept, Mname
+- Sdept -> Mname
-## 第三范式 (3NF)
+关系-2
+
+| Sno | Cname | Grade |
+| :---: | :---: |:---:|
+| 1 | 课程-1 | 90 |
+| 2 | 课程-2 | 80 |
+| 2 | 课程-1 | 100 |
+
+有以下函数依赖:
+
+- Sno, Cname -> Grade
+
+### 3. 第三范式 (3NF)
非主属性不传递依赖于键码。
-上述 S1 存在传递依赖,Mname 依赖于 Sdept,而 Sdept 又依赖于 Sno,可以继续分解。
+上面的关系-1 中存在以下传递依赖:Sno -> Sdept -> Mname,可以进行以下分解:
-
+关系-11
-## BC 范式(BCNF)
+| Sno | Sname | Sdept |
+| :---: | :---: | :---: |
+| 1 | 学生-1 | 学院-1 |
+| 2 | 学生-2 | 学院-2 |
+
+关系-12
+
+| Sdept | Mname |
+| :---: | :---: |
+| 学院-1 | 院长-1 |
+| 学院-2 | 院长-2 |
+
+### 4. BC 范式(BCNF)
所有属性不传递依赖于键码。
-关系模式 STC(Sname, Tname, Cname, Grade),其中四个属性分别为学生姓名、教师姓名、课程名和成绩。有以下函数依赖:
+关系 STC(Sname, Tname, Cname, Grade) 的四个属性分别为学生姓名、教师姓名、课程名和成绩,它的键码为 (Sname, Cname, Tname),有以下函数依赖:
-
+- Sname, Cname -> Tname
+- Sname, Cname -> Grade
+- Sname, Tname -> Cname
+- Sname, Tname -> Grade
+- Tname -> Cname
-
+存在着以下函数传递依赖:
-
+- Sname -> Tname -> Cname
-
+可以分解成 SC(Sname, Cname, Grade) 和 ST(Sname, Tname),对于 ST,属性之间是多对多关系,无函数依赖。
-
+# 八、数据库系统概述
-分解成 SC(Sname, Cname, Grade) 和 ST(Sname, Tname),对于 ST,属性之间是多对多关系,无函数依赖。
+## 基本术语
-# 约束
+### 1. 数据模型
-## 键码
+由数据结构、数据操作和完整性三个要素组成。
-用于唯一表示一个实体。键码可以由多个属性构成,每个构成键码的属性成为码。
+### 2. 数据库系统
-## 单值约束
+数据库系统包含所有与数据库相关的内容,包括数据库、数据库管理系统、应用程序以及数据库管理员和用户,还包括相关的硬件和软件。
-某个属性的值是唯一的。
+## 数据库的三层模式和两层映像
-## 引用完整性约束
+- 外模式:局部逻辑结构
+- 模式:全局逻辑结构
+- 内模式:物理结构
-一个实体的属性引用的值在另一个实体的某个属性中存在。
+
-## 域约束
+### 1. 外模式
-某个属性的值在特定范围之内。
+又称用户模式,是用户和数据库系统的接口,特定的用户只能访问数据库系统提供给他的外模式中的数据。例如不同的用户创建了不同数据库,那么一个用户只能访问他有权限访问的数据库。
-## 一般约束
+一个数据库可以有多个外模式,一个用户只能有一个外模式,但是一个外模式可以给多个用户使用。
-一般性约束,比如大小约束,数量约束。
+### 2. 模式
-# 数据库的三层模式和两层映像
+可以分为概念模式和逻辑模式,概念模式可以用概念-关系来描述;逻辑模式使用特定的数据模式(比如关系模型)来描述数据的逻辑结构,这种逻辑结构包括数据的组成、数据项的名称、类型、取值范围。不仅如此,逻辑模式还要描述数据之间的关系、数据的完整性与安全性要求。
-外模式:局部逻辑结构;模式:全局逻辑结构;内模式:物理结构。
-
-## 外模式
-
-又称用户模式,是用户和数据库系统的接口,特定的用户只能访问数据库系统提供给他的外模式中的数据。例如不同的用户创建了不同数据库,那么一个用户只能访问他有权限访问的数据库。一个数据库可以有多个外模式,一个用户只能有一个外模式,但是一个外模式可以给多个用户使用。
-
-## 模式
-
-可以分为概念模式和逻辑模式,概念模式可以用概念 - 关系来描述;逻辑模式使用特定的数据模式(比如关系模型)来描述数据的逻辑结构,这种逻辑结构包括数据的组成、数据项的名称、类型、取值范围。不仅如此,逻辑模式还要描述数据之间的关系,数据的完整性与安全性要求。
-
-## 内模式
+### 3. 内模式
又称为存储模式,描述记录的存储方式,例如索引的组织方式、数据是否压缩以及是否加密等等。
-## 外模式/模式映像
+### 4. 外模式/模式映像
把外模式的局部逻辑结构和模式的全局逻辑结构联系起来。该映像可以保证数据和应用程序的逻辑独立性。
-## 模式/内模式映像
+### 5. 模式/内模式映像
把模式的全局逻辑结构和内模式的物理结构联系起来,该映像可以保证数据和应用程序的物理独立性。
-# ER 图
+# 九、关系数据库建模
-Entity-Relationship,包含三个部分:实体、属性、联系。
+## ER 图
-## 实体的三种联系
+Entity-Relationship,有三个组成部分:实体、属性、联系。
-联系包含 1 对 1,1 对多,多对多三种。
+### 1. 实体的三种联系
-如果 A 到 B 是 1 对多关系,那么画个带箭头的线段指向 B;如果是 1 对 1,画两个带箭头的线段;如果是多对多,画两个不带箭头的线段。
+联系包含一对一,一对多,多对多三种。
+
+如果 A 到 B 是一对多关系,那么画个带箭头的线段指向 B;如果是一对一,画两个带箭头的线段;如果是多对多,画两个不带箭头的线段。下图的 Course 和 Student 是一对多的关系。
-## 表示出现多次的关系
+### 2. 表示出现多次的关系
-一个实体在联系出现几次,就要用几条线连接。如下表示一个课程的先修关系,先修关系中,应当出现两个 Course 实体,第一个是先修课程,后一个是后修课程,因此需要用两条线来表示这种关系。
+一个实体在联系出现几次,就要用几条线连接。下图表示一个课程的先修关系,先修关系出现两个 Course 实体,第一个是先修课程,后一个是后修课程,因此需要用两条线来表示这种关系。
-## 联系的多向性
+### 3. 联系的多向性
-下图中一个联系表示三个实体的关系。虽然老师可以开设多门课,并且可以教授多名学生,但是对于特定的学生和课程,只有一个老师教授,这就构成了一个三元联系。
+虽然老师可以开设多门课,并且可以教授多名学生,但是对于特定的学生和课程,只有一个老师教授,这就构成了一个三元联系。
@@ -332,19 +576,47 @@ Entity-Relationship,包含三个部分:实体、属性、联系。
-## 表示子类
+### 4. 表示子类
-用 is-a 联系来表示子类,具体做法是用一个三角形和两条线来连接类和子类。与子类有关的属性和联系都连到子类上,而与父类和子类都有关的连到父类上。
+用一个三角形和两条线来连接类和子类,与子类有关的属性和联系都连到子类上,而与父类和子类都有关的连到父类上。
-# 一些概念
+# 十、约束
-**数据模型** 由数据结构、数据操作和完整性三个要素组成。
+## 1. 键码
-**数据库系统** 包括了数据库,数据库管理系统,应用程序以及数据库管理员和用户,还包括相关的硬件和软件。也就是说数据库系统包含所有与数据库相关的内容。
+用于唯一表示一个实体。
+
+键码可以由多个属性构成,每个构成键码的属性称为码。
+
+## 2. 单值约束
+
+某个属性的值是唯一的。
+
+## 3. 引用完整性约束
+
+一个实体的属性引用的值在另一个实体的某个属性中存在。
+
+## 4. 域约束
+
+某个属性的值在特定范围之内。
+
+## 5. 一般约束
+
+比如大小约束,数量约束。
# 参考资料
- 史嘉权. 数据库系统概论[M]. 清华大学出版社有限公司, 2006.
-- [MySQL 乐观锁与悲观锁 ](https://www.jianshu.com/p/f5ff017db62a)
+- 施瓦茨. 高性能 MYSQL(第3版)[M]. 电子工业出版社, 2013.
+- [The InnoDB Storage Engine](https://dev.mysql.com/doc/refman/5.7/en/innodb-storage-engine.html)
+- [Transaction isolation levels](https://www.slideshare.net/ErnestoHernandezRodriguez/transaction-isolation-levels)
+- [Concurrency Control](http://scanftree.com/dbms/2-phase-locking-protocol)
+- [The Nightmare of Locking, Blocking and Isolation Levels!](https://www.slideshare.net/brshristov/the-nightmare-of-locking-blocking-and-isolation-levels-46391666)
+- [三级模式与两级映像](http://blog.csdn.net/d2457638978/article/details/48783923)
+- [Database Normalization and Normal Forms with an Example](https://aksakalli.github.io/2012/03/12/database-normalization-and-normal-forms-with-an-example.html)
+- [The basics of the InnoDB undo logging and history system](https://blog.jcole.us/2014/04/16/the-basics-of-the-innodb-undo-logging-and-history-system/)
+- [MySQL locking for the busy web developer](https://www.brightbox.com/blog/2013/10/31/on-mysql-locks/)
+- [浅入浅出 MySQL 和 InnoDB](https://draveness.me/mysql-innodb)
+- [fd945daf-4a6c-4f20-b9c2-5390f5955ce5.jpg](https://tech.meituan.com/innodb-lock.html)
diff --git a/notes/正则表达式.md b/notes/正则表达式.md
new file mode 100644
index 00000000..bd38a392
--- /dev/null
+++ b/notes/正则表达式.md
@@ -0,0 +1,387 @@
+
+* [一、概述](#一概述)
+* [二、匹配单个字符](#二匹配单个字符)
+* [三、匹配一组字符](#三匹配一组字符)
+* [四、使用元字符](#四使用元字符)
+* [五、重复匹配](#五重复匹配)
+* [六、位置匹配](#六位置匹配)
+* [七、使用子表达式](#七使用子表达式)
+* [八、回溯引用](#八回溯引用)
+* [九、前后查找](#九前后查找)
+* [十、嵌入条件](#十嵌入条件)
+* [参考资料](#参考资料)
+
+
+
+# 一、概述
+
+正则表达式用于文本内容的查找和替换。
+
+正则表达式内置于其它语言或者软件产品中,它本身不是一种语言或者软件。
+
+[正则表达式在线工具](http://tool.oschina.net/regex/)
+
+# 二、匹配单个字符
+
+正则表达式一般是区分大小写的,但是也有些实现是不区分。
+
+**.** 可以用来匹配任何的单个字符,但是在绝大多数实现里面,不能匹配换行符;
+
+**\\** 是元字符,表示它有特殊的含义,而不是字符本身的含义。如果需要匹配 . ,那么要用 \ 进行转义,即在 . 前面加上 \ 。
+
+**正则表达式**
+
+```
+nam.
+```
+
+**匹配结果**
+
+My **name** is Zheng.
+
+# 三、匹配一组字符
+
+**[ ]** 定义一个字符集合;
+
+0-9、a-z 定义了一个字符区间,区间使用 ASCII 码来确定,字符区间只能用在 [ ] 之间。
+
+**-** 元字符只有在 [ ] 之间才是元字符,在 [ ] 之外就是一个普通字符;
+
+**^** 是取非操作,必须在 [ ] 字符集合中使用;
+
+**应用**
+
+匹配以 abc 为开头,并且最后一个字母不为数字的字符串:
+
+**正则表达式**
+
+```
+abc[^0-9]
+```
+
+**匹配结果**
+
+1. **abcd**
+2. abc1
+3. abc2
+
+# 四、使用元字符
+
+## 匹配空白字符
+
+| 元字符 | 说明 |
+| :---: | :---: |
+| [\b] | 回退(删除)一个字符 |
+| \f | 换页符 |
+| \n | 换行符 |
+| \r | 回车符 |
+| \t | 制表符 |
+| \v | 垂直制表符 |
+
+\r\n 是 Windows 中的文本行结束标签,在 Unix/Linux 则是 \n ;\r\n\r\n 可以匹配 Windows 下的空白行,因为它将匹配两个连续的行尾标签,而这正是两条记录之间的空白行;
+
+. 是元字符,前提是没有对它们进行转义;f 和 n 也是元字符,但是前提是对它们进行了转义。
+
+## 匹配特定的字符类别
+
+### 1. 数字元字符
+
+| 元字符 | 说明 |
+| :---: | :---: |
+| \d | 数字字符,等价于 [0-9] |
+| \D | 非数字字符,等价于 [^0-9] |
+
+### 2. 字母数字元字符
+
+| 元字符 | 说明 |
+| :---: | :---: |
+| \w | 大小写字母,下划线和数字,等价于 [a-zA-Z0-9\_] |
+| \W | 对 \w 取非 |
+
+### 3. 空白字符元字符
+
+| 元字符 | 说明 |
+| :---: | :---: |
+| \s | 任何一个空白字符,等价于 [\f\n\r\t\v] |
+| \S | 对 \s 取非 |
+
+\x 匹配十六进制字符,\0 匹配八进制,例如 \x0A 对应 ASCII 字符 10 ,等价于 \n,也就是它会匹配 \n 。
+
+# 五、重复匹配
+
+**\+** 匹配 1 个或者多个字符, **\*** 匹配 0 个或者多个,**?** 匹配 0 个或者 1 个。
+
+**应用**
+
+匹配邮箱地址。
+
+**正则表达式**
+
+```
+[\w.]+@\w+\.\w+
+```
+
+[\w.] 匹配的是字母数字或者 . ,在其后面加上 + ,表示匹配多次。在字符集合 [ ] 里,. 不是元字符;
+
+**匹配结果**
+
+**abc.def@ qq.com**
+
+为了可读性,常常把转义的字符放到字符集合 [ ] 中,但是含义是相同的。
+
+```
+[\w.]+@\w+\.\w+
+[\w.]+@[\w]+\.[\w]+
+```
+
+**{n}** 匹配 n 个字符,**{m, n}** 匹配 m\~n 个字符,**{m,}** 至少匹配 m 个字符;
+
+\* 和 + 都是贪婪型元字符,会匹配最多的内容,在元字符后面加 ? 可以转换为懒惰型元字符,例如 \*?、+? 和 {m, n}? 。
+
+**正则表达式**
+
+```
+a.+c
+```
+
+由于 + 是贪婪型的,因此 .+ 会匹配更可能多的内容,所以会把整个 abcabcabc 文本都匹配,而不是只匹配前面的 abc 文本。用懒惰型可以实现匹配前面的。
+
+**匹配结果**
+
+**abcabcabc**
+
+# 六、位置匹配
+
+## 单词边界
+
+**\b** 可以匹配一个单词的边界,边界是指位于 \w 和 \W 之间的位置;**\B** 匹配一个不是单词边界的位置。
+
+\b 只匹配位置,不匹配字符,因此 \babc\b 匹配出来的结果为 3 个字符。
+
+## 字符串边界
+
+**^** 匹配整个字符串的开头,**$** 匹配结尾。
+
+^ 元字符在字符集合中用作求非,在字符集合外用作匹配字符串的开头。
+
+使用 (?m) 来打开分行匹配模式,在该模式下,换行被当做字符串的边界。
+
+**应用**
+
+匹配代码中以 // 开始的注释行
+
+**正则表达式**
+
+```
+(?m)^\s*//.*$
+```
+
+如果没用 (?m),则只会匹配 // 注释 1 以及之后的所有内容。用了分行匹配模式之后,换行符被当成是字符串分隔符,因此能正确匹配出两个注释内容。
+
+**匹配结果**
+
+1. public void fun() {
+2. **// 注释 1**
+3. int a = 1;
+4. int b = 2;
+5. **// 注释 2**
+6. int c = a + b;
+7. }
+
+# 七、使用子表达式
+
+使用 **( )** 定义一个子表达式。子表达式的内容可以当成一个独立元素,即可以将它看成一个字符,并且使用 * 等元字符。
+
+子表达式可以嵌套,但是嵌套层次过深会变得很难理解。
+
+**正则表达式**
+
+```
+(ab) {2,}
+```
+
+**匹配结果**
+
+**ababab**
+
+**|** 是或元字符,它把左边和右边所有的部分都看成单独的两个部分,两个部分只要有一个匹配就行。
+
+**正则表达式**
+
+```
+(19|20)\d{2}
+```
+
+**匹配结果**
+
+1. **1900**
+2. **2010**
+3. 1020
+
+**应用**
+
+匹配 IP 地址。IP 地址中每部分都是 0-255 的数字,用正则表达式匹配时以下情况是合法的:
+
+1. 一位或者两位的数字
+2. 1 开头的三位数
+3. 2 开头,第 2 位是 0-4 的三位数
+4. 25 开头,第 3 位是 0-5 的三位数
+
+**正则表达式**
+
+```
+(((\d{1,2})|(1\d{2})|(2[0-4]\d)|(25[0-5]))\.) {3}(((\d{1,2})|(1\d{2})|(2[0-4]\d)|(25[0-5])))
+```
+
+**匹配结果**
+
+1. **192.168.0.1**
+2. 555.555.555.555
+
+# 八、回溯引用
+
+回溯引用使用 **\n** 来引用某个子表达式,其中 n 代表的是子表达式的序号,从 1 开始。它和子表达式匹配的内容一致,比如子表达式匹配到 abc,那么回溯引用部分也需要匹配 abc 。
+
+**应用**
+
+匹配 HTML 中合法的标题元素。
+
+**正则表达式**
+
+\1 将回溯引用子表达式 (h[1-6]) 匹配的内容,也就是说必须和子表达式匹配的内容一致。
+
+```
+<(h[1-6])>\w*?\1>
+```
+
+**匹配结果**
+
+1. **<h1>x</h1>**
+2. **<h2>x</h2>**
+3. <h3>x</h1>
+
+## 替换
+
+需要用到两个正则表达式。
+
+**应用**
+
+修改电话号码格式。
+
+**文本**
+
+313-555-1234
+
+**查找正则表达式**
+
+```
+(\d{3})(-)(\d{3})(-)(\d{4})
+```
+
+**替换正则表达式**
+
+在第一个子表达式查找的结果加上 () ,然后加一个空格,在第三个和第五个字表达式查找的结果中间加上 - 进行分隔。
+
+```
+($1) $3-$5
+```
+
+**结果**
+
+(313) 555-1234
+
+## 大小写转换
+
+| 元字符 | 说明 |
+| :---: | :---: |
+| \l | 把下个字符转换为小写 |
+| \u| 把下个字符转换为大写 |
+| \L | 把\L 和\E 之间的字符全部转换为小写 |
+| \U | 把\U 和\E 之间的字符全部转换为大写 |
+| \E | 结束\L 或者\U |
+
+**应用**
+
+把文本的第二个和第三个字符转换为大写。
+
+**文本**
+
+abcd
+
+**查找**
+
+```
+(\w)(\w{2})(\w)
+```
+
+**替换**
+
+```
+$1\U$2\E$3
+```
+
+**结果**
+
+aBCd
+
+# 九、前后查找
+
+前后查找规定了匹配的内容首尾应该匹配的内容,但是又不包含首尾匹配的内容。向前查找用 **?=** 来定义,它规定了尾部匹配的内容,这个匹配的内容在 ?= 之后定义。所谓向前查找,就是规定了一个匹配的内容,然后以这个内容为尾部向前面查找需要匹配的内容。向后匹配用 ?<= 定义。
+
+**应用**
+
+查找出邮件地址 @ 字符前面的部分。
+
+**正则表达式**
+
+```
+\w+(?=@)
+```
+
+**结果**
+
+**abc** @qq.com
+
+对向前和向后查找取非,只要把 = 替换成 ! 即可,比如 (?=) 替换成 (?!) 。取非操作使得匹配那些首尾不符合要求的内容。
+
+# 十、嵌入条件
+
+## 回溯引用条件
+
+条件判断为某个子表达式是否匹配,如果匹配则需要继续匹配条件表达式后面的内容。
+
+**正则表达式**
+
+子表达式 (\\() 匹配一个左括号,其后的 ? 表示匹配 0 个或者 1 个。 ?(1) 为条件,当子表达式 1 匹配时条件成立,需要执行 \) 匹配,也就是匹配右括号。
+
+```
+(\()?abc(?(1)\))
+```
+
+**结果**
+
+1. **(abc)**
+2. **abc**
+3. (abc
+
+## 前后查找条件
+
+条件为定义的首尾是否匹配,如果匹配,则继续执行后面的匹配。注意,首尾不包含在匹配的内容中。
+
+**正则表达式**
+
+ ?(?=-) 为前向查找条件,只有在以 - 为前向查找的结尾能匹配 \d{5} ,才继续匹配 -\d{4} 。
+
+```
+\d{5}(?(?=-)-\d{4})
+```
+
+**结果**
+
+1. **11111**
+2. 22222-
+3. **33333-4444**
+
+# 参考资料
+
+- BenForta. 正则表达式必知必会 [M]. 人民邮电出版社, 2007.
diff --git a/notes/算法.md b/notes/算法.md
index f6fcf43e..997dc245 100644
--- a/notes/算法.md
+++ b/notes/算法.md
@@ -1,128 +1,70 @@
-* [算法分析](#算法分析)
- * [1. 函数转换](#1-函数转换)
- * [2. 数学模型](#2-数学模型)
- * [3. ThreeSum](#3-threesum)
- * [4. 倍率实验](#4-倍率实验)
- * [5. 注意事项](#5-注意事项)
-* [排序](#排序)
- * [1. 初级排序算法](#1-初级排序算法)
- * [1.1 约定](#11-约定)
- * [1.2 选择排序](#12-选择排序)
- * [1.3 插入排序](#13-插入排序)
- * [1.4 选择排序和插入排序的比较](#14-选择排序和插入排序的比较)
- * [1.5 希尔排序](#15-希尔排序)
- * [2 归并排序](#2-归并排序)
- * [2.1 归并方法](#21-归并方法)
- * [2.2 自顶向下归并排序](#22-自顶向下归并排序)
- * [2.3 自底向上归并排序](#23-自底向上归并排序)
- * [3. 快速排序](#3-快速排序)
- * [3.1 基本算法](#31-基本算法)
- * [3.2 切分](#32-切分)
- * [3.3 性能分析](#33-性能分析)
- * [3.4 算法改进](#34-算法改进)
- * [3.4.1 切换到插入排序](#341-切换到插入排序)
- * [3.4.2 三取样](#342-三取样)
- * [3.4.3 三向切分](#343-三向切分)
- * [4. 优先队列](#4-优先队列)
- * [4.1 堆](#41-堆)
- * [4.2 上浮和下沉](#42-上浮和下沉)
- * [4.3 插入元素](#43-插入元素)
- * [4.4 删除最大元素](#44-删除最大元素)
- * [4.5 堆排序](#45-堆排序)
- * [4.6 分析](#46-分析)
- * [5. 应用](#5-应用)
- * [5.1 排序算法的比较](#51-排序算法的比较)
- * [5.2 Java 的排序算法实现](#52-java-的排序算法实现)
- * [5.3 基于切分的快速选择算法](#53-基于切分的快速选择算法)
-* [查找](#查找)
- * [1. 符号表](#1-符号表)
- * [1.1 无序符号表](#11-无序符号表)
- * [1.2 有序符号表](#12-有序符号表)
- * [1.3 二分查找实现有序符号表](#13-二分查找实现有序符号表)
- * [2. 二叉查找树](#2-二叉查找树)
- * [2.1 get()](#21-get)
- * [2.2 put()](#22-put)
- * [2.3 分析](#23-分析)
- * [2.4 floor()](#24-floor)
- * [2.5 rank()](#25-rank)
- * [2.6 min()](#26-min)
- * [2.7 deleteMin()](#27-deletemin)
- * [2.8 delete()](#28-delete)
- * [2.9 keys()](#29-keys)
- * [2.10 性能分析](#210-性能分析)
- * [3. 平衡查找树](#3-平衡查找树)
- * [3.1 2-3 查找树](#31-2-3-查找树)
- * [3.1.1 插入操作](#311-插入操作)
- * [3.1.2 性质](#312-性质)
- * [3.2 红黑二叉查找树](#32-红黑二叉查找树)
- * [3.2.1 左旋转](#321-左旋转)
- * [3.2.2 右旋转](#322-右旋转)
- * [3.2.3 颜色转换](#323-颜色转换)
- * [3.2.4 插入](#324-插入)
- * [3.2.5 删除最小键](#325-删除最小键)
- * [3.2.6 分析](#326-分析)
- * [4. 散列表](#4-散列表)
- * [4.1 散列函数](#41-散列函数)
- * [4.2 基于拉链法的散列表](#42-基于拉链法的散列表)
- * [4.3 基于线性探测法的散列表](#43-基于线性探测法的散列表)
- * [4.3.1 查找](#431-查找)
- * [4.3.2 插入](#432-插入)
- * [4.3.3 删除](#433-删除)
- * [4.3.4 调整数组大小](#434-调整数组大小)
- * [5. 应用](#5-应用)
- * [5.1 各种符号表实现的比较](#51-各种符号表实现的比较)
- * [5.2 Java 的符号表实现](#52-java-的符号表实现)
- * [5.3 集合类型](#53-集合类型)
- * [5.4 稀疏向量乘法](#54-稀疏向量乘法)
+* [一、算法分析](#一算法分析)
+ * [数学模型](#数学模型)
+ * [ThreeSum](#threesum)
+ * [倍率实验](#倍率实验)
+ * [注意事项](#注意事项)
+* [二、栈和队列](#二栈和队列)
+ * [栈](#栈)
+ * [队列](#队列)
+* [三、union-find](#三union-find)
+ * [quick-find](#quick-find)
+ * [quick-union](#quick-union)
+ * [加权 quick-union](#加权-quick-union)
+ * [路径压缩的加权 quick-union](#路径压缩的加权-quick-union)
+ * [各种 union-find 算法的比较](#各种-union-find-算法的比较)
+* [四、排序](#四排序)
+ * [选择排序](#选择排序)
+ * [插入排序](#插入排序)
+ * [希尔排序](#希尔排序)
+ * [归并排序](#归并排序)
+ * [快速排序](#快速排序)
+ * [优先队列](#优先队列)
+ * [应用](#应用)
+* [五、查找](#五查找)
+ * [二分查找实现有序符号表](#二分查找实现有序符号表)
+ * [二叉查找树](#二叉查找树)
+ * [2-3 查找树](#2-3-查找树)
+ * [红黑二叉查找树](#红黑二叉查找树)
+ * [散列表](#散列表)
+ * [应用](#应用)
+* [参考资料](#参考资料)
-# 算法分析
+# 一、算法分析
-## 1. 函数转换
+## 数学模型
-指数函数可以转换为线性函数,从而在函数图像上显示的更直观。
+### 1. 近似
-T(N)=aN3 转换为 lg(T(N))=3lgN+lga
+N3 /6-N2 /2+N/3 \~ N3 /6。使用 \~f(N) 来表示所有随着 N 的增大除以 f(N) 的结果趋近于 1 的函数。
-
+### 2. 增长数量级
-## 2. 数学模型
+N3 /6-N2 /2+N/3 的增长数量级为 O(N3 )。增长数量级将算法与它的实现隔离开来,一个算法的增长数量级为 O(N3 ) 与它是否用 Java 实现,是否运行于特定计算机上无关。
-**近似**
-
-使用 \~f(N) 来表示所有随着 N 的增大除以 f(N) 的结果趋近于 1 的函数 , 例如 N3 /6-N2 /2+N/3 \~ N3 /6。
-
-
-
-**增长数量级**
-
-增长数量级将算法与它的实现隔离开来,一个算法的增长数量级为 N3 与它是否用 Java 实现,是否运行于特定计算机上无关。
-
-
-
-**内循环**
+### 3. 内循环
执行最频繁的指令决定了程序执行的总时间,把这些指令称为程序的内循环。
-**成本模型**
+### 4. 成本模型
使用成本模型来评估算法,例如数组的访问次数就是一种成本模型。
-## 3. ThreeSum
+## ThreeSum
-ThreeSum 程序用于统计一个数组中三元组的和为 0 的数量。
+ThreeSum 用于统计一个数组中三元组的和为 0 的数量。
```java
public class ThreeSum {
- public static int count(int[] a) {
- int N = a.length;
+ public int count(int[] nums) {
+ int N = nums.length;
int cnt = 0;
for (int i = 0; i < N; i++) {
for (int j = i + 1; j < N; j++) {
for (int k = j + 1; k < N; k++) {
- if (a[i] + a[j] + a[k] == 0) {
+ if (nums[i] + nums[j] + nums[k] == 0) {
cnt++;
}
}
@@ -133,99 +75,415 @@ public class ThreeSum {
}
```
-该程序的内循环为 if (a[i] + a[j] + a[k] == 0) 语句,总共执行的次数为 N3 /6-N2 /2+N/3,因此它的近似执行次数为 \~N3 /6,增长数量级为 N3 。
+该算法的内循环为 if(a[i]+a[j]+a[k]==0) 语句,总共执行的次数为 N(N-1)(N-2) = N3 /6-N2 /2+N/3,因此它的近似执行次数为 \~N3 /6,增长数量级为 O(N3 )。
-**改进**
+ **改进**
通过将数组先排序,对两个元素求和,并用二分查找方法查找是否存在该和的相反数,如果存在,就说明存在三元组的和为 0。
-该方法可以将 ThreeSum 算法增长数量级降低为 N2 logN。
+该方法可以将 ThreeSum 算法增长数量级降低为 O(N2 logN)。
```java
public class ThreeSumFast {
- public static int count(int[] a) {
- Arrays.sort(a);
- int N = a.length;
+ public int count(int[] nums) {
+ Arrays.sort(nums);
+ int N = nums.length;
int cnt = 0;
for (int i = 0; i < N; i++) {
for (int j = i + 1; j < N; j++) {
- for (int k = j + 1; k < N; k++) {
- // rank() 方法返回元素在数组中的下标,如果元素不存在,这里会返回 -1。应该注意这里的下标必须大于 j,这样就不会重复统计了。
- if (BinarySearch.rank(-a[i] - a[j], a) > j) {
- cnt++;
- }
+ // 应该注意这里的下标必须大于 j,否则会重复统计。
+ if (binarySearch(nums, -nums[i] - nums[j]) > j) {
+ cnt++;
}
}
}
return cnt;
}
+
+ private int binarySearch(int[] nums, int target) {
+ int l = 0, h = nums.length - 1;
+ while (l <= h) {
+ int m = l + (h - l) / 2;
+ if (target == nums[m]) return m;
+ else if (target > nums[m]) l = m + 1;
+ else h = m - 1;
+ }
+ return -1;
+ }
}
```
-## 4. 倍率实验
+## 倍率实验
-如果 T(N) \~ aNb lgN,那么 T(2N)/T(N) \~ 2b ,例如对于暴力方法的 ThreeSum 算法,近似时间为 \~N3 /6,对它进行倍率实验得到如下结果:
+如果 T(N) \~ aNb logN,那么 T(2N)/T(N) \~ 2b 。
-
+例如对于暴力方法的 ThreeSum 算法,近似时间为 \~N3 /6。进行如下实验:多次运行该算法,每次取的 N 值为前一次的两倍,统计每次执行的时间,并统计本次运行时间与前一次运行时间的比值,得到如下结果:
-可见 T(2N)/T(N)\~23 ,也就是 b 为 3。
+| N | Time | Ratio |
+| :---: | :---: | :---: |
+| 250 | 0.0 | 2.7 |
+| 500 | 0.0 | 4.8 |
+| 1000 | 0.1 | 6.9 |
+| 2000 | 0.8 | 7.7 |
+| 4000 | 6.4 | 8.0 |
+| 8000 | 51.1 | 8.0 |
-## 5. 注意事项
+可以看到,T(2N)/T(N) \~ 23 ,因此可以确定 T(N) \~ aN3 logN。
-**大常数**
+## 注意事项
+
+### 1. 大常数
在求近似时,如果低级项的常数系数很大,那么近似的结果就是错误的。
-**缓存**
+### 2. 缓存
计算机系统会使用缓存技术来组织内存,访问数组相邻的元素会比访问不相邻的元素快很多。
-**对最坏情况下的性能的保证**
+### 3. 对最坏情况下的性能的保证
在核反应堆、心脏起搏器或者刹车控制器中的软件,最坏情况下的性能是十分重要的。
-**随机化算法**
+### 4. 随机化算法
通过打乱输入,去除算法对输入的依赖。
-**均摊分析**
+### 5. 均摊分析
-将所有操作的总成本所以操作总数来将成本均摊。例如对一个空栈进行 N 次连续的 push() 调用需要访问数组的元素为 N+4+8+16+...+2N=5N-4(N 是向数组写入元素,其余的都是调整数组大小时进行复制需要的访问数组操作),均摊后每次操作访问数组的平均次数为常数。
+将所有操作的总成本除于操作总数来将成本均摊。例如对一个空栈进行 N 次连续的 push() 调用需要访问数组的元素为 N+4+8+16+...+2N=5N-4(N 是向数组写入元素,其余的都是调整数组大小时进行复制需要的访问数组操作),均摊后每次操作访问数组的平均次数为常数。
+
+# 二、栈和队列
+
+## 栈
+
+> First-In-Last-Out
+
+ **1. 数组实现**
+
+```java
+public class ResizeArrayStack- implements Iterable
- {
+ private Item[] a = (Item[]) new Object[1];
+ private int N = 0;
+
+ public void push(Item item) {
+ if (N >= a.length) {
+ resize(2 * a.length);
+ }
+ a[N++] = item;
+ }
+
+ public Item pop() {
+ Item item = a[--N];
+ if (N <= a.length / 4) {
+ resize(a.length / 2);
+ }
+ return item;
+ }
+
+ // 调整数组大小,使得栈具有伸缩性
+ private void resize(int size) {
+ Item[] tmp = (Item[]) new Object[size];
+ for (int i = 0; i < N; i++) {
+ tmp[i] = a[i];
+ }
+ a = tmp;
+ }
+
+ public boolean isEmpty() {
+ return N == 0;
+ }
+
+ public int size() {
+ return N;
+ }
+
+ @Override
+ public Iterator
- iterator() {
+ // 需要返回逆序遍历的迭代器
+ return new ReverseArrayIterator();
+ }
+
+ private class ReverseArrayIterator implements Iterator
- {
+ private int i = N;
+
+ @Override
+ public boolean hasNext() {
+ return i > 0;
+ }
+
+ @Override
+ public Item next() {
+ return a[--i];
+ }
+ }
+}
+```
+
+
**2. 链表实现**
+
+需要使用链表的头插法来实现,因为头插法中最后压入栈的元素在链表的开头,它的 next 指针指向前一个压入栈的元素,在弹出元素使就可以通过 next 指针遍历到前一个压入栈的元素从而让这个元素称为新的栈顶元素。
+
+```java
+public class Stack- {
+
+ private Node top = null;
+ private int N = 0;
+
+ private class Node {
+ Item item;
+ Node next;
+ }
+
+ public boolean isEmpty() {
+ return N == 0;
+ }
+
+ public int size() {
+ return N;
+ }
+
+ public void push(Item item) {
+ Node newTop = new Node();
+ newTop.item = item;
+ newTop.next = top;
+ top = newTop;
+ N++;
+ }
+
+ public Item pop() {
+ Item item = top.item;
+ top = top.next;
+ N--;
+ return item;
+ }
+}
+```
+
+## 队列
+
+> First-In-First-Out
+
+下面是队列的链表实现,需要维护 first 和 last 节点指针,分别指向队首和队尾。
+
+这里需要考虑 first 和 last 指针哪个作为链表的开头。因为出队列操作需要让队首元素的下一个元素成为队首,所以需要容易获取下一个元素,而链表的头部节点的 next 指针指向下一个元素,因此可以让 first 指针链表的开头。
+
+```java
+public class Queue
- {
+ private Node first;
+ private Node last;
+ int N = 0;
+ private class Node{
+ Item item;
+ Node next;
+ }
+
+ public boolean isEmpty(){
+ return N == 0;
+ }
+
+ public int size(){
+ return N;
+ }
+
+ // 入队列
+ public void enqueue(Item item){
+ Node newNode = new Node();
+ newNode.item = item;
+ newNode.next = null;
+ if(isEmpty()){
+ last = newNode;
+ first = newNode;
+ } else{
+ last.next = newNode;
+ last = newNode;
+ }
+ N++;
+ }
+
+ // 出队列
+ public Item dequeue() {
+ if (isEmpty()) return null;
+ Node node = first;
+ first = first.next;
+ N--;
+ if (isEmpty()) last = null;
+ return node.item;
+ }
+}
+```
+
+# 三、union-find
-# 排序
+用于解决动态连通性问题,能动态连接两个点,并且判断两个点是否连通。
-## 1. 初级排序算法
+
-### 1.1 约定
-待排序的元素需要实现 Java 的 Comparable 接口,该接口有 compareTo() 方法。
+| 方法 | 描述 |
+| :---: | :---: |
+| UF(int N) | 构造一个大小为 N 的并查集 |
+| void union(int p, int q) | 连接 p 和 q 节点 |
+| int find(int p) | 查找 p 所在的连通分量 |
+| boolean connected(int p, int q) | 判断 p 和 q 节点是否连通 |
+
+```java
+public class UF {
+ private int[] id;
+
+ public UF(int N) {
+ id = new int[N];
+ for (int i = 0; i < N; i++) {
+ id[i] = i;
+ }
+ }
+
+ public boolean connected(int p, int q) {
+ return find(p) == find(q);
+ }
+}
+```
+
+## quick-find
+
+可以快速进行 find 操作,即可以快速判断两个节点是否连通。
+
+同一连通分量的所有节点的 id 值相等。
+
+但是 union 操作代价却很高,需要将其中一个连通分量中的所有节点 id 值都修改为另一个节点的 id 值。
+
+
+
+```java
+public int find(int p) {
+ return id[p];
+}
+public void union(int p, int q) {
+ int pID = find(p);
+ int qID = find(q);
+
+ if (pID == qID) return;
+ for (int i = 0; i < id.length; i++) {
+ if (id[i] == pID) id[i] = qID;
+ }
+}
+```
+
+## quick-union
+
+可以快速进行 union 操作,只需要修改一个节点的 id 值即可。
+
+但是 find 操作开销很大,因为同一个连通分量的节点 id 值不同,id 值只是用来指向另一个节点。因此需要一直向上查找操作,直到找到最上层的节点。
+
+
+
+```java
+ public int find(int p) {
+ while (p != id[p]) p = id[p];
+ return p;
+ }
+
+ public void union(int p, int q) {
+ int pRoot = find(p);
+ int qRoot = find(q);
+ if (pRoot == qRoot) return;
+ id[pRoot] = qRoot;
+ }
+```
+
+这种方法可以快速进行 union 操作,但是 find 操作和树高成正比,最坏的情况下树的高度为触点的数目。
+
+
+
+## 加权 quick-union
+
+为了解决 quick-union 的树通常会很高的问题,加权 quick-union 在 union 操作时会让较小的树连接较大的树上面。
+
+理论研究证明,加权 quick-union 算法构造的树深度最多不超过 logN。
+
+
+
+```java
+public class WeightedQuickUnionUF {
+ private int[] id;
+ // 保存节点的数量信息
+ private int[] sz;
+
+ public WeightedQuickUnionUF(int N) {
+ id = new int[N];
+ sz = new int[N];
+ for (int i = 0; i < N; i++) {
+ id[i] = i;
+ sz[i] = 1;
+ }
+ }
+
+ public boolean connected(int p, int q) {
+ return find(p) == find(q);
+ }
+
+ public int find(int p) {
+ while (p != id[p]) p = id[p];
+ return p;
+ }
+
+ public void union(int p, int q) {
+ int i = find(p);
+ int j = find(q);
+ if (i == j) return;
+ if (sz[i] < sz[j]) {
+ id[i] = j;
+ sz[j] += sz[i];
+ } else {
+ id[j] = i;
+ sz[i] += sz[j];
+ }
+ }
+}
+```
+
+## 路径压缩的加权 quick-union
+
+在检查节点的同时将它们直接链接到根节点,只需要在 find 中添加一个循环即可。
+
+## 各种 union-find 算法的比较
+
+| 算法 | union | find |
+| :---: | :---: | :---: |
+| quick-find | N | 1 |
+| quick-union | 树高 | 树高 |
+| 加权 quick-union | logN | logN |
+| 路径压缩的加权 quick-union | 非常接近 1 | 非常接近 1 |
+
+# 四、排序
+
+待排序的元素需要实现 Java 的 Comparable 接口,该接口有 compareTo() 方法,可以用它来判断两个元素的大小关系。
研究排序算法的成本模型时,计算的是比较和交换的次数。
使用辅助函数 less() 和 exch() 来进行比较和交换的操作,使得代码的可读性和可移植性更好。
```java
-private boolean less(Comparable v, Comparable w){
+private boolean less(Comparable v, Comparable w) {
return v.compareTo(w) < 0;
}
-private void exch(Comparable[] a, int i, int j){
+private void exch(Comparable[] a, int i, int j) {
Comparable t = a[i];
a[i] = a[j];
a[j] = t;
}
```
-### 1.2 选择排序
+## 选择排序
-找到数组中的最小元素,然后将它与数组的第一个元素交换位置。然后再从剩下的元素中找到最小的元素,将它与数组的第二个元素交换位置。不断进行这样的操作,直到将整个数组排序。
+找到数组中的最小元素,将它与数组的第一个元素交换位置。再从剩下的元素中找到最小的元素,将它与数组的第二个元素交换位置。不断进行这样的操作,直到将整个数组排序。
-
+
```java
public class Selection {
- public static void sort(Comparable[] a) {
+ public void sort(Comparable[] a) {
int N = a.length;
for (int i = 0; i < N; i++) {
int min = i;
@@ -240,15 +498,15 @@ public class Selection {
选择排序需要 \~N2 /2 次比较和 \~N 次交换,它的运行时间与输入无关,这个特点使得它对一个已经排序的数组也需要这么多的比较和交换操作。
-### 1.3 插入排序
+## 插入排序
-将一个元素插入到已排序的数组中,使得插入之后的数组也是有序的。插入排序从左到右插入每个元素,每次插入之后左部的子数组是有序的。
+插入排序从左到右进行,每次都将当前元素插入到左部已经排序的数组中,使得插入之后左部数组依然有序。
-
+
```java
public class Insertion {
- public static void sort(Comparable[] a) {
+ public void sort(Comparable[] a) {
int N = a.length;
for (int i = 1; i < N; i++) {
for (int j = i; j > 0 && less(a[j], a[j - 1]); j--) {
@@ -259,15 +517,15 @@ public class Insertion {
}
```
-插入排序的复杂度取决于数组的初始顺序,如果数组已经部分有序了,那么插入排序会很快。平均情况下插入排序需要 \~N2 /4 比较以及 \~N2 /4 次交换,最坏的情况下需要 \~N2 /2 比较以及 \~N2 /2 次交换,最坏的情况是数组是逆序的;而最好的情况下需要 N-1 次比较和 0 次交换,最好的情况就是数组已经有序了。
+插入排序的复杂度取决于数组的初始顺序,如果数组已经部分有序了,那么插入排序会很快。
+
+- 平均情况下插入排序需要 \~N2 /4 比较以及 \~N2 /4 次交换;
+- 最坏的情况下需要 \~N2 /2 比较以及 \~N2 /2 次交换,最坏的情况是数组是逆序的;
+- 最好的情况下需要 N-1 次比较和 0 次交换,最好的情况就是数组已经有序了。
插入排序对于部分有序数组和小规模数组特别高效。
-### 1.4 选择排序和插入排序的比较
-
-对于随机排序的无重复主键的数组,插入排序和选择排序的运行时间是平方级别的,两者之比是一个较小的常数。
-
-### 1.5 希尔排序
+## 希尔排序
对于大规模的数组,插入排序很慢,因为它只能交换相邻的元素,如果要把元素从一端移到另一端,就需要很多次操作。
@@ -275,15 +533,15 @@ public class Insertion {
希尔排序使用插入排序对间隔 h 的序列进行排序,如果 h 很大,那么元素就能很快的移到很远的地方。通过不断减小 h,最后令 h=1,就可以使得整个数组是有序的。
-
+
```java
public class Shell {
- public static void sort(Comparable[] a) {
+ public void sort(Comparable[] a) {
int N = a.length;
int h = 1;
while (h < N / 3) {
- h = 3 * h + 1;// 1, 4, 13, 40, ...
+ h = 3 * h + 1; // 1, 4, 13, 40, ...
}
while (h >= 1) {
for (int i = h; i < N; i++) {
@@ -297,21 +555,23 @@ public class Shell {
}
```
-希尔排序的运行时间达不到平方级别,使用递增序列 1, 4, 13, 40, ... 的希尔排序所需要的比较次数不会超过 N 的若干倍乘于递增序列的长度。后面介绍的高级排序算法只会比希尔排序快两倍左右。
+希尔排序的运行时间达不到平方级别,使用递增序列 1, 4, 13, 40, ... 的希尔排序所需要的比较次数不会超过 N 的若干倍乘于递增序列的长度。后面介绍的高级排序算法只会比希尔排序快两倍左右。
-## 2 归并排序
+## 归并排序
归并排序的思想是将数组分成两部分,分别进行排序,然后归并起来。
-
+
-### 2.1 归并方法
+### 1. 归并方法
+
+归并方法将数组中两个已经排序的部分归并成一个。
```java
public class MergeSort {
- private static Comparable[] aux;
+ private Comparable[] aux;
- private static void merge(Comparable[] a, int lo, int mid, int hi) {
+ private void merge(Comparable[] a, int lo, int mid, int hi) {
int i = lo, j = mid + 1;
for (int k = lo; k <= hi; k++) {
@@ -328,15 +588,18 @@ public class MergeSort {
}
```
-### 2.2 自顶向下归并排序
+### 2. 自顶向下归并排序
+
+
+
```java
-public static void sort(Comparable[] a) {
+public void sort(Comparable[] a) {
aux = new Comparable[a.length];
sort(a, 0, a.length - 1);
}
-private static void sort(Comparable[] a, int lo, int hi) {
+private void sort(Comparable[] a, int lo, int hi) {
if (hi <= lo) return;
int mid = lo + (hi - lo) / 2;
sort(a, lo, mid);
@@ -345,22 +608,16 @@ private static void sort(Comparable[] a, int lo, int hi) {
}
```
-
+因为每次都将问题对半分成两个子问题,而这种对半分的算法复杂度一般为 O(NlogN),因此该归并排序方法的时间复杂度也为 O(NlogN)。
-
+小数组的递归操作会过于频繁,可以在数组过小时切换到插入排序来提高性能。
-很容易看出该排序算法的时间复杂度为 O(NlgN )。
-
-因为小数组的递归操作会过于频繁,因此使用插入排序来处理小数组将会获得更高的性能。
-
-### 2.3 自底向上归并排序
+### 3. 自底向上归并排序
先归并那些微型数组,然后成对归并得到的子数组。
-
-
```java
-public static void busort(Comparable[] a) {
+public void busort(Comparable[] a) {
int N = a.length;
aux = new Comparable[N];
for (int sz = 1; sz < N; sz += sz) {
@@ -371,38 +628,45 @@ public static void busort(Comparable[] a) {
}
```
-## 3. 快速排序
+## 快速排序
-### 3.1 基本算法
+### 1. 基本算法
-归并排序将数组分为两个子数组分别排序,并将有序的子数组归并使得整个数组排序;快速排序通过一个切分元素将数组分为两个子数组,左子数组小于等于切分元素,右子数组大于等于切分元素,将这两个子数组排序也就将整个数组排序了。
+- 归并排序将数组分为两个子数组分别排序,并将有序的子数组归并使得整个数组排序;
+- 快速排序通过一个切分元素将数组分为两个子数组,左子数组小于等于切分元素,右子数组大于等于切分元素,将这两个子数组排序也就将整个数组排序了。
-
+
```java
public class QuickSort {
- public static void sort(Comparable[] a) {
+ public void sort(Comparable[] a) {
shuffle(a);
sort(a, 0, a.length - 1);
}
- private static void sort(Comparable[] a, int lo, int hi) {
+ private void sort(Comparable[] a, int lo, int hi) {
if (hi <= lo) return;
int j = partition(a, lo, hi);
sort(a, lo, j - 1);
sort(a, j + 1, hi);
}
+
+ private void shuffle(Comparable[] array) {
+ List list = Arrays.asList(array);
+ Collections.shuffle(list);
+ list.toArray(array);
+ }
}
```
-### 3.2 切分
+### 2. 切分
-取 a[lo] 作为切分元素,然后从数组的左端向右扫描直到找到第一个大于等于它的元素,再从数组的右端向左扫描找到第一个小于等于它的元素,交换这两个元素,并不断继续这个过程,就可以保证左指针的左侧元素都不大于切分元素,右指针 j 的右侧元素都不小于切分元素。当两个指针相遇时,将切分元素 a[lo] 和左子数组最右侧的元素 a[j] 交换然后返回 j 即可。
+取 a[lo] 作为切分元素,然后从数组的左端向右扫描直到找到第一个大于等于它的元素,再从数组的右端向左扫描找到第一个小于等于它的元素,交换这两个元素,并不断进行这个过程,就可以保证左指针 i 的左侧元素都不大于切分元素,右指针 j 的右侧元素都不小于切分元素。当两个指针相遇时,将切分元素 a[lo] 和 a[j] 交换位置。
-
+
```java
-private static int partition(Comparable[] a, int lo, int hi) {
+private int partition(Comparable[] a, int lo, int hi) {
int i = lo, j = hi + 1;
Comparable v = a[lo];
while (true) {
@@ -416,35 +680,33 @@ private static int partition(Comparable[] a, int lo, int hi) {
}
```
-### 3.3 性能分析
+### 3. 性能分析
快速排序是原地排序,不需要辅助数组,但是递归调用需要辅助栈。
-快速排序最好的情况下是每次都正好能将数组对半分,这样递归调用次数才是最少的。这种情况下比较次数为 CN =2CN/2 +N,也就是复杂度为 O(NlgN )。
+快速排序最好的情况下是每次都正好能将数组对半分,这样递归调用次数才是最少的。这种情况下比较次数为 CN =2CN/2 +N,复杂度为 O(NlogN)。
最坏的情况下,第一次从最小的元素切分,第二次从第二小的元素切分,如此这般。因此最坏的情况下需要比较 N2 /2。为了防止数组最开始就是有序的,在进行快速排序时需要随机打乱数组。
-### 3.4 算法改进
+### 4. 算法改进
-#### 3.4.1 切换到插入排序
+(一)切换到插入排序
因为快速排序在小数组中也会调用自己,对于小数组,插入排序比快速排序的性能更好,因此在小数组中可以切换到插入排序。
-#### 3.4.2 三取样
+(二)三取样
最好的情况下是每次都能取数组的中位数作为切分元素,但是计算中位数的代价很高。人们发现取 3 个元素并将大小居中的元素作为切分元素的效果最好。
-#### 3.4.3 三向切分
+(三)三向切分
对于有大量重复元素的数组,可以将数组切分为三部分,分别对应小于、等于和大于切分元素。
三向切分快速排序对于只有若干不同主键的随机数组可以在线性时间内完成排序。
-
-
```java
public class Quick3Way {
- public static void sort(Comparable[] a, int lo, int hi) {
+ public void sort(Comparable[] a, int lo, int hi) {
if (hi <= lo) return;
int lt = lo, i = lo + 1, gt = hi;
Comparable v = a[lo];
@@ -460,17 +722,18 @@ public class Quick3Way {
}
```
-## 4. 优先队列
+## 优先队列
优先队列主要用于处理最大元素。
-### 4.1 堆
+### 1. 堆
-定义:一颗二叉树的每个节点都大于等于它的两个子节点。
+堆的某个节点的值总是大于等于子节点的值,并且堆是一颗完全二叉树。
-堆可以用数组来表示,因为堆是一种完全二叉树,而完全二叉树很容易就存储在数组中。位置 k 的节点的父节点位置为 k/2,而它的两个子节点的位置分别为 2k 和 2k+1。这里我们不使用数组索引为 0 的位置,是为了更清晰地理解节点的关系。
-
+堆可以用数组来表示,因为堆是一种完全二叉树,而完全二叉树很容易就存储在数组中。位置 k 的节点的父节点位置为 k/2,而它的两个子节点的位置分别为 2k 和 2k+1。这里我们不使用数组索引为 0 的位置,是为了更清晰地描述节点的位置关系。
+
+
```java
public class MaxPQ {
@@ -501,10 +764,12 @@ public class MaxPQ {
}
```
-### 4.2 上浮和下沉
+### 2. 上浮和下沉
在堆中,当一个节点比父节点大,那么需要交换这个两个节点。交换后还可能比它新的父节点大,因此需要不断地进行比较和交换操作。把这种操作称为上浮。
+
+
```java
private void swim(int k) {
while (k > 1 && less(k / 2, k)) {
@@ -516,6 +781,8 @@ private void swim(int k) {
类似地,当一个节点比子节点来得小,也需要不断的向下比较和交换操作,把这种操作称为下沉。一个节点有两个子节点,应当与两个子节点中最大那么节点进行交换。
+
+
```java
private void sink(int k) {
while (2 * k <= N) {
@@ -528,7 +795,7 @@ private void sink(int k) {
}
```
-### 4.3 插入元素
+### 3. 插入元素
将新元素放到数组末尾,然后上浮到合适的位置。
@@ -539,7 +806,7 @@ public void insert(Key v) {
}
```
-### 4.4 删除最大元素
+### 4. 删除最大元素
从数组顶端删除最大的元素,并将数组的最后一个元素放到顶端,并让这个元素下沉到合适的位置。
@@ -553,15 +820,23 @@ public Key delMax() {
}
```
-### 4.5 堆排序
+### 5. 堆排序
由于堆可以很容易得到最大的元素并删除它,不断地进行这种操作可以得到一个递减序列。如果把最大元素和当前堆中数组的最后一个元素交换位置,并且不删除它,那么就可以得到一个从尾到头的递减序列,从正向来看就是一个递增序列。因此很容易使用堆来进行排序,并且堆排序是原地排序,不占用额外空间。
-堆排序要分两个阶段,第一个阶段是把无序数组建立一个堆;第二个阶段是交换最大元素和当前堆的数组最后一个元素,并且进行下沉操作维持堆的有序状态。
+(一)构建堆
无序数组建立堆最直接的方法是从左到右遍历数组,然后进行上浮操作。一个更高效的方法是从右至左进行下沉操作,如果一个节点的两个节点都已经是堆有序,那么进行下沉操作可以使得这个节点为根节点的堆有序。叶子节点不需要进行下沉操作,因此可以忽略叶子节点的元素,因此只需要遍历一半的元素即可。
-
+
+
+(二)交换堆顶元素与最后一个元素
+
+交换之后需要进行下沉操作维持堆的有序状态。
+
+
+
+
```java
public static void sort(Comparable[] a){
@@ -576,33 +851,41 @@ public static void sort(Comparable[] a){
}
```
-### 4.6 分析
+### 6. 分析
-一个堆的高度为 lgN,因此在堆中插入元素和删除最大元素的复杂度都为 lgN。
+一个堆的高度为 logN,因此在堆中插入元素和删除最大元素的复杂度都为 logN。
-对于堆排序,由于要对 N 个节点进行下沉操作,因此复杂度为 NlgN。
+对于堆排序,由于要对 N 个节点进行下沉操作,因此复杂度为 NlogN。
堆排序时一种原地排序,没有利用额外的空间。
现代操作系统很少使用堆排序,因为它无法利用缓存,也就是数组元素很少和相邻的元素进行比较。
-## 5. 应用
+## 应用
-### 5.1 排序算法的比较
+### 1. 排序算法的比较
-
+| 算法 | 稳定 | 原地排序 | 时间复杂度 | 空间复杂度 | 备注 |
+| :---: | :---: | :---: | :---: | :---: | :---: |
+| 选择排序 | no | yes | N2 | 1 | |
+| 插入排序 | yes | yes | N \~ N2 | 1 | 时间复杂度和初始顺序有关 |
+| 希尔排序 | no | yes | N 的若干倍乘于递增序列的长度 | 1 | |
+| 快速排序 | no | yes | NlogN | logN | |
+| 三向切分快速排序 | no | yes | N \~ NlogN | logN | 适用于有大量重复主键|
+| 归并排序 | yes | no | NlogN | N | |
+| 堆排序 | no | yes | NlogN | 1 | | |
-快速排序时最快的通用排序算法,它的内循环的指令很少,而且它还能利用缓存,因为它总是顺序地访问数据。它的运行时间增长数量级为 \~cNlgN,这里的 c 比其他线性对数级别的排序算法都要小。使用三向切分之后,实际应用中可能出现的某些分布的输入能够达到线性级别,而其它排序算法仍然需要线性对数时间。
+快速排序是最快的通用排序算法,它的内循环的指令很少,而且它还能利用缓存,因为它总是顺序地访问数据。它的运行时间近似为 \~cNlogN,这里的 c 比其他线性对数级别的排序算法都要小。使用三向切分之后,实际应用中可能出现的某些分布的输入能够达到线性级别,而其它排序算法仍然需要线性对数时间。
-### 5.2 Java 的排序算法实现
+### 2. Java 的排序算法实现
Java 系统库中的主要排序方法为 java.util.Arrays.sort(),对于原始数据类型使用三向切分的快速排序,对于引用类型使用归并排序。
-### 5.3 基于切分的快速选择算法
+### 3. 基于切分的快速选择算法
-快速排序的 partition() 方法,会将数组的 a[lo] 至 a[hi] 重新排序并返回一个整数 j 使得 a[lo..j-1] 小于等于 a[j],且 a[j+1..hi] 大于等于 a[j]。那么如果 j=k,a[j] 就是第 k 个数。
+快速排序的 partition() 方法,会返回一个整数 j 使得 a[lo..j-1] 小于等于 a[j],且 a[j+1..hi] 大于等于 a[j],此时 a[j] 就是数组的第 j 大元素。
-该算法是线性级别的,因为每次正好将数组二分,那么比较的总次数为 (N+N/2+N/4+..),直到找到第 k 个元素,这个和显然小于 2N。
+可以利用这个特性找出数组的第 k 个元素。
```java
public static Comparable select(Comparable[] a, int k) {
@@ -617,33 +900,23 @@ public static Comparable select(Comparable[] a, int k) {
}
```
-# 查找
+该算法是线性级别的,因为每次正好将数组二分,那么比较的总次数为 (N+N/2+N/4+..),直到找到第 k 个元素,这个和显然小于 2N。
-本章使用三种经典的数据结构来实现高效的符号表:二叉查找树、红黑树和散列表。
+# 五、查找
-## 1. 符号表
+符号表是一种存储键值对的数据结构,主要支持两种操作:插入一个新的键值对、根据给定键得到值。
-### 1.1 无序符号表
-
-
-
-### 1.2 有序符号表
-
-
+符号表分为有序和无序两种,有序符号表主要指支持 min()、max() 等根据键的大小关系来实现的操作。
有序符号表的键需要实现 Comparable 接口。
-查找的成本模型:键的比较次数,在不进行比较时使用数组的访问次数。
-
-### 1.3 二分查找实现有序符号表
+## 二分查找实现有序符号表
使用一对平行数组,一个存储键一个存储值。
-需要创建一个 Key 类型的 Comparable 对象数组和一个 Value 类型的 Object 对象数组。
-
rank() 方法至关重要,当键在表中时,它能够知道该键的位置;当键不在表中时,它也能知道在何处插入新键。
-复杂度:二分查找最多需要 lgN+1 次比较,使用二分查找实现的符号表的查找操作所需要的时间最多是对数级别的。但是插入操作需要移动数组元素,是线性级别的。
+复杂度:二分查找最多需要 logN+1 次比较,使用二分查找实现的符号表的查找操作所需要的时间最多是对数级别的。但是插入操作需要移动数组元素,是线性级别的。
```java
public class BinarySearchST, Value> {
@@ -702,15 +975,21 @@ public class BinarySearchST, Value> {
}
```
-## 2. 二叉查找树
+## 二叉查找树
-**二叉树** 定义为一个空链接,或者是一个有左右两个链接的节点,每个链接都指向一颗子二叉树。
+**二叉树** 是一个空链接,或者是一个有左右两个链接的节点,每个链接都指向一颗子二叉树。
-**二叉查找树** (BST)是一颗二叉树,并且每个节点的键都大于其左子树中的任意节点的键而小于右子树的任意节点的键。
+
-
+**二叉查找树** (BST)是一颗二叉树,并且每个节点的值都大于等于其左子树中的所有节点的值而小于等于右子树的所有节点的值。
-二叉查找树的查找操作每次迭代都会让区间减少一半,和二分查找类似。
+
+
+BST 有一个重要性质,就是它的中序遍历结果递增排序。
+
+
+
+基本数据结构:
```java
public class BST, Value> {
@@ -741,9 +1020,13 @@ public class BST, Value> {
}
```
-### 2.1 get()
+(为了方便绘图,二叉树的空链接不画出来。)
-如果树是空的,则查找未命中;如果被查找的键和根节点的键相等,查找命中,否则递归地在子树中查找:如果被查找的键较小就在左子树中查找,较大就在右子树中查找。
+### 1. get()
+
+- 如果树是空的,则查找未命中;
+- 如果被查找的键和根节点的键相等,查找命中;
+- 否则递归地在子树中查找:如果被查找的键较小就在左子树中查找,较大就在右子树中查找。
```java
public Value get(Key key) {
@@ -758,10 +1041,12 @@ private Value get(Node x, Key key) {
}
```
-### 2.2 put()
+### 2. put()
当插入的键不存在于树中,需要创建一个新节点,并且更新上层节点的链接使得该节点正确链接到树中。
+
+
```java
public void put(Key key, Value val) {
root = put(root, key, val);
@@ -777,17 +1062,23 @@ private Node put(Node x, Key key, Value val) {
}
```
-### 2.3 分析
+### 3. 分析
-二叉查找树的算法运行时间取决于树的形状,而树的形状又取决于键被插入的先后顺序。最好的情况下树是完全平衡的,每条空链接和根节点的距离都为 lgN。在最坏的情况下,树的高度为 N。
+二叉查找树的算法运行时间取决于树的形状,而树的形状又取决于键被插入的先后顺序。最好的情况下树是完全平衡的,每条空链接和根节点的距离都为 logN。
-
+
-复杂度:查找和插入操作都为对数级别。
+在最坏的情况下,树的高度为 N。
-### 2.4 floor()
+
+
+### 4. floor()
+
+floor(key):小于等于键的最大键
+
+- 如果键小于根节点的键,那么 floor(key) 一定在左子树中;
+- 如果键大于根节点的键,需要先判断右子树中是否存在 floor(key),如果存在就找到,否则根节点就是 floor(key)。
-如果 key 小于根节点的 key,那么小于等于 key 的最大键节点一定在左子树中;如果 key 大于根节点的 key,只有当根节点右子树中存在小于等于 key 的节点,小于等于 key 的最大键节点才在右子树中,否则根节点就是小于等于 key 的最大键节点。
```java
public Key floor(Key key) {
@@ -809,7 +1100,13 @@ private Node floor(Node x, Key key) {
}
```
-### 2.5 rank()
+### 5. rank()
+
+rank(key) 返回 key 的排名。
+
+- 如果键和根节点的键相等,返回左子树的节点数;
+- 如果小于,递归计算在左子树中的排名;
+- 如果大于,递归计算在右子树中的排名,并加上左子树的节点数,再加上 1(根节点)。
```java
public int rank(Key key) {
@@ -824,20 +1121,21 @@ private int rank(Key key, Node x) {
}
```
-### 2.6 min()
+### 6. min()
```java
private Node min(Node x) {
+ if (x == null) return null;
if (x.left == null) return x;
return min(x.left);
}
```
-### 2.7 deleteMin()
+### 7. deleteMin()
令指向最小节点的链接指向最小节点的右子树。
-
+
```java
public void deleteMin() {
@@ -851,11 +1149,13 @@ public Node deleteMin(Node x) {
}
```
-### 2.8 delete()
+### 8. delete()
-如果待删除的节点只有一个子树,那么只需要让指向待删除节点的链接指向唯一的子树即可;否则,让右子树的最小节点替换该节点。
+- 如果待删除的节点只有一个子树,那么只需要让指向待删除节点的链接指向唯一的子树即可;
+- 否则,让右子树的最小节点替换该节点。
+
+
-
```java
public void delete(Key key) {
@@ -879,9 +1179,9 @@ private Node delete(Node x, Key key) {
}
```
-### 2.9 keys()
+### 9. keys()
-利用二叉查找树中序遍历的结果为有序序列的特点。
+利用二叉查找树中序遍历的结果为递增的特点。
```java
public Iterable keys(Key lo, Key hi) {
@@ -899,35 +1199,41 @@ private void keys(Node x, Queue queue, Key lo, Key hi) {
}
```
-### 2.10 性能分析
+### 10. 性能分析
复杂度:二叉查找树所有操作在最坏的情况下所需要的时间都和树的高度成正比。
-## 3. 平衡查找树
+## 2-3 查找树
-### 3.1 2-3 查找树
+
-
+2-3 查找树引入了 2- 节点和 3- 节点,目的是为了让树平衡。一颗完美平衡的 2-3 查找树的所有空链接到根节点的距离应该是相同的。
-一颗完美平衡的 2-3 查找树的所有空链接到根节点的距离应该是相同的。
+### 1. 插入操作
-#### 3.1.1 插入操作
+插入操作和 BST 的插入操作有很大区别,BST 的插入操作是先进行一次未命中的查找,然后再将节点插入到对应的空链接上。但是 2-3 查找树如果也这么做的话,那么就会破坏了平衡性。它是将新节点插入到叶子节点上。
-当插入之后产生一个临时 4- 节点时,需要将 4- 节点分裂成 3 个 2- 节点,并将中间的 2- 节点移到上层节点中。如果上移操作继续产生临时 4- 节点则一直进行分裂上移,直到不存在临时 4- 节点。
+根据叶子节点的类型不同,有不同的处理方式。
-
+插入到 2- 节点上,那么直接将新节点和原来的节点组成 3- 节点即可。
-#### 3.1.2 性质
+
+
+如果是插入到 3- 节点上,就会产生一个临时 4- 节点时,需要将 4- 节点分裂成 3 个 2- 节点,并将中间的 2- 节点移到上层节点中。如果上移操作继续产生临时 4- 节点则一直进行分裂上移,直到不存在临时 4- 节点。
+
+
+
+### 2. 性质
2-3 查找树插入操作的变换都是局部的,除了相关的节点和链接之外不必修改或者检查树的其它部分,而这些局部变换不会影响树的全局有序性和平衡性。
-2-3 查找树的查找和插入操作复杂度和插入顺序 **无关** ,在最坏的情况下查找和插入操作访问的节点必然不超过 logN 个,含有 10 亿个节点的 2-3 查找树最多只需要访问 30 个节点就能进行任意的查找和插入操作。
+2-3 查找树的查找和插入操作复杂度和插入顺序无关,在最坏的情况下查找和插入操作访问的节点必然不超过 logN 个,含有 10 亿个节点的 2-3 查找树最多只需要访问 30 个节点就能进行任意的查找和插入操作。
-### 3.2 红黑二叉查找树
+## 红黑二叉查找树
2-3 查找树需要用到 2- 节点和 3- 节点,红黑树使用红链接来实现 3- 节点。指向一个节点的链接颜色如果为红色,那么这个节点和上层节点表示的是一个 3- 节点,而黑色则是普通链接。
-
+
红黑树具有以下性质:
@@ -936,7 +1242,7 @@ private void keys(Node x, Queue queue, Key lo, Key hi) {
画红黑树时可以将红链接画平。
-
+
```java
public class RedBlackBST, Value> {
@@ -966,13 +1272,11 @@ public class RedBlackBST, Value> {
}
```
-#### 3.2.1 左旋转
+### 1. 左旋转
因为合法的红链接都为左链接,如果出现右链接为红链接,那么就需要进行左旋转操作。
-
-
-
+
```java
public Node rotateLeft(Node h) {
@@ -987,13 +1291,11 @@ public Node rotateLeft(Node h) {
}
```
-#### 3.2.2 右旋转
+### 2. 右旋转
进行右旋转是为了转换两个连续的左红链接,这会在之后的插入过程中探讨。
-
-
-
+
```java
public Node rotateRight(Node h) {
@@ -1007,13 +1309,11 @@ public Node rotateRight(Node h) {
}
```
-#### 3.2.3 颜色转换
+### 3. 颜色转换
一个 4- 节点在红黑树中表现为一个节点的左右子节点都是红色的。分裂 4- 节点除了需要将子节点的颜色由红变黑之外,同时需要将父节点的颜色由黑变红,从 2-3 树的角度看就是将中间节点移到上层节点。
-
-
-
+
```java
void flipColors(Node h){
@@ -1023,15 +1323,15 @@ void flipColors(Node h){
}
```
-#### 3.2.4 插入
+### 4. 插入
先将一个节点按二叉查找树的方法插入到正确位置,然后再进行如下颜色操作:
- 如果右子节点是红色的而左子节点是黑色的,进行左旋转;
-- 如果左子节点是红色的且它的左子节点也是红色的,进行右旋转;
+- 如果左子节点是红色的,而且左子节点的左子节点也是红色的,进行右旋转;
- 如果左右子节点均为红色的,进行颜色转换。
-
+
```java
public void put(Key key, Value val) {
@@ -1055,37 +1355,23 @@ private Node put(Node x, Key key, Value val) {
}
```
-可以看到该插入操作和 BST 的插入操作类似,只是在最后加入了旋转和颜色变换操作即可。
+可以看到该插入操作和二叉查找树的插入操作类似,只是在最后加入了旋转和颜色变换操作即可。
根节点一定为黑色,因为根节点没有上层节点,也就没有上层节点的左链接指向根节点。flipColors() 有可能会使得根节点的颜色变为红色,每当根节点由红色变成黑色时树的黑链接高度加 1.
-#### 3.2.5 删除最小键
+### 5. 分析
-如果最小键在一个 2- 节点中,那么删除该键会留下一个空链接,就破坏了平衡性,因此要确保最小键不在 2- 节点中。将 2- 节点转换成 3- 节点或者 4- 节点有两种方法,一种是向上层节点拿一个 key,一种是向兄弟节点拿一个 key。如果上层节点是 2- 节点,那么就没办法从上层节点拿 key 了,因此要保证删除路径上的所有节点都不是 2- 节点。在向下删除的过程中,保证以下情况之一发生:
-
-1. 如果当前节点的左子节点不是 2- 节点,完成;
-2. 如果当前节点的左子节点是 2- 节点而它的兄弟节点不是 2- 节点,向兄弟节点拿一个 key 过来;
-3. 如果当前节点的左子节点和它的兄弟节点都是 2- 节点,将左子节点、父节点中的最小键和最近的兄弟节点合并为一个 4- 节点。
-
-
-
-最后得到一个含有最小键的 3- 节点或者 4- 节点,直接从中删除。然后再从头分解所有临时的 4- 节点。
-
-
-
-#### 3.2.6 分析
-
-一颗大小为 N 的红黑树的高度不会超过 2lgN。最坏的情况下是它所对应的 2-3 树中构成最左边的路径节点全部都是 3- 节点而其余都是 2- 节点。
+一颗大小为 N 的红黑树的高度不会超过 2logN。最坏的情况下是它所对应的 2-3 树,构成最左边的路径节点全部都是 3- 节点而其余都是 2- 节点。
红黑树大多数的操作所需要的时间都是对数级别的。
-## 4. 散列表
+## 散列表
-散列表类似于数组,可以把散列表的散列值看成数组的索引值。访问散列表和访问数组元素一样快速,它可以在常数时间内实现查找和插入的符号表。
+散列表类似于数组,可以把散列表的散列值看成数组的索引值。访问散列表和访问数组元素一样快速,它可以在常数时间内实现查找和插入操作。
由于无法通过散列值知道键的大小关系,因此散列表无法实现有序性操作。
-### 4.1 散列函数
+### 1. 散列函数
对于一个大小为 M 的散列表,散列函数能够把任意键转换为 [0, M-1] 内的正整数,该正整数即为 hash 值。
@@ -1093,7 +1379,7 @@ private Node put(Node x, Key key, Value val) {
散列函数应该满足以下三个条件:
-1. 一致性:相等的键应当有相等的 hash 值。
+1. 一致性:相等的键应当有相等的 hash 值,两个键相等表示调用 equals() 返回的值相等。
2. 高效性:计算应当简便,有必要的话可以把 hash 值缓存起来,在调用 hash 函数时直接返回。
3. 均匀性:所有键的 hash 值应当均匀地分布到 [0, M-1] 之间,这个条件至关重要,直接影响到散列表的性能。
@@ -1111,13 +1397,13 @@ for(int i = 0; i < s.length(); i++)
hash = (R * hash + s.charAt(i)) % M;
```
-再比如,拥有多个成员的自定义类的哈希函数如下
+再比如,拥有多个成员的自定义类的哈希函数如下:
```java
int hash = (((day * R + month) % M) * R + year) % M;
```
-R 的值不是很重要,通常取 31。
+R 通常取 31。
Java 中的 hashCode() 实现了 hash 函数,但是默认使用对象的内存地址值。在使用 hashCode() 函数时,应当结合除留余数法来使用。因为内存地址是 32 位整数,我们只需要 31 位的非负整数,因此应当屏蔽符号位之后再使用除留余数法。
@@ -1143,19 +1429,19 @@ public class Transaction{
}
```
-### 4.2 基于拉链法的散列表
+### 2. 基于拉链法的散列表
拉链法使用链表来存储 hash 值相同的键,从而解决冲突。此时查找需要分两步,首先查找 Key 所在的链表,然后在链表中顺序查找。
-
+
对于 N 个键,M 条链表 (N>M),如果哈希函数能够满足均匀性的条件,每条链表的大小趋向于 N/M,因此未命中的查找和插入操作所需要的比较次数为 \~N/M。
-### 4.3 基于线性探测法的散列表
+### 3. 基于线性探测法的散列表
-线性探测法使用空位来解决冲突,当冲突发生时,向前探测一个空位来存储冲突的键。使用线程探测法,数组的大小 M 应当大于键的个数 N(M>N)。
+线性探测法使用空位来解决冲突,当冲突发生时,向前探测一个空位来存储冲突的键。使用线性探测法,数组的大小 M 应当大于键的个数 N(M>N)。
-
+
```java
public class LinearProbingHashST {
@@ -1184,7 +1470,7 @@ public class LinearProbingHashST {
}
```
-#### 4.3.1 查找
+**(一)查找**
```java
public Value get(Key key) {
@@ -1197,7 +1483,7 @@ public Value get(Key key) {
}
```
-#### 4.3.2 插入
+**(二)插入**
```java
public void put(Key key, Value val) {
@@ -1215,9 +1501,9 @@ public void put(Key key, Value val) {
}
```
-#### 4.3.3 删除
+**(三)删除**
-删除操作应当将右侧所有相邻的键值重新插入散列表中。
+删除操作应当将右侧所有相邻的键值对重新插入散列表中。
```java
public void delete(Key key) {
@@ -1243,14 +1529,14 @@ public void delete(Key key) {
}
```
-#### 4.3.4 调整数组大小
+**(四)调整数组大小**
-线性探测法的成本取决于连续条目的长度,连续条目也叫聚簇。当聚簇很长时,在查找和插入时也需要进行很多次探测。
+线性探测法的成本取决于连续条目的长度,连续条目也叫聚簇。当聚簇很长时,在查找和插入时也需要进行很多次探测。例如下图中 2\~5 位置就是一个聚簇。
+
+
α = N/M,把 α 称为利用率。理论证明,当 α 小于 1/2 时探测的预计次数只在 1.5 到 2.5 之间。
-
-
为了保证散列表的性能,应当调整数组的大小,使得 α 在 [1/4, 1/2] 之间。
```java
@@ -1272,33 +1558,34 @@ private void resize(int cap) {
}
```
-虽然每次重新调整数组都需要重新把每个键值对插入到散列表,但是从摊还分析的角度来看,所需要的代价却是很小的。从下图可以看出,每次数组长度加倍后,累计平均值都会增加 1,因为表中每个键都需要重新计算散列值,但是随后平均值会下降。
+## 应用
-
+### 1. 各种符号表实现的比较
-## 5. 应用
-
-### 5.1 各种符号表实现的比较
-
-
+| 算法 | 插入 | 查找 | 是否有序 |
+| :---: | :---: | :---: | :---: |
+| 二分查找实现的有序表 | N | logN | yes |
+| 二叉查找树 | logN | logN | yes |
+| 2-3 查找树 | logN | logN | yes |
+| 拉链法实现的散列表 | N/M | N/M | no |
+| 线性探测法试下的散列表 | 1 | 1 | no |
应当优先考虑散列表,当需要有序性操作时使用红黑树。
-### 5.2 Java 的符号表实现
+### 2. Java 的符号表实现
-Java 的 java.util.TreeMap 和 java.util.HashMap 分别是基于红黑树和拉链法的散列表的符号表实现。
+- java.util.TreeMap:红黑树
+- java.util.HashMap:拉链法的散列表
-### 5.3 集合类型
+### 3. 集合类型
除了符号表,集合类型也经常使用,它只有键没有值,可以用集合类型来存储一系列的键然后判断一个键是否在集合中。
-### 5.4 稀疏向量乘法
+### 4. 稀疏向量乘法
当向量为稀疏向量时,可以使用符号表来存储向量中的非 0 索引和值,使得乘法运算只需要对那些非 0 元素进行即可。
```java
-import java.util.HashMap;
-
public class SparseVector {
private HashMap hashMap;
@@ -1324,3 +1611,8 @@ public class SparseVector {
}
}
```
+
+# 参考资料
+
+- Sedgewick, Robert, and Kevin Wayne. _Algorithms_. Addison-Wesley Professional, 2011.
+
diff --git a/notes/计算机操作系统.md b/notes/计算机操作系统.md
index 4108d7a6..e992bd63 100644
--- a/notes/计算机操作系统.md
+++ b/notes/计算机操作系统.md
@@ -1,89 +1,37 @@
-* [第一章 概述](#第一章-概述)
+* [一、 概述](#一-概述)
* [操作系统基本特征](#操作系统基本特征)
- * [1. 并发](#1-并发)
- * [2. 共享](#2-共享)
- * [3. 虚拟](#3-虚拟)
- * [4. 异步](#4-异步)
+ * [操作系统基本功能](#操作系统基本功能)
* [系统调用](#系统调用)
- * [中断分类](#中断分类)
- * [1. 外中断](#1-外中断)
- * [2. 异常](#2-异常)
- * [3. 陷入](#3-陷入)
* [大内核和微内核](#大内核和微内核)
- * [1. 大内核](#1-大内核)
- * [2. 微内核](#2-微内核)
-* [第二章 进程管理](#第二章-进程管理)
+ * [中断分类](#中断分类)
+* [二、进程管理](#二进程管理)
* [进程与线程](#进程与线程)
- * [1. 进程](#1-进程)
- * [2. 线程](#2-线程)
- * [3. 区别](#3-区别)
* [进程状态的切换](#进程状态的切换)
* [调度算法](#调度算法)
- * [1. 批处理系统中的调度](#1-批处理系统中的调度)
- * [1.1 先来先服务](#11-先来先服务)
- * [1.2 短作业优先](#12-短作业优先)
- * [1.3 最短剩余时间优先](#13-最短剩余时间优先)
- * [2. 交互式系统中的调度](#2-交互式系统中的调度)
- * [2.1 优先权优先](#21-优先权优先)
- * [2.2 时间片轮转](#22-时间片轮转)
- * [2.3 多级反馈队列](#23-多级反馈队列)
- * [2.4 短进程优先](#24-短进程优先)
- * [3. 实时系统中的调度](#3-实时系统中的调度)
* [进程同步](#进程同步)
- * [1. 临界区](#1-临界区)
- * [2. 同步与互斥](#2-同步与互斥)
- * [3. 信号量](#3-信号量)
- * [4. 管程](#4-管程)
- * [进程通信](#进程通信)
- * [1. 管道](#1-管道)
- * [2. 信号量](#2-信号量)
- * [3. 消息队列](#3-消息队列)
- * [4. 信号](#4-信号)
- * [5. 共享内存](#5-共享内存)
- * [6. 套接字](#6-套接字)
* [经典同步问题](#经典同步问题)
- * [1. 读者-写者问题](#1-读者-写者问题)
- * [2. 哲学家进餐问题](#2-哲学家进餐问题)
-* [第三章 死锁](#第三章-死锁)
+ * [进程通信](#进程通信)
+* [三、死锁](#三死锁)
* [死锁的必要条件](#死锁的必要条件)
* [死锁的处理方法](#死锁的处理方法)
- * [1. 鸵鸟策略](#1-鸵鸟策略)
- * [2. 死锁预防](#2-死锁预防)
- * [2.1 破坏互斥条件](#21-破坏互斥条件)
- * [2.2 破坏占有和等待条件](#22-破坏占有和等待条件)
- * [2.3 破坏不可抢占条件](#23-破坏不可抢占条件)
- * [2.4 破坏环路等待](#24-破坏环路等待)
- * [3. 死锁避免](#3-死锁避免)
- * [3.1 安全状态](#31-安全状态)
- * [3.2 单个资源的银行家算法](#32-单个资源的银行家算法)
- * [3.3 多个资源的银行家算法](#33-多个资源的银行家算法)
- * [4. 死锁检测与死锁恢复](#4-死锁检测与死锁恢复)
- * [4.1 死锁检测算法](#41-死锁检测算法)
- * [4.2 死锁恢复](#42-死锁恢复)
-* [第四章 存储器管理](#第四章-存储器管理)
+* [四、内存管理](#四内存管理)
* [虚拟内存](#虚拟内存)
* [分页与分段](#分页与分段)
- * [1. 分页](#1-分页)
- * [2. 分段](#2-分段)
- * [3. 段页式](#3-段页式)
- * [4. 分页与分段区别](#4-分页与分段区别)
+ * [分页系统地址映射](#分页系统地址映射)
* [页面置换算法](#页面置换算法)
- * [1. 最佳(Optimal)](#1-最佳optimal)
- * [2. 先进先出(FIFO)](#2-先进先出fifo)
- * [3. 最近最久未使用(LRU, Least Recently Used)](#3-最近最久未使用lru,-least-recently-used)
- * [4. 时钟(Clock)](#4-时钟clock)
-* [第五章 设备管理](#第五章-设备管理)
+* [五、设备管理](#五设备管理)
* [磁盘调度算法](#磁盘调度算法)
- * [1. 先来先服务(FCFS, First Come First Serverd)](#1-先来先服务fcfs,-first-come-first-serverd)
- * [2. 最短寻道时间优先(SSTF, Shortest Seek Time First)](#2-最短寻道时间优先sstf,-shortest-seek-time-first)
- * [3. 扫描算法(SCAN)](#3-扫描算法scan)
- * [4. 循环扫描算法(CSCAN)](#4-循环扫描算法cscan)
+* [六、链接](#六链接)
+ * [编译系统](#编译系统)
+ * [目标文件](#目标文件)
+ * [静态链接](#静态链接)
+ * [动态链接](#动态链接)
* [参考资料](#参考资料)
-# 第一章 概述
+# 一、 概述
## 操作系统基本特征
@@ -111,13 +59,60 @@
### 4. 异步
-异步是指进程不是一次性执行完毕,而是走走停停,以不可知的速度向前推进。
+异步指进程不是一次性执行完毕,而是走走停停,以不可知的速度向前推进。
+
+## 操作系统基本功能
+
+### 1. 进程管理
+
+进程控制、进程同步、进程通信、死锁处理、处理机调度等。
+
+### 2. 内存管理
+
+内存分配、地址映射、内存保护与共享和内存扩充等功能。
+
+### 3. 文件管理
+
+文件存储空间的管理、目录管理及文件读写管理和保护等。
+
+### 4. 设备管理
+
+完成用户的 I/O 请求,方便用户使用各种设备,并提高设备的利用率,主要包括缓冲管理、设备分配、设备处理和虛拟设备等功能。
## 系统调用
-如果一个进程在用户态需要用到操作系统的一些功能,就需要使用系统调用从而陷入内核,由操作系统代为完成。
+如果一个进程在用户态需要使用内核态的功能,就进行系统调用从而陷入内核,由操作系统代为完成。
-可以由系统调用请求的功能有设备管理、文件管理、进程管理、进程通信、存储器管理等。
+
+
+Linux 的系统调用主要有以下这些:
+
+| Task | Commands |
+| :---: | --- |
+| 进程控制 | fork(); exit(); wait(); |
+| 进程通信 | pipe(); shmget(); mmap(); |
+| 文件操作 | open(); read(); write(); |
+| 设备操作 | ioctl(); read(); write(); |
+| 信息维护 | getpid(); alarm(); sleep(); |
+| 安全 | chmod(); umask(); chown(); |
+
+## 大内核和微内核
+
+### 1. 大内核
+
+大内核是将操作系统功能作为一个紧密结合的整体放到内核。
+
+由于各模块共享信息,因此有很高的性能。
+
+### 2. 微内核
+
+由于操作系统不断复杂,因此将一部分操作系统功能移出内核,从而降低内核的复杂性。移出的部分根据分层的原则划分成若干服务,相互独立。
+
+在微内核结构下,操作系统被划分成小的、定义良好的模块,只有微内核这一个模块运行在内核态,其余模块运行在用户态。
+
+因为需要频繁地在用户态和核心态之间进行切换,所以会有一定的性能损失。
+
+
## 中断分类
@@ -133,35 +128,27 @@
在用户程序中使用系统调用。
-## 大内核和微内核
-
-### 1. 大内核
-
-大内核是将操作系统功能作为一个紧密结合的整体放到内核。
-
-由于各模块共享信息,因此有很高的性能。
-
-### 2. 微内核
-
-由于操作系统不断复杂,因此将一部分操作系统功能移出内核,从而降低内核的复杂性。移出的部分根据分层的原则划分成若干服务,相互独立。
-
-因为需要频繁地在用户态和核心态之间进行切换,所以会有一定的性能损失。
-
-# 第二章 进程管理
+# 二、进程管理
## 进程与线程
### 1. 进程
-进程是操作系统进行资源分配的基本单位。
+进程是资源分配的基本单位。
进程控制块 (Process Control Block, PCB) 描述进程的基本信息和运行状态,所谓的创建进程和撤销进程,都是指对 PCB 的操作。
+下图显示了 4 个程序创建了 4 个进程,这 4 个进程可以并发地执行。
+
+
+
### 2. 线程
-一个进程中可以有多个线程,线程是独立调度的基本单位。
+线程是独立调度的基本单位。
-同一个进程中的多个线程之间可以并发执行,它们共享进程资源。
+一个进程中可以有多个线程,它们共享进程资源。
+
+
### 3. 区别
@@ -177,11 +164,16 @@
## 进程状态的切换
-
+
-阻塞状态是缺少需要的资源从而由运行状态转换而来,但是该资源不包括 CPU,缺少 CPU 会让进程从运行态转换为就绪态。
+- 就绪状态(ready):等待被调度
+- 运行状态(running)
+- 阻塞状态(waiting):等待资源
-只有就绪态和运行态可以相互转换,其它的都是单向转换。就绪状态的进程通过调度算法从而获得 CPU 时间,转为运行状态;而运行状态的进程,在分配给它的 CPU 时间片用完之后就会转为就绪状态,等待下一次调度。
+应该注意以下内容:
+
+- 只有就绪态和运行态可以相互转换,其它的都是单向转换。就绪状态的进程通过调度算法从而获得 CPU 时间,转为运行状态;而运行状态的进程,在分配给它的 CPU 时间片用完之后就会转为就绪状态,等待下一次调度。
+- 阻塞状态是缺少需要的资源从而由运行状态转换而来,但是该资源不包括 CPU 时间,缺少 CPU 时间会从运行态转换为就绪态。
## 调度算法
@@ -191,7 +183,7 @@
#### 1.1 先来先服务
-first-come first-serverd(FCFS)。
+> first-come first-serverd(FCFS)
调度最先进入就绪队列的作业。
@@ -199,45 +191,39 @@ first-come first-serverd(FCFS)。
#### 1.2 短作业优先
-shortest job first(SJF)。
+> shortest job first(SJF)
调度估计运行时间最短的作业。
-长作业有可能会饿死,处于一直等待短作业执行完毕的状态。如果一直有短作业到来,那么长作业永远得不到调度。
+长作业有可能会饿死,处于一直等待短作业执行完毕的状态。因为如果一直有短作业到来,那么长作业永远得不到调度。
#### 1.3 最短剩余时间优先
-shortest remaining time next(SRTN)。
+> shortest remaining time next(SRTN)
### 2. 交互式系统中的调度
-#### 2.1 优先权优先
+#### 2.1 优先级调度
除了可以手动赋予优先权之外,还可以把响应比作为优先权,这种调度方式叫做高响应比优先调度算法。
响应比 = (等待时间 + 要求服务时间) / 要求服务时间 = 响应时间 / 要求服务时间
-这种调度算法主要是为了解决 SJF 中长作业可能会饿死的问题,因为随着等待时间的增长,响应比也会越来越高。
+这种调度算法主要是为了解决短作业优先调度算法长作业可能会饿死的问题,因为随着等待时间的增长,响应比也会越来越高。
#### 2.2 时间片轮转
-将所有就绪进程按 FCFS 的原则排成一个队列,每次调度时,把 CPU 分配给队首进程,该进程可以执行一个时间片。当时间片用完时,由计时器发出时钟中断,调度程序便停止该进程的执行,并将它送往就绪队列的末尾,同时继续把 CPU 分配给队首的进程。
+将所有就绪进程按 FCFS 的原则排成一个队列,每次调度时,把 CPU 时间分配给队首进程,该进程可以执行一个时间片。当时间片用完时,由计时器发出时钟中断,调度程序便停止该进程的执行,并将它送往就绪队列的末尾,同时继续把 CPU 时间分配给队首的进程。
-时间片轮转算法的效率和时间片的大小有很大关系。因为每次进程切换都要保存进程的信息并且载入新进程的信息,如果时间片太小,进程切换太频繁,在进程切换上就会花过多时间。
+时间片轮转算法的效率和时间片的大小有很大关系。因为进程切换都要保存进程的信息并且载入新进程的信息,如果时间片太小,会导致进程切换得太频繁,在进程切换上就会花过多时间。
#### 2.3 多级反馈队列
-1. 设置多个就绪队列,并为各个队列赋予不同的优先级。第一个队列的优先级最高,第二个队列次之,其余各队列的优先权逐个降低。该算法赋予各个队列中进程执行时间片的大小也各不相同,在优先权越高的队列中,为每个进程所规定的执行时间片就越小。
+如果一个进程需要执行 100 个时间片,如果采用轮转调度算法,那么需要交换 100 次。多级队列是为这种需要连续执行多个时间片的进程考虑,它设置了多个队列,每个队列时间片大小都不同,例如 1,2,4,8,..。进程在第一个队列没执行完,就会被移到下一个队列。这种方式下,之前的进程只需要交换 7 次。
-2. 当一个新进程进入内存后,首先将它放入第一队列的末尾,按 FCFS 原则排队等待调度。当轮到该进程执行时,如它能在该时间片内完成,便可准备撤离系统;如果它在一个时间片结束时尚未完成,调度程序便将该进程转入下一个队列的队尾。
-
-3. 仅当前 i-1 个队列均空时,才会调度第 i 个队列中的进程。
-
-优点:实时性好,同时适合运行短作业和长作业。
-
-#### 2.4 短进程优先
+每个队列优先权也不同,最上面的优先权最高。因此只有上一个队列没有进程在排队,才能调度当前队列上的进程。
### 3. 实时系统中的调度
@@ -261,15 +247,15 @@ shortest remaining time next(SRTN)。
### 2. 同步与互斥
-- 同步指多个进程按一定顺序执行;
-- 互斥指多个进程在同一时刻只有一个进程能进入临界区。
+- 同步:多个进程按一定顺序执行;
+- 互斥:多个进程在同一时刻只有一个进程能进入临界区。
### 3. 信号量
-**信号量(Semaphore)** 是一个整型变量,可以对其执行 down 和 up 操作,也就是常见的 P 和 V 操作。
+信号量(Semaphore)是一个整型变量,可以对其执行 down 和 up 操作,也就是常见的 P 和 V 操作。
-- **down** : 如果信号量大于 0 ,执行 -1 操作;如果信号量等于 0,将进程睡眠,等待信号量大于 0;
-- **up** :对信号量执行 +1 操作,并且唤醒睡眠的进程,让进程完成 down 操作。
+- **down** : 如果信号量大于 0 ,执行 -1 操作;如果信号量等于 0,进程睡眠,等待信号量大于 0;
+- **up** :对信号量执行 +1 操作,唤醒睡眠的进程让其完成 down 操作。
down 和 up 操作需要被设计成原语,不可分割,通常的做法是在执行这些操作的时候屏蔽中断。
@@ -279,27 +265,27 @@ down 和 up 操作需要被设计成原语,不可分割,通常的做法是
typedef int semaphore;
semaphore mutex = 1;
void P1() {
- down(mutex);
+ down(&mutex);
// 临界区
- up(mutex);
+ up(&mutex);
}
void P2() {
- down(mutex);
+ down(&mutex);
// 临界区
- up(mutex);
+ up(&mutex);
}
```
-**使用信号量实现生产者-消费者问题**
+ **使用信号量实现生产者-消费者问题**
-使用一个缓冲区来保存物品,只有缓冲区没有满,生产者才可以放入物品;只有缓冲区不为空,消费者才可以拿走物品。
+问题描述:使用一个缓冲区来保存物品,只有缓冲区没有满,生产者才可以放入物品;只有缓冲区不为空,消费者才可以拿走物品。
-需要使用一个互斥量 mutex 来对缓冲区这个临界资源进行互斥访问。
+因为缓冲区属于临界资源,因此需要使用一个互斥量 mutex 来控制对缓冲区的互斥访问。
-为了控制生产者和消费者的行为,需要记录缓冲区中物品的数量。数量可以使用信号量来进行统计,这里需要使用两个信号量:empty 记录空缓冲区的数量,full 记录满缓冲区的数量。其中,empty 信号量是在生产者进程中使用,当 empty 不为 0 时,生产者才可以放入物品;full 信号量是在消费者进行中使用,当 full 信号量不为 0 时,消费者才可以取走物品。
+为了同步生产者和消费者的行为,需要记录缓冲区中物品的数量。数量可以使用信号量来进行统计,这里需要使用两个信号量:empty 记录空缓冲区的数量,full 记录满缓冲区的数量。其中,empty 信号量是在生产者进程中使用,当 empty 不为 0 时,生产者才可以放入物品;full 信号量是在消费者进程中使用,当 full 信号量不为 0 时,消费者才可以取走物品。
-注意,不能先对缓冲区进行加锁,再测试信号量。也就是说,不能先执行 down(mutex) 再执行 down(empty)。如果这么做了,那么可能会出现这种情况:生产者对缓冲区加锁后,执行 down(empty) 操作,发现 empty=0,此时生成者睡眠。消费者此时不能进入临界区,因为生产者对缓冲区加锁了,也就无法执行 up(empty) 操作,那么生产者和消费者就会一直等待下去。
+注意,不能先对缓冲区进行加锁,再测试信号量。也就是说,不能先执行 down(mutex) 再执行 down(empty)。如果这么做了,那么可能会出现这种情况:生产者对缓冲区加锁后,执行 down(empty) 操作,发现 empty = 0,此时生产者睡眠。消费者不能进入临界区,因为生产者对缓冲区加锁了,也就无法执行 up(empty) 操作,empty 永远都为 0,那么生产者和消费者就会一直等待下去,造成死锁。
```c
#define N 100
@@ -310,22 +296,22 @@ semaphore full = 0;
void producer() {
while(TRUE){
- int item = produce_item;
- down(empty);
- down(mutex);
+ int item = produce_item();
+ down(&empty);
+ down(&mutex);
insert_item(item);
- up(mutex);
- up(full);
+ up(&mutex);
+ up(&full);
}
}
void consumer() {
while(TRUE){
- down(full);
- down(mutex);
- int item = remove_item(item);
- up(mutex);
- up(empty);
+ down(&full);
+ down(&mutex);
+ int item = remove_item();
+ up(&mutex);
+ up(&empty);
consume_item(item);
}
}
@@ -335,7 +321,7 @@ void consumer() {
使用信号量机制实现的生产者消费者问题需要客户端代码做很多控制,而管程把控制的代码独立出来,不仅不容易出错,也使得客户端代码调用更容易。
-c 语言不支持管程,下面的示例代码使用了类 Pascal 语言来描述管程。示例代码中的管程提供了 insert() 和 remove() 方法,客户端代码通过调用这两个方法来解决生产者-消费者问题。
+c 语言不支持管程,下面的示例代码使用了类 Pascal 语言来描述管程。示例代码的管程提供了 insert() 和 remove() 方法,客户端代码通过调用这两个方法来解决生产者-消费者问题。
```pascal
monitor ProducerConsumer
@@ -356,11 +342,12 @@ end monitor;
管程有一个重要特性:在一个时刻只能有一个进程使用管程。进程在无法继续执行的时候不能一直占用管程,否者其它进程永远不能使用管程。
-管程引入了 **条件变量** 以及相关的操作:**wait()** 和 **signal()** 来实现同步操作。对条件变量执行 wait() 操作会导致调用进程阻塞,把管程让出来让另一个进程持有。signal() 操作用于唤醒被阻塞的进程。
+管程引入了 **条件变量** 以及相关的操作:**wait()** 和 **signal()** 来实现同步操作。对条件变量执行 wait() 操作会导致调用进程阻塞,把管程让出来给另一个进程持有。signal() 操作用于唤醒被阻塞的进程。
-**使用管程实现生成者-消费者问题**
+ **使用管程实现生成者-消费者问题**
```pascal
+// 管程
monitor ProducerConsumer
condition full, empty;
integer count := 0;
@@ -383,6 +370,7 @@ monitor ProducerConsumer
end;
end monitor;
+// 生产者客户端
procedure producer
begin
while true do
@@ -392,6 +380,7 @@ begin
end
end;
+// 消费者客户端
procedure consumer
begin
while true do
@@ -402,46 +391,6 @@ begin
end;
```
-## 进程通信
-
-进程通信可以看成是不同进程间的线程通信,对于同一个进程内线程的通信方式,主要使用信号量、条件变量等同步机制。
-
-### 1. 管道
-
-管道是单向的、先进先出的、无结构的、固定大小的字节流,它把一个进程的标准输出和另一个进程的标准输入连接在一起。写进程在管道的尾端写入数据,读进程在管道的首端读出数据。数据读出后将从管道中移走,其它读进程都不能再读到这些数据。
-
-管道提供了简单的流控制机制,进程试图读空管道时,在有数据写入管道前,进程将一直阻塞。同样地,管道已经满时,进程再试图写管道,在其它进程从管道中移走数据之前,写进程将一直阻塞。
-
-Linux 中管道是通过空文件来实现。
-
-管道有三种:
-
-1. 普通管道:有两个限制,一是只能单向传输;二是只能在父子进程之间使用;
-2. 流管道:去除第一个限制,支持双向传输;
-3. 命名管道:去除第二个限制,可以在不相关进程之间进行通信。
-
-### 2. 信号量
-
-信号量是一个计数器,可以用来控制多个进程对共享资源的访问。它常作为一种锁机制,防止某进程正在访问共享资源时,其它进程也访问该资源。主要作为进程间以及同一进程内不同线程之间的同步手段。
-
-### 3. 消息队列
-
-消息队列克服了信号传递信息少、管道只能承载无格式字节流以及缓冲区大小受限等缺点。
-
-### 4. 信号
-
-信号是一种比较复杂的通信方式,用于通知接收进程某个事件已经发生。
-
-### 5. 共享内存
-
-共享内存就是映射一段能被其它进程所访问的内存,这段共享内存由一个进程创建,但多个进程都可以访问。
-
-共享内存是最快的 IPC 方式,它是针对其它 IPC 运行效率低而专门设计的。它往往与其它通信机制(如信号量)配合使用,来实现进程间的同步和通信。
-
-### 6. 套接字
-
-套接字也是一种进程间通信机制,与其它通信机制不同的是,它可用于不同机器间的进程通信。
-
## 经典同步问题
生产者和消费者问题前面已经讨论过了。
@@ -460,23 +409,23 @@ int count = 0;
void reader() {
while(TRUE) {
- down(count_mutex);
+ down(&count_mutex);
count++;
- if(count == 1) down(data_mutex); // 第一个读者需要对数据进行加锁,防止写进程访问
- up(count_mutex);
+ if(count == 1) down(&data_mutex); // 第一个读者需要对数据进行加锁,防止写进程访问
+ up(&count_mutex);
read();
- down(count_mutex);
+ down(&count_mutex);
count--;
- if(count == 0) up(data_mutex);
- up(count_mutex);
+ if(count == 0) up(&data_mutex);
+ up(&count_mutex);
}
}
void writer() {
while(TRUE) {
- down(data_mutex);
+ down(&data_mutex);
write();
- up(data_mutex);
+ up(&data_mutex);
}
}
```
@@ -485,51 +434,129 @@ void writer() {
-五个哲学家围着一张圆周,每个哲学家面前放着饭。哲学家的生活有两种交替活动:吃饭以及思考。当一个哲学家吃饭时,需要先一根一根拿起左右两边的筷子。
+五个哲学家围着一张圆桌,每个哲学家面前放着食物。哲学家的生活有两种交替活动:吃饭以及思考。当一个哲学家吃饭时,需要先拿起自己左右两边的两根筷子,并且一次只能拿起一根筷子。
-下面是一种错误的解法,考虑到如果每个哲学家同时拿起左手边的筷子,那么就无法拿起右手边的筷子,造成死锁。
+下面是一种错误的解法,考虑到如果所有哲学家同时拿起左手边的筷子,那么就无法拿起右手边的筷子,造成死锁。
```c
#define N 5
-#define LEFT (i + N - 1) % N
-#define RIGHT (i + N) % N
-typedef int semaphore;
-semaphore chopstick[N];
void philosopher(int i) {
- while(TURE){
+ while(TRUE) {
think();
- down(chopstick[LEFT[i]]);
- down(chopstick[RIGHT[i]]);
+ take(i); // 拿起左边的筷子
+ take((i+1)%N); // 拿起右边的筷子
eat();
- up(chopstick[RIGHT[i]]);
- up(chopstick[LEFT[i]]);
+ put(i);
+ put((i+1)%N);
}
}
```
-为了防止死锁的发生,可以加一点限制,只允许同时拿起左右两边的筷子。方法是引入一个互斥量,对拿起两个筷子的那段代码加锁。
+为了防止死锁的发生,可以设置两个条件:
+
+1. 必须同时拿起左右两根筷子;
+2. 只有在两个邻居都没有进餐的情况下才允许进餐。
```c
-semaphore mutex = 1;
+#define N 5
+#define LEFT (i + N - 1) % N // 左邻居
+#define RIGHT (i + 1) % N // 右邻居
+#define THINKING 0
+#define HUNGRY 1
+#define EATING 2
+typedef int semaphore;
+int state[N]; // 跟踪每个哲学家的状态
+semaphore mutex = 1; // 临界区的互斥
+semaphore s[N]; // 每个哲学家一个信号量
void philosopher(int i) {
- while(TURE){
+ while(TRUE) {
think();
- down(mutex);
- down(chopstick[LEFT[i]]);
- down(chopstick[RIGHT[i]]);
- up(mutex);
+ take_two(i);
eat();
- down(mutex);
- up(chopstick[RIGHT[i]]);
- up(chopstick[LEFT[i]]);
- up(mutex);
+ put_tow(i);
+ }
+}
+
+void take_two(int i) {
+ down(&mutex);
+ state[i] = HUNGRY;
+ test(i);
+ up(&mutex);
+ down(&s[i]);
+}
+
+void put_tow(i) {
+ down(&mutex);
+ state[i] = THINKING;
+ test(LEFT);
+ test(RIGHT);
+ up(&mutex);
+}
+
+void test(i) { // 尝试拿起两把筷子
+ if(state[i] == HUNGRY && state[LEFT] != EATING && state[RIGHT] !=EATING) {
+ state[i] = EATING;
+ up(&s[i]);
}
}
```
-# 第三章 死锁
+## 进程通信
+
+### 1. 进程同步与进程通信的区别
+
+- 进程同步:控制多个进程按一定顺序执行;
+- 进程通信:进程间传输信息。
+
+进程通信是一种手段,而进程同步是一种目的。也可以说,为了能够达到进程同步的目的,需要让进程进行通信,传输一些进程同步所需要的信息。
+
+在进程同步中介绍的信号量也属于进程通信的一种方式,但是属于低级别的进程通信,因为它传输的信息非常小。
+
+### 2. 进程通信方式
+
+#### 2.1 消息传递
+
+操作系统提供了用于通信的通道(Channel),进程可以通过读写这个通道进行通信。
+
+
+
+ **(一)管道**
+
+写进程在管道的尾端写入数据,读进程在管道的首端读出数据。管道提供了简单的流控制机制,进程试图读空管道时,在有数据写入管道前,进程将一直阻塞。同样地,管道已经满时,进程再试图写管道,在其它进程从管道中移走数据之前,写进程将一直阻塞。
+
+Linux 中管道通过空文件实现。
+
+管道有三种:
+
+1. 普通管道:有两个限制,一是只能单向传输;二是只能在父子进程之间使用;
+2. 流管道:去除第一个限制,支持双向传输;
+3. 命名管道:去除第二个限制,可以在不相关进程之间进行通信。
+
+
+
+ **(二)消息队列**
+
+消息队列克服了信号传递信息少、管道只能承载无格式字节流以及缓冲区大小受限等缺点。
+
+
+
+ **(三)套接字**
+
+套接字也是一种进程间通信机制,与其它通信机制不同的是,它可用于不同机器间的进程通信。
+
+
+
+#### 2.2 共享内存
+
+操作系统建立一块共享内存,并将其映射到每个进程的地址空间上,进程就可以直接对这块共享内存进行读写。
+
+共享内存是最快的进程通信方式。
+
+
+
+# 三、死锁
## 死锁的必要条件
@@ -537,7 +564,7 @@ void philosopher(int i) {
1. 互斥:每个资源要么已经分配给了一个进程,要么就是可用的。
2. 占有和等待:已经得到了某个资源的进程可以再请求新的资源。
-3. 不可抢占:已经分配给一个进程的资源不能强制性地被抢占,它只能被占有它的进程显示地释放。
+3. 不可抢占:已经分配给一个进程的资源不能强制性地被抢占,它只能被占有它的进程显式地释放。
4. 环路等待:有两个或者两个以上的进程组成一条环路,该环路中的每个进程都在等待下一个进程所占有的资源。
## 死锁的处理方法
@@ -546,39 +573,84 @@ void philosopher(int i) {
把头埋在沙子里,假装根本没发生问题。
-这种策略不可取。
+因为解决死锁问题的代价很高,因此鸵鸟策略这种不采取任务措施的方案会获得更高的性能。当发生死锁时不会对用户造成多大影响,或发生死锁的概率很低,可以采用鸵鸟策略。
-### 2. 死锁预防
+大多数操作系统,包括 Unix,Linux 和 Windows,处理死锁问题的办法仅仅是忽略它。
+
+### 2. 死锁检测与死锁恢复
+
+不试图阻止死锁,而是当检测到死锁发生时,采取措施进行恢复。
+
+**(一)每种类型一个资源的死锁检测**
+
+
+
+上图为资源分配图,其中方框表示资源,圆圈表示进程。资源指向进程表示该资源已经分配给该进程,进程指向资源表示进程请求获取该资源。
+
+图 a 可以抽取出环,如图 b,它满足了环路等待条件,因此会发生死锁。
+
+每种类型一个资源的死锁检测算法是通过检测有向图是否存在环来实现,从一个节点出发进行深度优先搜索,对访问过的节点进行标记,如果访问了已经标记的节点,就表示有向图存在环,也就是检测到死锁的发生。
+
+**(二)每种类型多个资源的死锁检测**
+
+
+
+上图中,有三个进程四个资源,每个数据代表的含义如下:
+
+- E 向量:资源总量
+- A 向量:资源剩余量
+- C 矩阵:每个进程所拥有的资源数量,每一行都代表一个进程拥有资源的数量
+- R 矩阵:每个进程请求的资源数量
+
+进程 P1 和 P2 所请求的资源都得不到满足,只有进程 P3 可以,让 P3 执行,之后释放 P3 拥有的资源,此时 A = (2 2 2 0)。P2 可以执行,执行后释放 P2 拥有的资源,A = (4 2 2 1) 。P1 也可以执行。所有进程都可以顺利执行,没有死锁。
+
+算法总结如下:
+
+每个进程最开始时都不被标记,执行过程有可能被标记。当算法结束时,任何没有被标记的进程都是死锁进程。
+
+1. 寻找一个没有标记的进程 Pi ,它所请求的资源小于等于 A。
+2. 如果找到了这样一个进程,那么将 C 矩阵的第 i 行向量加到 A 中,标记该进程,并转回 1。
+3. 如果没有这样一个进程,算法终止。
+
+**(三)死锁恢复**
+
+- 利用抢占恢复
+- 利用回滚恢复
+- 通过杀死进程恢复
+
+### 3. 死锁预防
在程序运行之前预防发生死锁。
-#### 2.1 破坏互斥条件
+**(一)破坏互斥条件**
例如假脱机打印机技术允许若干个进程同时输出,唯一真正请求物理打印机的进程是打印机守护进程。
-#### 2.2 破坏占有和等待条件
+**(二)破坏占有和等待条件**
一种实现方式是规定所有进程在开始执行前请求所需要的全部资源。
-#### 2.3 破坏不可抢占条件
+**(三)破坏不可抢占条件**
-#### 2.4 破坏环路等待
+**(四)破坏环路等待**
给资源统一编号,进程只能按编号顺序来请求资源。
-### 3. 死锁避免
+### 4. 死锁避免
在程序运行时避免发生死锁。
-#### 3.1 安全状态
+**(一)安全状态**
-图 a 的第二列 has 表示已拥有的资源数,第三列 max 表示总共需要的资源数,free 表示还有可以使用的资源数。从图 a 开始出发,先让 B 拥有所需的所有资源,运行结束后释放 B,此时 free 变为 4;接着以同样的方式运行 C 和 A,使得所有进程都能成功运行,因此可以称图 a 所示的状态时安全的。
+图 a 的第二列 Has 表示已拥有的资源数,第三列 Max 表示总共需要的资源数,Free 表示还有可以使用的资源数。从图 a 开始出发,先让 B 拥有所需的所有资源(图 b),运行结束后释放 B,此时 Free 变为 5(图 c);接着以同样的方式运行 C 和 A,使得所有进程都能成功运行,因此可以称图 a 所示的状态时安全的。
定义:如果没有死锁发生,并且即使所有进程突然请求对资源的最大需求,也仍然存在某种调度次序能够使得每一个进程运行完毕,则称该状态是安全的。
-#### 3.2 单个资源的银行家算法
+安全状态的检测与死锁的检测类似,因为安全状态必须要求不能发生死锁。下面的银行家算法与死锁检测算法非常类似,可以结合着做参考对比。
+
+**(二)单个资源的银行家算法**
一个小城镇的银行家,他向一群客户分别承诺了一定的贷款额度,算法要做的是判断对请求的满足是否会进入不安全状态,如果是,就拒绝请求;否则予以分配。
@@ -586,7 +658,7 @@ void philosopher(int i) {
上图 c 为不安全状态,因此算法会拒绝之前的请求,从而避免进入图 c 中的状态。
-#### 3.3 多个资源的银行家算法
+**(三)多个资源的银行家算法**
@@ -598,59 +670,31 @@ void philosopher(int i) {
- 假若找到这样一行,将该进程标记为终止,并将其已分配资源加到 A 中。
- 重复以上两步,直到所有进程都标记为终止,则状态时安全的。
-### 4. 死锁检测与死锁恢复
+如果一个状态不是安全的,也需要拒绝进入这个状态。
-不试图阻止死锁,而是当检测到死锁发生时,采取措施进行恢复。
-
-#### 4.1 死锁检测算法
-
-死锁检测的基本思想是,如果一个进程所请求的资源能够被满足,那么就让它执行,释放它拥有的所有资源,然后让其它能满足条件的进程执行。
-
-
-
-上图中,有三个进程四个资源,每个数据代表的含义如下:
-
-- E 向量:资源总量
-- A 向量:资源剩余量
-- C 矩阵:每个进程所拥有的资源数量,每一行都代表一个进程拥有资源的数量
-- R 矩阵:每个进程请求的资源数量
-
-进程 P1 和 P2 所请求的资源都得不到满足,只有进程 P3 可以,让 P3 执行,之后释放 P3 拥有的资源,此时 A = (2 2 2 0)。P1 可以执行,执行后释放 P1 拥有的资源,A = (4 2 2 2) ,P2 也可以执行。所有进程都可以顺利执行,没有死锁。
-
-算法总结如下:
-
-每个进程最开始时都不被标记,执行过程有可能被标记。当算法结束时,任何没有被标记的进程都是死锁进程。
-
-1. 寻找一个没有标记的进程 Pi ,它所请求的资源小于等于 A。
-2. 如果找到了这样一个进程,那么将 C 矩阵的第 i 行向量加到 A 中,标记该进程,并转回 1。
-3. 如果有没有这样一个进程,算法终止。
-
-可以看到,死锁检测和银行家算法中判断是否为安全状态的方法类似。
-
-#### 4.2 死锁恢复
-
-- 利用抢占恢复
-- 杀死进程
-
-# 第四章 存储器管理
+# 四、内存管理
## 虚拟内存
-每个程序拥有自己的地址空间,这个地址空间被分割成多个块,每一块称为一页。这些页被映射到物理内存,但不需要映射到连续的物理内存,也不需要所有页都必须在物理内存中。
-
-当程序引用到一部分在物理内存中的地址空间时,由硬件执行必要的映射,将缺失的部分装入物理内存并重新执行失败的指令。
+每个程序拥有自己的地址空间,这个地址空间被分割成多个块,每一块称为一页。这些页被映射到物理内存,但不需要映射到连续的物理内存,也不需要所有页都必须在物理内存中。当程序引用到一部分不在物理内存中的地址空间时,由硬件执行必要的映射,将缺失的部分装入物理内存并重新执行失败的指令。
## 分页与分段
### 1. 分页
-用户程序的地址空间被划分为若干固定大小的区域,称为“页”。相应地,内存空间分成若干个物理块,页和块的大小相等。可将用户程序的任一页放在内存的任一块中,实现了离散分配,由一个页表来维护它们之间的映射关系。
+大部分虚拟内存系统都使用分页技术。把由程序产生的地址称为虚拟地址,它们构成了一个虚拟地址空间。例如有一台计算机可以产生 16 位地址,它的虚拟地址空间为 0\~64K,然而计算机只有 32KB 的物理内存,因此虽然可以编写 64KB 的程序,但它们不能被完全调入内存运行。
+
+
+
+虚拟地址空间划分成固定大小的页,在物理内存中对应的单元称为页框,页和页框大小通常相同,它们之间通过页表进行映射。
+
+程序最开始只将一部分页调入页框中,当程序引用到没有在页框的页时,产生缺页中断,进行页面置换,按一定的原则将一部分页框换出,并将页调入。
### 2. 分段
-上图为一个编译器在编译过程中建立的多个表,有 4 个表是动态增长的,如果使用分页系统的一维地址空间,动态递增的特点会导致覆盖问题的出现。
+上图为一个编译器在编译过程中建立的多个表,有 4 个表是动态增长的,如果使用分页系统的一维地址空间,动态增长的特点会导致覆盖问题的出现。
@@ -676,59 +720,90 @@ void philosopher(int i) {
- 出现的原因:分页主要用于实现虚拟内存,从而获得更大的地址空间;分段主要是为了使程序和数据可以被划分为逻辑上独立的地址空间并且有助于共享和保护。
+## 分页系统地址映射
+
+- 内存管理单元(MMU):管理着虚拟地址空间和物理内存的转换。
+- 页表(Page table):页(虚拟地址空间)和页框(物理内存空间)的映射表。例如下图中,页表的第 0 个表项为 010,表示第 0 个页映射到第 2 个页框。页表项的最后一位用来标记页是否在内存中。
+
+下图的页表存放着 16 个页,这 16 个页需要用 4 个比特位来进行索引定位。因此对于虚拟地址(0010 000000000100),前 4 位是用来存储页面号,而后 12 位存储在页中的偏移量。
+
+(0010 000000000100)根据前 4 位得到页号为 2,读取表项内容为(110 1),它的前 3 为为页框号,最后 1 位表示该页在内存中。最后映射得到物理内存地址为(110 000000000100)。
+
+
+
## 页面置换算法
-在程序运行过程中,若其所要访问的页面不在内存而需要把它们调入内存,但是内存已无空闲空间时,系统必须从内存中调出一个页面到磁盘对换区中,并且将程序所需要的页面调入内存中。页面置换算法的主要目标是使页面置换频率最低(也可以说缺页率最低)。
+在程序运行过程中,如果要访问的页面不在内存中,就发生缺页中断从而将该页调入内存中。此时如果内存已无空闲空间,系统必须从内存中调出一个页面到磁盘对换区中来腾出空间。
-### 1. 最佳(Optimal)
+页面置换算法的主要目标是使页面置换频率最低(也可以说缺页率最低)。
+
+### 1. 最佳
+
+> Optimal
所选择的被换出的页面将是最长时间内不再被访问,通常可以保证获得最低的缺页率。
-是一种理论上的算法,因为无法知道一个页面多长时间会被再访问到。
+是一种理论上的算法,因为无法知道一个页面多长时间不再被访问。
举例:一个系统为某进程分配了三个物理块,并有如下页面引用序列:
-进程运行时,先将 7,0,1 三个页面装入内存。当进程要访问页面 2 时,产生缺页中断,会将页面 7 换出,因为页面 7 再次被访问的时间最长。
+开始运行时,先将 7, 0, 1 三个页面装入内存。当进程要访问页面 2 时,产生缺页中断,会将页面 7 换出,因为页面 7 再次被访问的时间最长。
-### 2. 先进先出(FIFO)
+### 2. 先进先出
+
+> FIFO, First In First Out
所选择换出的页面是最先进入的页面。
该算法会将那些经常被访问的页面也被换出,从而使缺页率升高。
-### 3. 最近最久未使用(LRU, Least Recently Used)
+### 3. 最近最久未使用
+
+> LRU, Least Recently Used
虽然无法知道将来要使用的页面情况,但是可以知道过去使用页面的情况。LRU 将最近最久未使用的页面换出。
-可以用栈来实现该算法,栈中存储页面的页面号。当进程访问一个页面时,将该页面的页面号从栈移除,并将它压入栈顶。这样,最近被访问的页面的页面号总是在栈顶,而最近最久未使用的页面的页面号总是在栈底。
+可以用栈来实现该算法,栈中存储页面的页面号。当进程访问一个页面时,将该页面的页面号从栈移除,并将它压入栈顶。这样,最近被访问的页面总是在栈顶,而最近最久未使用的页面总是在栈底。
-### 4. 时钟(Clock)
+### 4. 时钟
-Clock 页面置换算法需要用到一个访问位,当一个页面被访问时,将访问位置为 1。
+> Clock
+
+需要用到一个访问位,当一个页面被访问时,将访问位置为 1。
首先,将内存中的所有页面链接成一个循环队列,当缺页中断发生时,检查当前指针所指向页面的访问位,如果访问位为 0,就将该页面换出;否则将该页的访问位设置为 0,给该页面第二次的机会,移动指针继续检查。
-# 第五章 设备管理
+
+
+# 五、设备管理
## 磁盘调度算法
-当多个进程同时请求访问磁盘时,需要进行磁盘调度来控制对磁盘的访问。磁盘调度的主要目标是使磁盘的平均寻道时间最少。
+当多个进程同时请求访问磁盘时,需要进行磁盘调度来控制对磁盘的访问。
-### 1. 先来先服务(FCFS, First Come First Serverd)
+磁盘调度的主要目标是使磁盘的平均寻道时间最少。
+
+### 1. 先来先服务
+
+> FCFS, First Come First Served
根据进程请求访问磁盘的先后次序来进行调度。优点是公平和简单,缺点也很明显,因为未对寻道做任何优化,使平均寻道时间可能较长。
-### 2. 最短寻道时间优先(SSTF, Shortest Seek Time First)
+### 2. 最短寻道时间优先
+
+> SSTF, Shortest Seek Time First
要求访问的磁道与当前磁头所在磁道距离最近的优先进行调度。这种算法并不能保证平均寻道时间最短,但是比 FCFS 好很多。
-### 3. 扫描算法(SCAN)
+### 3. 扫描算法
+
+> SCAN
SSTF 会出现饥饿现象。考虑以下情况,新进程请求访问的磁道与磁头所在磁道的距离总是比一个在等待的进程来的近,那么等待的进程会一直等待下去。
@@ -736,13 +811,79 @@ SCAN 算法在 SSTF 算法之上考虑了磁头的移动方向,要求所请求
当一个磁头自里向外移动时,移到最外侧会改变移动方向为自外向里,这种移动的规律类似于电梯的运行,因此又常称 SCAN 算法为电梯调度算法。
-### 4. 循环扫描算法(CSCAN)
+### 4. 循环扫描算法
+
+> CSCAN
CSCAN 对 SCAN 进行了改动,要求磁头始终沿着一个方向移动。
+# 六、链接
+
+## 编译系统
+
+以下是一个 hello.c 程序:
+
+```c
+#include
+
+int main()
+{
+ printf("hello, world\n");
+ return 0;
+}
+```
+
+在 Unix 系统上,由编译器把源文件转换为目标文件。
+
+```bash
+gcc -o hello hello.c
+```
+
+这个过程大致如下:
+
+
+
+1. 预处理阶段:处理以 # 开头的预处理命令;
+2. 编译阶段:翻译成汇编程序;
+3. 汇编阶段:将汇编程序翻译可重定向目标程序,它是二进制的;
+4. 链接阶段:将可重定向目标程序和 printf.o 等单独预编译好的目标文件进行合并,得到最终的可执行目标程序。
+
+## 目标文件
+
+1. 可执行目标文件:可以直接在内存中执行;
+2. 可重定向目标文件:可与其他可重定向目标文件在链接阶段合并,创建一个可执行目标文件;
+3. 共享目标文件:可以在运行时被动态加载进内存并链接;
+
+## 静态链接
+
+静态连接器以一组可重定向目标文件为输入,生成一个完全链接的可执行目标文件作为输出。链接器主要完成以下两个任务:
+
+1. 符号解析:每个符号对应于一个函数、一个全局变量或一个静态变量,符号解析的目的是将每个符号引用与一个符号定义关联起来。
+2. 重定位:编译器和汇编器生成从地址 0 开始的代码和数据节,链接器通过把每个符号定义与一个内存位置关联起来,从而重定位这些节,然后修改所有对这些符号的引用,使得它们指向这个内存位置。
+
+
+
+## 动态链接
+
+静态库有以下两个问题:
+
+- 当静态库更新时那么整个程序都要重新进行链接;
+- 对于 printf 这种标准函数库,如果每个程序都要有代码,这会极大浪费资源。
+
+共享库是为了解决静态库的这两个问题而设计的,在 Linux 系统中通常用 .so 后缀来表示,Windows 系统上它们被称为 DLL。它具有以下特点:
+
+1. 在给定的文件系统中一个库只有一个 .so 文件,所有引用该库的可执行目标文件都共享这个文件,它不会被复制到引用它的可执行文件中;
+2. 在内存中,一个共享库的 .text 节的一个副本可以被不同的正在运行的进程共享。
+
+
+
# 参考资料
- Tanenbaum A S, Bos H. Modern operating systems[M]. Prentice Hall Press, 2014.
- 汤子瀛, 哲凤屏, 汤小丹. 计算机操作系统[M]. 西安电子科技大学出版社, 2001.
- Bryant, R. E., & O’Hallaron, D. R. (2004). 深入理解计算机系统.
+- [Operating System Notes](https://applied-programming.github.io/Operating-Systems-Notes/)
- [进程间的几种通信方式](http://blog.csdn.net/yufaw/article/details/7409596)
+- [Operating-System Structures](https://www.cs.uic.edu/\~jbell/CourseNotes/OperatingSystems/2_Structures.html)
+- [Processes](http://cse.csusb.edu/tongyu/courses/cs460/notes/process.php)
+- [Inter Process Communication Presentation[1]](https://www.slideshare.net/rkolahalam/inter-process-communication-presentation1)
diff --git a/notes/计算机网络.md b/notes/计算机网络.md
index 9e52f3d0..56623655 100644
--- a/notes/计算机网络.md
+++ b/notes/计算机网络.md
@@ -1,64 +1,38 @@
-* [第一章 概述](#第一章-概述)
+* [一、概述](#一概述)
* [网络的网络](#网络的网络)
* [ISP](#isp)
- * [互联网的组成](#互联网的组成)
* [主机之间的通信方式](#主机之间的通信方式)
* [电路交换与分组交换](#电路交换与分组交换)
- * [1. 电路交换](#1-电路交换)
- * [2. 报文交换](#2-报文交换)
- * [3. 分组交换](#3-分组交换)
* [时延](#时延)
- * [1. 发送时延](#1-发送时延)
- * [2. 传播时延](#2-传播时延)
- * [3. 处理时延](#3-处理时延)
- * [4. 排队时延](#4-排队时延)
* [计算机网络体系结构*](#计算机网络体系结构)
- * [1. 七层协议](#1-七层协议)
- * [2. 五层协议](#2-五层协议)
- * [3. 数据在各层之间的传递过程](#3-数据在各层之间的传递过程)
- * [4. TCP/IP 体系结构](#4-tcpip-体系结构)
-* [第二章 物理层](#第二章-物理层)
+* [二、物理层](#二物理层)
* [通信方式](#通信方式)
* [带通调制](#带通调制)
* [信道复用技术](#信道复用技术)
- * [1. 频分复用、时分复用](#1-频分复用时分复用)
- * [2. 统计时分复用](#2-统计时分复用)
- * [3. 波分复用](#3-波分复用)
- * [4. 码分复用](#4-码分复用)
-* [第三章 数据链路层](#第三章-数据链路层)
+* [三、数据链路层](#三数据链路层)
+ * [信道分类](#信道分类)
* [三个基本问题](#三个基本问题)
- * [1. 封装成帧](#1-封装成帧)
- * [2. 透明传输](#2-透明传输)
- * [3. 差错检测](#3-差错检测)
- * [点对点信道 - PPP 协议](#点对点信道---ppp-协议)
- * [局域网的拓扑](#局域网的拓扑)
- * [广播信道 - CSMA/CD 协议*](#广播信道---csmacd-协议)
- * [集线器](#集线器)
+ * [局域网](#局域网)
+ * [PPP 协议](#ppp-协议)
+ * [CSMA/CD 协议*](#csmacd-协议)
+ * [扩展局域网*](#扩展局域网)
* [MAC 层*](#mac-层)
- * [虚拟局域网](#虚拟局域网)
-* [第四章 网络层*](#第四章-网络层)
+* [四、网络层*](#四网络层)
* [网际协议 IP 概述](#网际协议-ip-概述)
* [IP 数据报格式](#ip-数据报格式)
- * [IP 地址编址](#ip-地址编址)
- * [1. 分类](#1-分类)
- * [2. 子网划分](#2-子网划分)
- * [3. 无分类](#3-无分类)
+ * [IP 地址编址方式](#ip-地址编址方式)
* [IP 地址和 MAC 地址](#ip-地址和-mac-地址)
* [地址解析协议 ARP](#地址解析协议-arp)
* [路由器的结构](#路由器的结构)
- * [交换机与路由器的区别](#交换机与路由器的区别)
* [路由器分组转发流程](#路由器分组转发流程)
* [路由选择协议](#路由选择协议)
- * [1. 内部网关协议 RIP](#1-内部网关协议-rip)
- * [2. 内部网关协议 OSPF](#2-内部网关协议-ospf)
- * [3. 外部网关协议 BGP](#3-外部网关协议-bgp)
* [网际控制报文协议 ICMP](#网际控制报文协议-icmp)
* [分组网间探测 PING](#分组网间探测-ping)
- * [IP 多播](#ip-多播)
+ * [Traceroute](#traceroute)
* [虚拟专用网 VPN](#虚拟专用网-vpn)
* [网络地址转换 NAT](#网络地址转换-nat)
-* [第五章 运输层*](#第五章-运输层)
+* [五、运输层*](#五运输层)
* [UDP 和 TCP 的特点](#udp-和-tcp-的特点)
* [UDP 首部格式](#udp-首部格式)
* [TCP 首部格式](#tcp-首部格式)
@@ -68,19 +42,11 @@
* [TCP 可靠传输](#tcp-可靠传输)
* [TCP 流量控制](#tcp-流量控制)
* [TCP 拥塞控制](#tcp-拥塞控制)
- * [慢开始与拥塞避免](#慢开始与拥塞避免)
- * [快重传与快恢复](#快重传与快恢复)
-* [第六章 应用层*](#第六章-应用层)
+* [六、应用层*](#六应用层)
* [域名系统 DNS](#域名系统-dns)
- * [1. 层次结构](#1-层次结构)
- * [2. 解析过程](#2-解析过程)
* [文件传输协议 FTP](#文件传输协议-ftp)
* [远程终端协议 TELNET](#远程终端协议-telnet)
- * [万维网 WWW](#万维网-www)
* [电子邮件协议](#电子邮件协议)
- * [POP3](#pop3)
- * [IMAP](#imap)
- * [SMTP](#smtp)
* [动态主机配置协议 DHCP](#动态主机配置协议-dhcp)
* [点对点传输 P2P](#点对点传输-p2p)
* [Web 页面请求过程](#web-页面请求过程)
@@ -89,44 +55,40 @@
-# 第一章 概述
+# 一、概述
## 网络的网络
网络把主机连接起来,而互联网是把多种不同的网络连接起来,因此互联网是网络的网络。
-
+
## ISP
互联网服务提供商 ISP 可以从互联网管理机构获得许多 IP 地址,同时拥有通信线路以及路由器等联网设备,个人或机构向 ISP 缴纳一定的费用就可以接入互联网。
+
+
目前的互联网是一种多层次 ISP 结构,ISP 根据覆盖面积的大小分为主干 ISP、地区 ISP 和本地 ISP。
互联网交换点 IXP 允许两个 ISP 直接相连而不用经过第三个 ISP。
-
+
-## 互联网的组成
-
-1. 边缘部分:所有连接在互联网上的主机,用户可以直接使用;
-2. 核心部分:由大量的网络和连接这些网络的路由器组成,为边缘部分的主机提供服务。
-
-
## 主机之间的通信方式
-**1. 客户-服务器(C/S)**
+1. 客户-服务器(C/S):客户是服务的请求方,服务器是服务的提供方。
-客户是服务的请求方,服务器是服务的提供方。
+2. 对等(P2P):不区分客户和服务器。
-**2. 对等(P2P)**
-
-不区分客户和服务器。
+
## 电路交换与分组交换
-
+
+
+(以上分别为:电路交换、报文交换以及分组交换)
### 1. 电路交换
@@ -140,9 +102,7 @@
分组交换也使用了存储转发,但是转发的是分组而不是报文。把整块数据称为一个报文,由于一个报文可能很长,需要先进行切分,来满足分组能处理的大小。在每个切分的数据前面加上首部之后就成为了分组,首部包含了目的地址和源地址等控制信息。
-
-
-存储转发允许在一条传输线路上传送多个主机的分组,因此两个用户之间的通信不需要占用端到端的线路资源。
+存储转发允许在一条传输线路上传送多个主机的分组,也就是说两个用户之间的通信不需要占用端到端的线路资源。
相比于报文交换,由于分组比报文更小,因此分组交换的存储转发速度更加快速。
@@ -150,7 +110,7 @@
总时延 = 发送时延 + 传播时延 + 处理时延 + 排队时延
-
+
### 1. 发送时延
@@ -170,7 +130,7 @@
### 3. 处理时延
-主机或路由器收到分组时进行处理所需要的时间,例如分析首部,从分组中提取数据部分等。
+主机或路由器收到分组时进行处理所需要的时间,例如分析首部、从分组中提取数据部、进行差错检验或查找适当的路由等。
### 4. 排队时延
@@ -178,7 +138,7 @@
## 计算机网络体系结构*
-
+
### 1. 七层协议
@@ -193,11 +153,11 @@
2. 运输层:提供的是进程间的通用数据传输服务。由于应用层协议很多,定义通用的运输层协议就可以支持不断增多的应用层协议。运输层包括两种协议:传输控制协议 TCP,提供面向连接、可靠的数据传输服务,数据单位为报文段;用户数据报协议 UDP,提供无连接、尽最大努力的数据传输服务,数据单位为用户数据报。TCP 主要提供完整性服务,UDP 主要提供及时性服务。
-3. 网络层:为主机之间提供服务,而不是像运输层协议那样是为主机中的进程提供服务。网络层把运输层传递下来的报文段或者用户数据报封装成分组来进行传输。
+3. 网络层:为主机之间提供数据传输服务,而运输层协议是为主机中的进程提供服务。网络层把运输层传递下来的报文段或者用户数据报封装成分组。
-4. 数据链路层:网络层针对的还是主机之间,而主机之间可以有很多链路,链路层协议就是为相邻结点之间提供服务。数据链路层把网络层传来的分组封装成帧。
+4. 数据链路层:网络层针对的还是主机之间的数据传输服务,而主机之间可以有很多链路,链路层协议就是为同一链路的结点提供服务。数据链路层把网络层传来的分组封装成帧。
-5. 物理层:考虑的是怎样在传输媒体上传输数据比特流,而不是指具体的传输媒体。物理层的作用是尽可能屏蔽传输媒体和通信手段的差异,使物理层上的数据链路层感觉不到这些差异。
+5. 物理层:考虑的是怎样在传输媒体上传输数据比特流,而不是指具体的传输媒体。物理层的作用是尽可能屏蔽传输媒体和通信手段的差异,使数据链路层感觉不到这些差异。
### 3. 数据在各层之间的传递过程
@@ -205,7 +165,7 @@
路由器只有下面三层协议,因为路由器位于网络核心中,不需要为进程或者应用程序提供服务,因此也就不需要运输层和应用层。
-
+
### 4. TCP/IP 体系结构
@@ -213,13 +173,13 @@
现在的 TCP/IP 体系结构不严格遵循 OSI 分层概念,应用层可能会直接使用 IP 层或者网络接口层。
-
+
TCP/IP 协议族是一种沙漏形状,中间小两边大,IP 协议在其中占用举足轻重的地位。
-
+
-# 第二章 物理层
+# 二、物理层
## 通信方式
@@ -231,7 +191,7 @@ TCP/IP 协议族是一种沙漏形状,中间小两边大,IP 协议在其中
模拟信号是连续的信号,数字信号是离散的信号。带通调制把数字信号转换为模拟信号。
-
+
## 信道复用技术
@@ -239,29 +199,29 @@ TCP/IP 协议族是一种沙漏形状,中间小两边大,IP 协议在其中
频分复用的所有用户在相同的时间占用不同的频率带宽资源;时分复用的所有用户在不同的时间占用相同的频率带宽资源。
-使用这两种方式进行通信,在通信的过程中用户会一直占用一部分信道资源。但是由于计算机数据的突发性质,没必要一直占用信道资源而不让出给其它用户使用,因此这两种方式对信道的利用率都不高。
+使用这两种方式进行通信,在通信的过程中用户会一直占用一部分信道资源。但是由于计算机数据的突发性质,通信过程没必要一直占用信道资源而不让出给其它用户使用,因此这两种方式对信道的利用率都不高。
-
+
### 2. 统计时分复用
是对时分复用的一种改进,不固定每个用户在时分复用帧中的位置,只要有数据就集中起来组成统计时分复用帧然后发送。
-
+
### 3. 波分复用
光的频分复用。由于光的频率很高,因此习惯上用波长而不是频率来表示所使用的光载波。
-
+
### 4. 码分复用
为每个用户分配 m bit 的码片,并且所有的码片正交,对于任意两个码片 和 有
-
+
-为了方便,取 m=8,设码片 为 00011011。在拥有该码片的用户发送比特 1 时就发送该码片,发送比特 0 时就发送该码片的反码 11100100。
+为了讨论方便,取 m=8,设码片 为 00011011。在拥有该码片的用户发送比特 1 时就发送该码片,发送比特 0 时就发送该码片的反码 11100100。
在计算时将 00011011 记作 (-1 -1 -1 +1 +1 -1 +1 +1),可以得到
@@ -275,9 +235,14 @@ TCP/IP 协议族是一种沙漏形状,中间小两边大,IP 协议在其中
码分复用需要发送的数据量为原先的 m 倍。
-
+
-# 第三章 数据链路层
+# 三、数据链路层
+
+## 信道分类
+
+1. 点对点信道:一对一通信方式;
+2. 广播信道:一对多通信方式。
## 三个基本问题
@@ -285,81 +250,105 @@ TCP/IP 协议族是一种沙漏形状,中间小两边大,IP 协议在其中
将网络层传下来的分组添加首部和尾部,用于标记帧的开始和结束。
-
+
### 2. 透明传输
透明表示一个实际存在的事物看起来好像不存在一样。
-帧中有首部和尾部,如果帧的数据部分含有和首部尾部相同的内容,那么帧的开始和结束位置就会被错误的判定。需要在数据中出现首部尾部相同的内容前面插入转义字符,如果需要传输的内容正好就是转义字符,那么就在转义字符前面再加个转义字符,在接收端进行处理之后可以还原出原始数据。这个过程透明传输的内容是转义字符,用户察觉不到转义字符的存在。
+帧使用首部和尾部进行定界,如果帧的数据部分含有和首部尾部相同的内容,那么帧的开始和结束位置就会被错误的判定。需要在数据部分出现首部尾部相同的内容前面插入转义字符,如果出现转义字符,那么就在转义字符前面再加个转义字符,在接收端进行处理之后可以还原出原始数据。这个过程透明传输的内容是转义字符,用户察觉不到转义字符的存在。
-
+
### 3. 差错检测
目前数据链路层广泛使用了循环冗余检验(CRC)来检查比特差错。
-## 点对点信道 - PPP 协议
+## 局域网
-互联网用户通常需要连接到某个 ISP 之后才能接入到互联网,PPP 协议就是用户计算机和 ISP 进行通信时所使用的数据链路层协议。
+局域网是典型的一种广播信道,主要特点是网络为一个单位所拥有,且地理范围和站点数目均有限。
-
+可以按照网络拓扑对局域网进行分类:
-在 PPP 的帧中
+
+
+## PPP 协议
+
+用于点对点信道中。互联网用户通常需要连接到某个 ISP 之后才能接入到互联网,PPP 协议是用户计算机和 ISP 进行通信时所使用的数据链路层协议。
+
+
+
+在 PPP 的帧中:
- F 字段为帧的定界符
- A 和 C 字段暂时没有意义
- FCS 字段是使用 CRC 的检验序列
- 信息部分的长度不超过 1500
-
+
-## 局域网的拓扑
+## CSMA/CD 协议*
-
-
-## 广播信道 - CSMA/CD 协议*
-
-在广播信道上,同一时间只能允许一台计算机发送数据。
+用于广播信道中。在广播信道上,同一时间只能允许一台计算机发送数据。
CSMA/CD 表示载波监听多点接入 / 碰撞检测。
- **多点接入** :说明这是总线型网络,许多计算机以多点的方式连接到总线上。
-- **载波监听** :每个站都必须不停地检听信道。在发送前,如果检听信道正在使用,就必须等待。
-- **碰撞检测** :在发送中,如果检听到信道已有其它站正在发送数据,就表示发生了碰撞。虽然每一个站在发送数据之前都已经检听到信道为空闲,但是由于电磁波的传播时延的存在,还是有可能会发生碰撞。
+- **载波监听** :每个站都必须不停地监听信道。在发送前,如果监听到信道正在使用,就必须等待。
+- **碰撞检测** :在发送中,如果监听到信道已有其它站正在发送数据,就表示发生了碰撞。虽然每一个站在发送数据之前都已经监听到信道为空闲,但是由于电磁波的传播时延的存在,还是有可能会发生碰撞。
-
+
记端到端的传播时延为 τ,最先发送的站点最多经过 2τ 就可以知道是否发生了碰撞,称 2τ 为 **争用期** 。只有经过争用期之后还没有检测到碰撞,才能肯定这次发送不会发生碰撞。
当发生碰撞时,站点要停止发送,等待一段时间再发送。这个时间采用 **截断二进制指数退避算法** 来确定,从离散的整数集合 {0, 1, .., (2k -1)} 中随机取出一个数,记作 r,然后取 r 倍的争用期作为重传等待时间。
-## 集线器
+## 扩展局域网*
-从表面上看,使用集线器的局域网在物理上是一个星型网。但是集线器使用电子器件来模拟实际缆线的工作,逻辑上仍是一个总线网,整个系统仍像一个传统以太网那样运行。
+### 1. 在物理层进行扩展
-
+使用集线器进行扩展。
-
+集线器的主要功能是对接收到的信号进行放大,以扩大网络的传输距离。
+
+集线器不能根据 MAC 地址进行转发,而是以广播的方式发送数据帧。
+
+集线器是一种共享式的传输设备,意味着同一时刻只能传输一组数据帧。
+
+
+
+### 2. 在链路层进行扩展
+
+最开始使用的是网桥,它收到一个帧时,根据帧的 MAC 地址,查找网桥中的地址表,确定帧转发的接口。
+
+网桥不是共享式设备,因此性能比集线器这种共享式设备更高。
+
+交换机的问世很快就淘汰了网桥,它实质上是一个多接口网桥,而网桥是两接口。交换机的每个接口都能直接与一个主机或者另一个交换机相连,并且一般都工作在全双工方式。
+
+交换机具有自学习能力,学习的是交换表的内容。交换表中存储着 MAC 地址到接口的映射。下图中,交换机有 4 个接口,主机 A 向主机 B 发送数据帧时,交换机把主机 A 到接口 1 的映射写入交换表中。为了发送数据帧到 B,先查交换表,此时没有主机 B 的表项,那么主机 A 就发送广播帧,主机 C 和主机 D 会丢弃该帧。主机 B 收下之后,查找交换表得到主机 A 映射的接口为 1,就发送数据帧到接口 1,同时交换机添加主机 B 到接口 3 的映射。
+
+
+
+### 3. 虚拟局域网
+
+虚拟局域网可以建立与物理位置无关的逻辑组,只有在同一个虚拟局域网中的成员才会收到链路层广播信息,例如下图中 (A1, A2, A3, A4) 属于一个虚拟局域网,A1 发送的广播会被 A2、A3、A4 收到,而其它站点收不到。
+
+
## MAC 层*
-MAC 地址是 6 字节(48 位)的地址,用于唯一表示网络适配器(网卡),一台主机拥有多少个适配器就有多少个 MAC 地址,例如笔记本电脑普遍存在无线网络适配器和有线网络适配器。
+MAC 地址是 6 字节(48 位)的地址,用于唯一标识网络适配器(网卡),一台主机拥有多少个适配器就有多少个 MAC 地址,例如笔记本电脑普遍存在无线网络适配器和有线网络适配器。
-
+
+
+在 MAC 帧中:
- **类型** :标记上层使用的协议;
- **数据** :长度在 46-1500 之间,如果太小则需要填充;
- **FCS** :帧检验序列,使用的是 CRC 检验方法;
- **前同步码** :只是为了计算 FCS 临时加入的,计算结束之后会丢弃。
-## 虚拟局域网
-
-虚拟局域网可以建立与物理位置无关的逻辑组,只有在同一个虚拟局域网中的成员才会收到链路层广播信息,例如下图中 (A1, A2, A3, A4) 属于一个虚拟局域网,A1 发送的广播会被 A2、A3、A4 收到,而其它站点收不到。
-
-
-
-# 第四章 网络层*
+# 四、网络层*
## 网际协议 IP 概述
@@ -367,7 +356,7 @@ MAC 地址是 6 字节(48 位)的地址,用于唯一表示网络适配器
使用 IP 协议,可以把异构的物理网络连接起来,使得在网络层看起来好像是一个统一的网络。
-
+
与 IP 协议配套使用的还有三个协议:
@@ -375,15 +364,15 @@ MAC 地址是 6 字节(48 位)的地址,用于唯一表示网络适配器
2. 网际控制报文协议 ICMP(Internet Control Message Protocol)
3. 网际组管理协议 IGMP(Internet Group Management Protocol)
-
+
## IP 数据报格式
-
+
- **版本** : 有 4(IPv4)和 6(IPv6)两个值;
-- **首部长度** : 占 4 位,因此最大值为 15。值为 1 表示的是 1 个 32 位字的长度,也就是 4 字节。因为首部固定长度为 20 字节,因此该值最小为 5。如果可选部分的长度不是 4 字节的整数倍,就用尾部的填充部分来填充。
+- **首部长度** : 占 4 位,因此最大值为 15。值为 1 表示的是 1 个 32 位字的长度,也就是 4 字节。因为首部固定长度为 20 字节,因此该值最小为 5。如果可选字段的长度不是 4 字节的整数倍,就用尾部的填充部分来填充。
- **区分服务** : 用来获得更好的服务,一般情况下不使用。
@@ -393,7 +382,7 @@ MAC 地址是 6 字节(48 位)的地址,用于唯一表示网络适配器
- **片偏移** : 和标识符一起,用于发生分片的情况。片偏移的单位为 8 字节。
-
+
- **生存时间** :TTL,它的存在是为了防止无法交付的数据报在互联网中不断兜圈子。以路由器跳数为单位,当 TTL 为 0 时就丢弃数据报。
@@ -401,21 +390,21 @@ MAC 地址是 6 字节(48 位)的地址,用于唯一表示网络适配器
- **首部检验和** :因为数据报每经过一个路由器,都要重新计算检验和,因此检验和不包含数据部分可以减少计算的工作量。
-## IP 地址编址
+## IP 地址编址方式
IP 地址的编址方式经历了三个历史阶段:
-1. 分类;
-2. 子网划分;
-3. 无分类。
+1. 分类
+2. 子网划分
+3. 无分类
### 1. 分类
-由两部分组成,网络号和主机号,其中不同类别具有不同的网络号长度,并且是固定的。
+由两部分组成,网络号和主机号,其中不同分类具有不同的网络号长度,并且是固定的。
IP 地址 ::= {< 网络号 >, < 主机号 >}
-
+
### 2. 子网划分
@@ -423,7 +412,7 @@ IP 地址 ::= {< 网络号 >, < 主机号 >}
IP 地址 ::= {< 网络号 >, < 子网号 >, < 主机号 >}
-要使用子网,必须配置子网掩码。一个 B 类地址的默认子网掩码为 255.255.0.0,如果 B 类地址的子网占两个比特,那么子网掩码为 11111111 11111111 11000000 000000,也就是 255.255.192.0。
+要使用子网,必须配置子网掩码。一个 B 类地址的默认子网掩码为 255.255.0.0,如果 B 类地址的子网占两个比特,那么子网掩码为 11111111 11111111 11000000 00000000,也就是 255.255.192.0。
### 3. 无分类
@@ -443,43 +432,38 @@ CIDR 的地址掩码可以继续称为子网掩码,子网掩码首 1 长度为
网络层实现主机之间的通信,而链路层实现具体每段链路之间的通信。因此在通信过程中,IP 数据报的源地址和目的地址始终不变,而 MAC 地址随着链路的改变而改变。
-
+
## 地址解析协议 ARP
实现由 IP 地址得到 MAC 地址。
-每个主机都有一个 ARP 高速缓存,存放映射表。如果一个 IP 地址到 MAC 地址的映射不在该表中,主机通过广播的方式发送 ARP 请求分组,匹配 IP 地址的主机会发送 ARP 响应分组告知其 MAC 地址。
+
-
+每个主机都有一个 ARP 高速缓存,里面有本局域网上的各主机和路由器的 IP 地址到硬件地址的映射表。
+
+如果主机 A 知道主机 B 的 IP 地址,但是 ARP 高速缓存中没有该 IP 地址到 MAC 地址的映射,此时主机 A 通过广播的方式发送 ARP 请求分组,主机 B 收到该请求后会发送 ARP 响应分组给主机 A 告知其 MAC 地址,随后主机 A 向其高速缓存中写入主机 B 的 IP 地址到硬件地址的映射。
+
+
## 路由器的结构
-路由器从功能上可以划分为两大部分:路由选择和分组转发。
+路由器从功能上可以划分为:路由选择和分组转发。
-分组转发部分由三部分组成:交换结构、一组输入端口和一组输出端口。
+分组转发结构由三个部分组成:交换结构、一组输入端口和一组输出端口。
-
-
-交换结构的交换网络有以下三种实现方式:
-
-
-
-## 交换机与路由器的区别
-
-- 交换机工作于数据链路层,能识别 MAC 地址,根据 MAC 地址转发链路层数据帧。具有自学机制来维护 IP 地址与 MAC 地址的映射。
-- 路由器位于网络层,能识别 IP 地址并根据 IP 地址转发分组。维护着路由表,根据路由表选择最佳路线。
+
## 路由器分组转发流程
-1. 从数据报的首部提取目的主机的 IP 地址 D,得到目的网络地址 N。(路由表项是网络号而不是 IP 地址,这样做大大减少了路由表条目数量)
+1. 从数据报的首部提取目的主机的 IP 地址 D,得到目的网络地址 N。
2. 若 N 就是与此路由器直接相连的某个网络地址,则进行直接交付;
3. 若路由表中有目的地址为 D 的特定主机路由,则把数据报传送给表中所指明的下一跳路由器;
4. 若路由表中有到达网络 N 的路由,则把数据报传送给路由表中所指明的下一跳路由器;
5. 若路由表中有一个默认路由,则把数据报传送给路由表中所指明的默认路由器;
6. 报告转发分组出错。
-
+
## 路由选择协议
@@ -492,7 +476,7 @@ CIDR 的地址掩码可以继续称为子网掩码,子网掩码首 1 长度为
1. 内部网关协议 IGP(Interior Gateway Protocol):在 AS 内部使用,如 RIP 和 OSPF。
2. 外部网关协议 EGP(External Gateway Protocol):在 AS 之间使用,如 BGP。
-
+
### 1. 内部网关协议 RIP
@@ -532,32 +516,32 @@ BGP 只能寻找一条比较好的路由,而不是最佳路由。它采用路
每个 AS 都必须配置 BGP 发言人,通过在两个相邻 BGP 发言人之间建立 TCP 连接来交换路由信息。
-
+
## 网际控制报文协议 ICMP
ICMP 是为了更有效地转发 IP 数据报和提高交付成功的机会。它封装在 IP 数据报中,但是不属于高层协议。
-
+
ICMP 报文分为差错报告报文和询问报文。
-
+
## 分组网间探测 PING
PING 是 ICMP 的一个重要应用,主要用来测试两台主机之间的连通性。
-PING 的过程:
+Ping 发送的 IP 数据报封装的是无法交付的 UDP 用户数据报。
-1. PING 同一个网段的主机:查找目的主机的 MAC 地址,然后直接交付。如果无法查找到 MAC 地址,就要进行一次 ARP 请求。
-2. PING 不同网段的主机:发送到网关让其进行转发。同样要发送到网关也需要通过查找网关的 MAC 地址,根据 MAC 地址进行转发。
+## Traceroute
-## IP 多播
+Traceroute 是 ICMP 的另一个应用,用来跟踪一个分组从源点到终点的路径。
-在一对多的通信中,多播不需要将分组复制多份,从而大大节约网络资源。
-
-
+1. 源主机向目的主机发送一连串的 IP 数据报。第一个数据报 P1 的生存时间 TTL 设置为 1,但 P1 到达路径上的第一个路由器 R1 时,R1 收下它并把 TTL 减 1,此时 TTL 等于 0,R1 就把 P1 丢弃,并向源主机发送一个 ICMP 时间超过差错报告报文;
+2. 源主机接着发送第二个数据报 P2,并把 TTL 设置为 2。P2 先到达 R1,R1 收下后把 TTL 减 1 再转发给 R2,R2 收下后也把 TTL 减 1,由于此时 TTL 等于 0,R2 就丢弃 P2,并向源主机发送一个 ICMP 时间超过差错报文。
+3. 不断执行这样的步骤,直到最后一个数据报刚刚到达目的主机,主机不转发数据报,也不把 TTL 值减 1。但是因为数据报封装的是无法交付的 UDP,因此目的主机要向源主机发送 ICMP 终点不可达差错报告报文。
+4. 之后源主机知道了到达目的主机所经过的路由器 IP 地址以及到达每个路由器的往返时间。
## 虚拟专用网 VPN
@@ -571,9 +555,9 @@ PING 的过程:
VPN 使用公用的互联网作为本机构各专用网之间的通信载体。专用指机构内的主机只与本机构内的其它主机通信;虚拟指“好像是”,而实际上并不是,它有经过公用的互联网。
-下图中,场所 A 和 B 的通信部经过互联网,如果场所 A 的主机 X 要和另一个场所 B 的主机 Y 通信,IP 数据报的源地址是 10.1.0.1,目的地址是 10.2.0.3。数据报先发送到与互联网相连的路由器 R1,R1 对内部数据进行加密,然后重新加上数据报的首部,源地址是路由器 R1 的全球地址 125.1.2.3,目的地址是路由器 R2 的全球地址 194.4.5.6。路由器 R2 收到数据报后将数据部分进行解密,恢复原来的数据报,此时目的地址为 10.2.0.3,就交付给 Y。
+下图中,场所 A 和 B 的通信经过互联网,如果场所 A 的主机 X 要和另一个场所 B 的主机 Y 通信,IP 数据报的源地址是 10.1.0.1,目的地址是 10.2.0.3。数据报先发送到与互联网相连的路由器 R1,R1 对内部数据进行加密,然后重新加上数据报的首部,源地址是路由器 R1 的全球地址 125.1.2.3,目的地址是路由器 R2 的全球地址 194.4.5.6。路由器 R2 收到数据报后将数据部分进行解密,恢复原来的数据报,此时目的地址为 10.2.0.3,就交付给 Y。
-
+
## 网络地址转换 NAT
@@ -581,29 +565,27 @@ VPN 使用公用的互联网作为本机构各专用网之间的通信载体。
在以前,NAT 将本地 IP 和全球 IP 一一对应,这种方式下拥有 n 个全球 IP 地址的专用网内最多只可以同时有 n 台主机接入互联网。为了更有效地利用全球 IP 地址,现在常用的 NAT 转换表把运输层的端口号也用上了,使得多个专用网内部的主机共用一个全球 IP 地址。使用端口号的 NAT 也叫做网络地址与端口转换 NAPT。
-
+
-# 第五章 运输层*
+# 五、运输层*
-网络层只把分组发送到目的主机,但是真正通信的并不是主机而是主机中的进程。
-
-运输层提供了应用进程间的逻辑通信。运输层向高层用户屏蔽了下面网络层的核心细节,使应用程序看见的好像在两个运输层实体之间有一条端到端的逻辑通信信道。
+网络层只把分组发送到目的主机,但是真正通信的并不是主机而是主机中的进程。运输层提供了进程间的逻辑通信,运输层向高层用户屏蔽了下面网络层的核心细节,使应用程序看见的好像在两个运输层实体之间有一条端到端的逻辑通信信道。
## UDP 和 TCP 的特点
-- 用户数据包协议 UDP(User Datagram Protocol)是无连接的,尽最大可能交付,没有拥塞控制,面向报文(对于应用程序传下来的报文不合并也不拆分,只是添加 UDP 首部)。
+- 用户数据报协议 UDP(User Datagram Protocol)是无连接的,尽最大可能交付,没有拥塞控制,面向报文(对于应用程序传下来的报文不合并也不拆分,只是添加 UDP 首部)。
-- 传输控制协议 TCP(Transmission Control Protocol) 是面向连接的,提供可靠交付,有流量控制,拥塞控制,提供全双工通信,面向字节流(把应用层传下来的报文看成字节流,把字节流组织成大小不等的数据块)
+- 传输控制协议 TCP(Transmission Control Protocol)是面向连接的,提供可靠交付,有流量控制,拥塞控制,提供全双工通信,面向字节流(把应用层传下来的报文看成字节流,把字节流组织成大小不等的数据块)。
## UDP 首部格式
-
+
-首部字段只有 8 个字节,包括源端口、目的端口、长度、检验和。12 字节的伪首部是为了计算检验和而临时添加的。
+首部字段只有 8 个字节,包括源端口、目的端口、长度、检验和。12 字节的伪首部是为了计算检验和临时添加的。
## TCP 首部格式
-
+
- **序号** :用于对字节流进行编号,例如序号为 301,表示第一个字节的编号为 301,如果携带的数据长度为 100 字节,那么下一个报文段的序号应为 401。
@@ -621,38 +603,57 @@ VPN 使用公用的互联网作为本机构各专用网之间的通信载体。
## TCP 的三次握手
-
+
假设 A 为客户端,B 为服务器端。
1. 首先 B 处于 LISTEN(监听)状态,等待客户的连接请求。
+
2. A 向 B 发送连接请求报文段,SYN=1,ACK=0,选择一个初始的序号 x。
+
3. B 收到连接请求报文段,如果同意建立连接,则向 A 发送连接确认报文段,SYN=1,ACK=1,确认号为 x+1,同时也选择一个初始的序号 y。
+
4. A 收到 B 的连接确认报文段后,还要向 B 发出确认,确认号为 y+1,序号为 x+1。
+
5. B 收到 A 的确认后,连接建立。
+**三次握手的原因**
+
+第三次握手是为了防止失效的连接请求到达服务器,让服务器错误打开连接。
+
+失效的连接请求是指,客户端发送的连接请求在网络中滞留,客户端因为没及时收到服务器端发送的连接确认,因此就重新发送了连接请求。滞留的连接请求并不是丢失,之后还是会到达服务器。如果不进行第三次握手,那么服务器会误认为客户端重新请求连接,然后打开了连接。但是并不是客户端真正打开这个连接,因此客户端不会给服务器发送数据,这个连接就白白浪费了。
## TCP 的四次挥手
-
+
以下描述不讨论序号和确认号,因为序号和确认号的规则比较简单。并且不讨论 ACK,因为 ACK 在连接建立之后都为 1。
-1. A 发送连接释放报文段,FIN=1;
-2. B 收到之后发出确认,此时 TCP 属于半关闭状态,B 能向 A 发送数据但是 A 不能向 B 发送数据;
-3. 当 B 要不再需要连接时,发送连接释放请求报文段,FIN=1;
-4. A 收到后发出确认,此时连接释放。
+1. A 发送连接释放报文段,FIN=1。
+
+2. B 收到之后发出确认,此时 TCP 属于半关闭状态,B 能向 A 发送数据但是 A 不能向 B 发送数据。
+
+3. 当 B 要不再需要连接时,发送连接释放请求报文段,FIN=1。
+
+4. A 收到后发出确认,进入 TIME-WAIT 状态,等待 2MSL 时间后释放连接。
+
+5. B 收到 A 的确认后释放连接。
+
+**四次挥手的原因**
+
+客户端发送了 FIN 连接释放报文之后,服务器收到了这个报文,就进入了 CLOSE-WAIT 状态。这个状态是为了让服务器端发送还未传送完毕的数据,传送完毕之后,服务器会发送 FIN 连接释放报文。
**TIME_WAIT**
-客户端接收到服务器端的 FIN 报文后进入此状态,此时并不是直接进入 CLOSED 状态,还需要等待一个时间计时器设置的时间。这么做有两个理由:
+客户端接收到服务器端的 FIN 报文后进入此状态,此时并不是直接进入 CLOSED 状态,还需要等待一个时间计时器设置的时间 2MSL。这么做有两个理由:
1. 确保最后一个确认报文段能够到达。如果 B 没收到 A 发送来的确认报文段,那么就会重新发送连接释放请求报文段,A 等待一段时间就是为了处理这种情况的发生。
-2. 可能存在“已失效的连接请求报文段”,为了防止这种报文段出现在本次连接之外,需要等待一段时间。
+
+2. 等待一段时间是为了让本连接持续时间内所产生的所有报文段都从网络中消失,使得下一个新的连接不会出现旧的连接请求报文段。
## TCP 滑动窗口
-
+
窗口是缓存的一部分,用来暂时存放字节流。发送方和接收方各有一个窗口,接收方通过 TCP 报文段中的窗口字段告诉发送方自己的窗口大小,发送方根据这个值和其它信息设置自己的窗口大小。
@@ -668,52 +669,52 @@ TCP 使用超时重传来实现可靠传输:如果一个已经发送的报文
-超时时间 RTO 应该略大于 RRTs,TCP 使用的超时时间计算如下:
+超时时间 RTO 应该略大于 RTTs,TCP 使用的超时时间计算如下:
-其中 RTTd 为偏差,它与新的 RRT 和 RRTs 有关。
+其中 RTTd 为偏差。
## TCP 流量控制
流量控制是为了控制发送方发送速率,保证接收方来得及接收。
-接收方发送的确认报文中的窗口字段可以用来控制发送方窗口大小,从而影响发送方的发送速率。例如将窗口字段设置为 0,则发送方不能发送数据。
+接收方发送的确认报文中的窗口字段可以用来控制发送方窗口大小,从而影响发送方的发送速率。将窗口字段设置为 0,则发送方不能发送数据。
## TCP 拥塞控制
如果网络出现拥塞,分组将会丢失,此时发送方会继续重传,从而导致网络拥塞程度更高。因此当出现拥塞时,应当控制发送方的速率。这一点和流量控制很像,但是出发点不同。流量控制是为了让接收方能来得及接受,而拥塞控制是为了降低整个网络的拥塞程度。
-
+
-TCP 主要通过四种算法来进行拥塞控制:慢开始、拥塞避免、快重传、快恢复。发送方需要维护有一个叫做拥塞窗口(cwnd)的状态变量。注意拥塞窗口与发送方窗口的区别,拥塞窗口只是一个状态变量,实际决定发送方能发送多少数据的是发送方窗口。
+TCP 主要通过四种算法来进行拥塞控制:慢开始、拥塞避免、快重传、快恢复。发送方需要维护一个叫做拥塞窗口(cwnd)的状态变量。注意拥塞窗口与发送方窗口的区别,拥塞窗口只是一个状态变量,实际决定发送方能发送多少数据的是发送方窗口。
为了便于讨论,做如下假设:
1. 接收方有足够大的接收缓存,因此不会发生流量控制;
2. 虽然 TCP 的窗口基于字节,但是这里设窗口的大小单位为报文段。
-
+
-### 慢开始与拥塞避免
+### 1. 慢开始与拥塞避免
-发送的最初执行慢开始,令 cwnd=1,发送方只能发送 1 个报文段;当收到确认后,将 cwnd 加倍,因此之后发送方能够发送的报文段为:2、4、8 ...
+发送的最初执行慢开始,令 cwnd=1,发送方只能发送 1 个报文段;当收到确认后,将 cwnd 加倍,因此之后发送方能够发送的报文段数量为:2、4、8 ...
注意到慢开始每个轮次都将 cwnd 加倍,这样会让 cwnd 增长速度非常快,从而使得发送方发送的速度增长速度过快,网络拥塞的可能也就更高。设置一个慢开始门限 ssthresh,当 cwnd >= ssthresh 时,进入拥塞避免,每个轮次只将 cwnd 加 1。
-如果出现了超时,则令 ssthresh = cwnd / 2,然后重新执行慢开始。
+如果出现了超时,则令 ssthresh = cwnd/2,然后重新执行慢开始。
-### 快重传与快恢复
+### 2. 快重传与快恢复
在接收方,要求每次接收到报文段都应该发送对已收到有序报文段的确认,例如已经接收到 M1 和 M2 ,此时收到 M4 ,应当发送对 M2 的确认。
在发送方,如果收到三个重复确认,那么可以确认下一个报文段丢失,例如收到三个 M2 ,则 M3 丢失。此时执行快重传,立即重传下一个报文段。
-在这种情况下,只是丢失个别报文段,而不是网络拥塞,因此执行快恢复,令 ssthresh = cwnd / 2 ,cwnd = ssthresh,注意到此时直接进入拥塞避免。
+在这种情况下,只是丢失个别报文段,而不是网络拥塞,因此执行快恢复,令 ssthresh = cwnd/2 ,cwnd = ssthresh,注意到此时直接进入拥塞避免。
-
+
-# 第六章 应用层*
+# 六、应用层*
## 域名系统 DNS
@@ -725,9 +726,9 @@ TCP 主要通过四种算法来进行拥塞控制:慢开始、拥塞避免、
一个域名由多个层次构成,从上层到下层分别为顶级域名、二级域名、三级域名以及四级域名。所有域名可以画成一颗域名树。
-
+
-
+
域名服务器可以分为以下四类:
@@ -738,25 +739,29 @@ TCP 主要通过四种算法来进行拥塞控制:慢开始、拥塞避免、
区和域的概念不同,可以在一个域中划分多个区。图 b 在域 abc.com 中划分了两个区:abc.com 和 y.abc.com
-
+
因此就需要两个权限域名服务器:
-
+
### 2. 解析过程
主机向本地域名服务器解析的过程采用递归,而本地域名服务器向其它域名服务器解析可以使用递归和迭代两种方式。
-迭代的方式下,本地域名服务器向一个域名服务器解析请求解析之后,结果返回到本地域名服务器,然后本地域名服务器继续向其它域名服务器请求解析;而递归地方式下,结果不是直接返回的,而是继续向前请求解析,最后的结果才会返回。
+迭代的方式下,本地域名服务器向一个域名服务器解析请求解析之后,结果返回到本地域名服务器,然后本地域名服务器继续向其它域名服务器请求解析;而递归的方式下,结果不是直接返回的,而是继续向前请求解析,最后的结果才会返回。
-
+
+
+### 3. 使用的运输层协议
+
+DNS 在解析的过程使用 UDP 进行传输,因为 UDP 最大只支持 512 字节的数据,如果超过的话就需要使用 TCP 传输。
## 文件传输协议 FTP
FTP 在运输层使用 TCP,并且需要建立两个并行的 TCP 连接:控制连接和数据连接。控制连接在整个会话期间一直保持打开,而数据连接在数据传送完毕之后就关闭。控制连接使用端口号 21,数据连接使用端口号 20。
-
+
## 远程终端协议 TELNET
@@ -764,29 +769,25 @@ TELNET 用于登录到远程主机上,并且远程主机上的输出也会返
TELNET 可以适应许多计算机和操作系统的差异,例如不同操作系统系统的换行符定义。
-## 万维网 WWW
-
-[HTTP](https://github.com/CyC2018/InterviewNotes/blob/master/notes/HTTP.md)
-
## 电子邮件协议
一个电子邮件系统由三部分组成:用户代理、邮件服务器以及邮件发送协议和读取协议。其中发送协议常用 SMTP,读取协议常用 POP3 和 IMAP。
-
+
-### POP3
+### 1. POP3
POP3 的特点是只要用户从服务器上读取了邮件,就把该邮件删除。
-### IMAP
+### 2. IMAP
IMAP 协议中客户端和服务器上的邮件保持同步,如果不去手动删除邮件,那么服务器上的邮件也不会被删除。IMAP 这种做法可以让用户随时随地去访问服务器上的邮件。IMAP 协议也支持创建自定义的文件夹。
-### SMTP
+### 3. SMTP
SMTP 只能发送 ASCII 码,而互联网邮件扩充 MIME 可以发送二进制文件。MIME 并没有改动或者取代 SMTP,而是增加邮件主题的结构,定义了非 ASCII 码的编码规则。
-
+
## 动态主机配置协议 DHCP
@@ -812,37 +813,85 @@ P2P 是一个分布式系统,任何时候都有对等方加入或者退出。
## Web 页面请求过程
-1. 向 DNS 服务器发送 DNS 查询报文来解析域名。
+### 1. DHCP 配置主机信息
-2. 开始进行 HTTP 会话,需要先建立 TCP 连接。
+1. 假设主机最开始没有 IP 地址以及其它信息,那么就需要先使用 DHCP 来获取。
-3. 在运输层的传输过程中,HTTP 报文被封装进 TCP 中。HTTP 请求报文使用端口号 80,因为服务器监听的是 80 端口。连接建立之后,服务器会随机分配一个端口号给特定的客户端,之后的 TCP 传输都是用这个分配的端口号。
+2. 主机生成一个 DHCP 请求报文,并将这个报文放入具有目的端口 67 和源端口 68 的 UDP 报文段中。
-4. 在网络层的传输过程中,TCP 报文段会被封装进 IP 分组中,IP 分组经过路由选择,最后到达目的地。
+3. 该报文段则被放入在一个具有广播 IP 目的地址(255.255.255.255) 和源 IP 地址(0.0.0.0)的 IP 数据报中。
-5. 在链路层,IP 分组会被封装进 MAC 帧中,IP 地址解析成 MAC 地址需要使用 ARP。
+4. 该数据报则被放置在 MAC 帧中,该帧具有目的地址 FF:FF:FF:FF:FF:FF,将广播到与交换机连接的所有设备。
-6. 客户端发送 HTTP 请求报文,请求获取页面。
+5. 连接在交换机的 DHCP 服务器收到广播帧之后,不断地向上分解得到 IP 数据报、UDP 报文段、DHCP 请求报文,之后生成 DHCP ACK 报文,该报文包含以下信息:IP 地址、DNS 服务器的 IP 地址、默认网关路由器的 IP 地址和子网掩码。该报文被放入 UDP 报文段中,UDP 报文段有被放入 IP 数据报中,最后放入 MAC 帧中。
-7. 服务器发送 HTTP 相应报文,客户端从而获取该页面。
+8. 该帧的目的地址是请求主机的 MAC 地址,因为交换机具有自学习能力,之前主机发送了广播帧之后就记录了 MAC 地址到其转发接口的交换表项,因此现在交换机就可以直接知道应该向哪个接口发送该帧。
-8. 浏览器得到页面内容之后,解析并渲染,向用户展示页面。
+9. 主机收到该帧后,不断分解得到 DHCP 报文。之后就配置它的 IP 地址、子网掩码和 DNS 服务器的 IP 地址,并在其 IP 转发表中安装默认网关。
+### 2. ARP 解析 MAC 地址
+
+1. 主机通过浏览器生成一个 TCP 套接字,套接字向 HTTP 服务器发送 HTTP 请求。为了生成该套接字,主机需要知道网站的域名对应的 IP 地址。
+
+2. 主机生成一个 DNS 查询报文,该报文具有 53 号端口,因为 DNS 服务器的端口号是 53。
+
+3. 该 DNS 查询报文被放入目的地址为 DNS 服务器 IP 地址的 IP 数据报中。
+
+4. 该 IP 数据报被放入一个以太网帧中,该帧将发送到网关路由器。
+
+5. DHCP 过程只知道网关路由器的 IP 地址,为了获取网关路由器的 MAC 地址,需要使用 ARP 协议。
+
+6. 主机生成一个包含目的地址为网关路由器 IP 地址的 ARP 查询报文,将该 ARP 查询报文放入一个具有广播目的地址(FF:FF:FF:FF:FF:FF)的以太网帧中,并向交换机发送该以太网帧,交换机将该帧转发给所有的连接设备,包括网关路由器。
+
+7. 网关路由器接收到该帧后,不断向上分解得到 ARP 报文,发现其中的 IP 地址与其接口的 IP 地址匹配,因此就发送一个 ARP 回答报文,包含了它的 MAC 地址,发回给主机。
+
+### 3. DNS 解析域名
+
+1. 知道了网关路由器的 MAC 地址之后,就可以继续 DNS 的解析过程了。
+
+2. 网关路由器接收到包含 DNS 查询报文的以太网帧后,抽取出 IP 数据报,并根据转发表决定该 IP 数据报应该转发的路由器。
+
+3. 因为路由器具有内部网关协议(RIP、OSPF)和外部网关协议(BGP)这两种路由选择协议,因此路由表中已经配置了网关路由器到达 DNS 服务器的路由表项。
+
+4. 到达 DNS 服务器之后,DNS 服务器抽取出 DNS 查询报文,并在 DNS 数据库中查找待解析的域名。
+
+5. 找到 DNS 记录之后,发送 DNS 回答报文,将该回答报文放入 UDP 报文段中,然后放入 IP 数据报中,通过路由器反向转发回网关路由器,并经过以太网交换机到达主机。
+
+### 4. HTTP 请求页面
+
+1. 有了 HTTP 服务器的 IP 地址之后,主机就能够生成 TCP 套接字,该套接字将用于向 Web 服务器发送 HTTP GET 报文。
+
+2. 在生成 TCP 套接字之前,必须先与 HTTP 服务器进行三次握手来建立连接。生成一个具有目的端口 80 的 TCP SYN 报文段,并向 HTTP 服务器发送该报文段。
+
+3. HTTP 服务器收到该报文段之后,生成 TCP SYNACK 报文段,发回给主机。
+
+4. 连接建立之后,浏览器生成 HTTP GET 报文,并交付给 HTTP 服务器。
+
+5. HTTP 服务器从 TCP 套接字读取 HTTP GET 报文,生成一个 HTTP 响应报文,将 Web 页面内容放入报文主体中,发回给主机。
+
+6. 浏览器收到 HTTP 响应报文后,抽取出 Web 页面内容,之后进行渲染,显示 Web 页面。
## 常用端口
-| 应用层协议 | 端口号 | 运输层协议 |
-| -- | -- | -- |
-| DNS | 53 | UDP |
-| FTP | 控制连接 21,数据连接 20 | TCP |
-| TELNET | 23 | TCP |
-| DHCP | 67 68 | UDP |
-| HTTP | 80 | TCP |
-| SMTP | 25 | TCP |
-| POP3 | 110 | TCP |
-| IMAP | 143 | TCP |
+|应用| 应用层协议 | 端口号 | 运输层协议 | 备注 |
+| :---: | :--: | :--: | :--: | :--:
+| 域名解析 | DNS | 53 | UDP/TCP | 长度超过 512 字节时使用 TCP |
+| 动态主机配置协议 | DHCP | 67/68 | UDP | |
+| 简单网络管理协议 | SNMP | 161/162 | UDP | |
+| 文件传送协议 | FTP | 20/21 | TCP | 控制连接 21,数据连接 20
+| 远程终端协议 | TELNET | 23 | TCP | |
+|超文本传送协议 | HTTP | 80 | TCP | |
+| 简单邮件传送协议 | SMTP | 25 | TCP | |
+| 邮件读取协议 | POP3 | 110 | TCP | |
+| 网际报文存取协议 | IMAP | 143 | TCP | |
+
# 参考资料
-- 计算机网络 第七版
-- 计算机网络 自顶向下方法
+- 计算机网络, 谢希仁
+- JamesF.Kurose, KeithW.Ross, 库罗斯, 等. 计算机网络: 自顶向下方法 [M]. 机械工业出版社, 2014.
+- [Tackling emissions targets in Tokyo](http://www.climatechangenews.com/2011/html/university-tokyo.html)
+- [What does my ISP know when I use Tor?](http://www.climatechangenews.com/2011/html/university-tokyo.html)
+- [Technology-Computer Networking[1]-Computer Networks and the Internet](http://www.linyibin.cn/2017/02/12/technology-ComputerNetworking-Internet/)
+- [P2P 网络概述.](http://slidesplayer.com/slide/11616167/)
+- [Circuit Switching (a) Circuit switching. (b) Packet switching.](http://slideplayer.com/slide/5115386/)
diff --git a/notes/设计模式.md b/notes/设计模式.md
index aecb0f28..b7e6c76a 100644
--- a/notes/设计模式.md
+++ b/notes/设计模式.md
@@ -1,758 +1,53 @@
-* [前言](#前言)
-* [第一章 设计模式入门](#第一章-设计模式入门)
-* [第二章 观察者模式](#第二章-观察者模式)
-* [第三章 装饰模式](#第三章-装饰模式)
-* [第四章 工厂模式](#第四章-工厂模式)
- * [1. 简单工厂](#1-简单工厂)
- * [2. 工厂方法模式](#2--工厂方法模式)
- * [3. 抽象工厂模式](#3--抽象工厂模式)
-* [第五章 单件模式](#第五章-单件模式)
-* [第六章 命令模式](#第六章-命令模式)
-* [第七章 适配器模式与外观模式](#第七章-适配器模式与外观模式)
- * [1. 适配器模式](#1-适配器模式)
- * [2. 外观模式](#2-外观模式)
-* [第八章 模板方法模式](#第八章-模板方法模式)
-* [第九章 迭代器和组合模式](#第九章-迭代器和组合模式)
- * [1. 迭代器模式](#1-迭代器模式)
- * [2. Java 内置的迭代器](#2-java-内置的迭代器)
- * [3. 组合模式](#3-组合模式)
-* [第十章 状态模式](#第十章-状态模式)
-* [第十一章 代理模式](#第十一章-代理模式)
-* [第十二章 复合模式](#第十二章-复合模式)
- * [MVC](#mvc)
-* [第十三章 与设计模式相处](#第十三章-与设计模式相处)
-* [第十四章 剩下的模式](#第十四章-剩下的模式)
+* [一、概述](#一概述)
+* [二、单例模式](#二单例模式)
+* [三、简单工厂](#三简单工厂)
+* [四、工厂方法模式](#四工厂方法模式)
+* [五、抽象工厂模式](#五抽象工厂模式)
+* [参考资料](#参考资料)
-# 前言
-
-文中涉及一些 UML 类图,为了更好地理解,可以先阅读 [UML 类图](https://github.com/CyC2018/Interview-Notebook/blob/master/notes/%E9%9D%A2%E5%90%91%E5%AF%B9%E8%B1%A1%E6%80%9D%E6%83%B3.md#1-%E7%B1%BB%E5%9B%BE)。
-
-需要说明的一点是,文中的 UML 类图和规范的 UML 类图不大相同,其中组合关系使用以下箭头表示:
-
-
-
-# 第一章 设计模式入门
-
-**1. 设计模式概念**
+# 一、概述
设计模式不是代码,而是解决问题的方案,学习现有的设计模式可以做到经验复用。
拥有设计模式词汇,在沟通时就能用更少的词汇来讨论,并且不需要了解底层细节。
-**2. 问题描述**
+# 二、单例模式
-设计不同种类的鸭子拥有不同的叫声和飞行方式。
-
-**3. 简单实现方案**
-
-使用继承的解决方案如下,这种方案代码无法复用,如果两个鸭子类拥有同样的飞行方式,就有两份重复的代码。
-
-
-
-**4. 设计原则**
-
-**封装变化** 在这里变化的是鸭子叫和飞行的行为方式。
-
-**针对接口编程,而不是针对实现编程** 变量声明的类型为父类,而不是具体的某个子类。父类中的方法实现不在父类,而是在各个子类。程序在运行时可以动态改变变量所指向的子类类型。
-
-运用这一原则,将叫和飞行的行为抽象出来,实现多种不同的叫和飞行的子类,让子类去实现具体的叫和飞行方式。
-
-
-
-**多用组合,少用继承** 组合也就是 has-a 关系,通过组合,可以在运行时动态改变实现,只要通过改变父类对象具体指向哪个子类即可。而继承就不能做到这些,继承体系在创建类时就已经确定。
-
-运用这一原则,在 Duck 类中组合 FlyBehavior 和 QuackBehavior 类,performQuack() 和 performFly() 方法委托给这两个类去处理。通过这种方式,一个 Duck 子类可以根据需要去实例化 FlyBehavior 和 QuackBehavior 的子类对象,并且也可以动态地进行改变。
-
-
-
-**5. 整体设计图**
-
-
-
-**6. 模式定义**
-
-**策略模式** :定义了算法族,分别封装起来,让它们之间可以互相替换,此模式让算法的变化独立于使用算法的客户。
-
-**7. 实现代码**
-
-```java
-public abstract class Duck {
- FlyBehavior flyBehavior;
- QuackBehavior quackBehavior;
-
- public Duck(){
- }
-
- public void performFly(){
- flyBehavior.fly();
- }
-
- public void setFlyBehavior(FlyBehavior fb){
- flyBehavior = fb;
- }
-
- public void performQuack(){
- quackBehavior.quack();
- }
-
- public void setQuackBehavior(QuackBehavior qb){
- quackBehavior = qb;
- }
-}
-```
-```java
-public class MallarDuck extends Duck{
- public MallarDuck(){
- flyBehavior = new FlyWithWings();
- quackBehavior = new Quack();
- }
-}
-```
-```java
-public interface FlyBehavior {
- void fly();
-}
-```
-```java
-public class FlyNoWay implements FlyBehavior{
- @Override
- public void fly() {
- System.out.println("FlyBehavior.FlyNoWay");
- }
-}
-```
-```java
-public class FlyWithWings implements FlyBehavior{
- @Override
- public void fly() {
- System.out.println("FlyBehavior.FlyWithWings");
- }
-}
-```
-```java
-public interface QuackBehavior {
- void quack();
-}
-```
-```java
-public class Quack implements QuackBehavior{
- @Override
- public void quack() {
- System.out.println("QuackBehavior.Quack");
- }
-}
-```
-```java
-public class MuteQuack implements QuackBehavior{
- @Override
- public void quack() {
- System.out.println("QuackBehavior.MuteQuack");
- }
-}
-```
-```java
-public class Squeak implements QuackBehavior{
- @Override
- public void quack() {
- System.out.println("QuackBehavior.Squeak");
- }
-}
-```
-```java
-public class MiniDuckSimulator {
- public static void main(String[] args) {
- Duck mallarDuck = new MallarDuck();
- mallarDuck.performQuack();
- mallarDuck.performFly();
- mallarDuck.setFlyBehavior(new FlyNoWay());
- mallarDuck.performFly();
- }
-}
-```
-执行结果
-```html
-QuackBehavior.Quack
-FlyBehavior.FlyWithWings
-FlyBehavior.FlyNoWay
-```
-
-# 第二章 观察者模式
-
-**1. 模式定义**
-
-定义了对象之间的一对多依赖,当一个对象改变状态时,它的所有依赖者都会受到通知并自动更新。主题(Subject)是被观察的对象,而其所有依赖者(Observer)成为观察者。
-
-
-
-**2. 模式类图**
-
-主题中具有注册和移除观察者,并通知所有注册者的功能,主题是通过维护一张观察者列表来实现这些操作的。
-
-观察者拥有一个主题对象的引用,因为注册、移除还有数据都在主题当中,必须通过操作主题才能完成相应功能。
-
-
-
-**3. 问题描述**
-
-天气数据布告板会在天气信息发生改变时更新其内容,布告板有多个,并且在将来会继续增加。
-
-**4. 解决方案类图**
-
-
-
-**5. 设计原则**
-
-**为交互对象之间的松耦合设计而努力** 当两个对象之间松耦合,它们依然可以交互,但是不太清楚彼此的细节。由于松耦合的两个对象之间互相依赖程度很低,因此系统具有弹性,能够应对变化。
-
-**6. 实现代码**
-
-```java
-public interface Subject {
- public void resisterObserver(Observer o);
- public void removeObserver(Observer o);
- public void notifyObserver();
-}
-```
-```java
-import java.util.ArrayList;
-import java.util.List;
-
-public class WeatherData implements Subject {
- private List observers;
- private float temperature;
- private float humidity;
- private float pressure;
-
- public WeatherData() {
- observers = new ArrayList<>();
- }
-
- @Override
- public void resisterObserver(Observer o) {
- observers.add(o);
- }
-
- @Override
- public void removeObserver(Observer o) {
- int i = observers.indexOf(o);
- if (i >= 0) {
- observers.remove(i);
- }
- }
-
- @Override
- public void notifyObserver() {
- for (Observer o : observers) {
- o.update(temperature, humidity, pressure);
- }
- }
-
- public void setMeasurements(float temperature, float humidity, float pressure) {
- this.temperature = temperature;
- this.humidity = humidity;
- this.pressure = pressure;
- notifyObserver();
- }
-}
-```
-```java
-public interface Observer {
- public void update(float temp, float humidity, float pressure);
-}
-```
-```java
-public class CurrentConditionsDisplay implements Observer {
- private Subject weatherData;
-
- public CurrentConditionsDisplay(Subject weatherData) {
- this.weatherData = weatherData;
- weatherData.resisterObserver(this);
- }
-
- @Override
- public void update(float temp, float humidity, float pressure) {
- System.out.println("CurrentConditionsDisplay.update:" + temp + " " + humidity + " " + pressure);
- }
-}
-```
-```java
-public class StatisticsDisplay implements Observer {
- private Subject weatherData;
-
- public StatisticsDisplay(Subject weatherData) {
- this.weatherData = weatherData;
- weatherData.resisterObserver(this);
- }
-
- @Override
- public void update(float temp, float humidity, float pressure) {
- System.out.println("StatisticsDisplay.update:" + temp + " " + humidity + " " + pressure);
- }
-}
-```
-```java
-public class WeatherStation {
- public static void main(String[] args) {
- WeatherData weatherData = new WeatherData();
- CurrentConditionsDisplay currentConditionsDisplay = new CurrentConditionsDisplay(weatherData);
- StatisticsDisplay statisticsDisplay = new StatisticsDisplay(weatherData);
-
- weatherData.setMeasurements(0, 0, 0);
- weatherData.setMeasurements(1, 1, 1);
- }
-}
-```
-执行结果
-```html
-CurrentConditionsDisplay.update:0.0 0.0 0.0
-StatisticsDisplay.update:0.0 0.0 0.0
-CurrentConditionsDisplay.update:1.0 1.0 1.0
-StatisticsDisplay.update:1.0 1.0 1.0
-```
-
-# 第三章 装饰模式
-
-**1. 问题描述**
-
-设计不同种类的饮料,并且每种饮料可以动态添加新的材料,比如可以添加牛奶。计算一种饮料的价格。
-
-**2. 模式定义**
-
-动态地将责任附加到对象上。在扩展功能上,装饰者提供了比继承更有弹性的替代方案。
-
-下图中 DarkRoast 对象被 Mocha 包裹,Mocha 对象又被 Whip 包裹,并且他们都继承自相同父类,都有 cost() 方法,但是外层对象的 cost() 方法实现调用了内层对象的 cost() 方法。因此,如果要在 DarkRoast 上添加 Mocha,那么只需要用 Mocha 包裹 DarkRoast,如果还需要 Whip ,就用 Whip 包裹 Mocha,最后调用 cost() 方法能把三种对象的价格都包含进去。
-
-
-
-**3. 模式类图**
-
-装饰者和具体组件都继承自组件类型,其中具体组件的方法实现不需要依赖于其它对象,而装饰者拥有一个组件类型对象,这样它可以装饰其它装饰者或者具体组件。所谓装饰,就是把这个装饰者套在被装饰的对象之外,从而动态扩展被装饰者的功能。装饰者的方法有一部分是自己的,这属于它的功能,然后调用被装饰者的方法实现,从而也保留了被装饰者的功能。可以看到,具体组件应当是装饰层次的最低层,因为只有具体组件有直接实现而不需要委托给其它对象去处理。
-
-
-
-**4. 问题解决方案的类图**
-
-
-
-**5. 设计原则**
-
-**类应该对扩展开放,对修改关闭。** 也就是添加新功能时不需要修改代码。在本章问题中该原则体现在,在饮料中添加新的材料,而不需要去修改饮料的代码。观察则模式也符合这个原则。不可能所有类都能实现这个原则,应当把该原则应用于设计中最有可能改变的地方。
-
-**6. Java I/O 中的装饰者模式**
-
-
-
-**7. 代码实现**
-
-```java
-public interface Beverage {
- public double cost();
-}
-```
-```java
-public class HouseBlend implements Beverage{
- @Override
- public double cost() {
- return 1;
- }
-}
-```
-```java
-public class DarkRoast implements Beverage{
- @Override
- public double cost() {
- return 1;
- }
-}
-```
-```java
-public abstract class CondimentDecorator implements Beverage{
- protected Beverage beverage;
-}
-```
-```java
-public class Mocha extends CondimentDecorator {
-
- public Mocha(Beverage beverage) {
- this.beverage = beverage;
- }
-
- @Override
- public double cost() {
- return 1 + beverage.cost();
- }
-}
-```
-```java
-public class Milk extends CondimentDecorator {
-
- public Milk(Beverage beverage) {
- this.beverage = beverage;
- }
-
- @Override
- public double cost() {
- return 1 + beverage.cost();
- }
-}
-```
-```java
-public class StartbuzzCoffee {
- public static void main(String[] args) {
- Beverage beverage = new HouseBlend();
- beverage = new Mocha(beverage);
- beverage = new Milk(beverage);
- System.out.println(beverage.cost());
- }
-}
-```
-
-输出
-
-```html
-3.0
-```
-
-# 第四章 工厂模式
-
-## 1. 简单工厂
-
-**1. 问题描述**
-
-有不同的 Pizza,根据不同的情况用不同的子类实例化一个 Pizza 对象。
-
-**2. 定义**
-
-简单工厂不是设计模式,更像是一种编程习惯。在实例化一个超类的对象时,可以用它的所有子类来进行实例化,要根据具体需求来决定使用哪个子类。在这种情况下,把实例化的操作放到工厂来中,让工厂类来决定应该用哪个子类来实例化。这样做把客户对象和具体子类的实现解耦,客户对象不再需要知道有哪些子类以及实例化哪个子类。因为客户类往往有多个,如果不使用简单工厂,那么所有的客户类都要知道所有子类的细节,一旦子类发生改变,例如增加子类,那么所有的客户类都要发生改变。
-
-
-
-**3. 解决方案类图**
-
-
-
-
-**4. 代码实现**
-
-```java
-public interface Pizza {
- public void make();
-}
-```
-```java
-public class CheesePizza implements Pizza{
- @Override
- public void make() {
- System.out.println("CheesePizza");
- }
-}
-```
-```java
-public class GreekPizza implements Pizza{
- @Override
- public void make() {
- System.out.println("GreekPizza");
- }
-}
-```
-```java
-public class SimplePizzaFactory {
- public Pizza createPizza(String type) {
- if (type.equals("cheese")) {
- return new CheesePizza();
- } else if (type.equals("greek")) {
- return new GreekPizza();
- } else {
- throw new UnsupportedOperationException();
- }
- }
-}
-```
-```java
-public class PizzaStore {
- public static void main(String[] args) {
- SimplePizzaFactory simplePizzaFactory = new SimplePizzaFactory();
- Pizza pizza = simplePizzaFactory.createPizza("cheese");
- pizza.make();
- }
-}
-```
-
-运行结果
-
-```java
-CheesePizza
-```
-
-## 2. 工厂方法模式
-
-**1. 问题描述**
-
-每个地区的 Pizza 店虽然种类相同,但是都有自己的风味,需要单独区分。例如,一个客户点了纽约的 cheese 种类的 Pizza 和在芝加哥点的相同种类的 Pizza 是不同的。
-
-**2. 模式定义**
-
-定义了一个创建对象的接口,但由子类决定要实例化的类是哪一个。工厂方法让类把实例化推迟到子类。
-
-**3. 模式类图**
-
-在简单工厂中,创建对象的是另一个类,而在工厂方法中,是由子类来创建对象。下图中,Creator 有一个 anOperation() 方法,这个方法需要用到一组产品类,这组产品类由每个子类来创建。
-
-可以为每个子类创建单独的简单工厂来创建每一个产品类,但是把简单工厂中创建对象的代码放到子类中来可以减少类的数目,因为子类不算是产品类,因此完全可以这么做。
-
-
-
-**4. 解决方案类图**
-
-
-
-**5. 代码实现**
-
-```java
-public interface Pizza {
- public void make();
-}
-```
-```java
-public interface PizzaStore {
- public Pizza orderPizza(String item);
-}
-```
-```java
-public class NYStyleCheesePizza implements Pizza{
- @Override
- public void make() {
- System.out.println("NYStyleCheesePizza is making..");
- }
-}
-```
-```java
-public class NYStyleVeggiePizza implements Pizza {
- @Override
- public void make() {
- System.out.println("NYStyleVeggiePizza is making..");
- }
-}
-```
-```java
-public class ChicagoStyleCheesePizza implements Pizza{
- @Override
- public void make() {
- System.out.println("ChicagoStyleCheesePizza is making..");
- }
-}
-```
-```java
-public class ChicagoStyleVeggiePizza implements Pizza{
- @Override
- public void make() {
- System.out.println("ChicagoStyleVeggiePizza is making..");
- }
-}
-```
-```java
-public class NYPizzaStore implements PizzaStore {
- @Override
- public Pizza orderPizza(String item) {
- Pizza pizza = null;
- if (item.equals("cheese")) {
- pizza = new NYStyleCheesePizza();
- } else if (item.equals("veggie")) {
- pizza = new NYStyleVeggiePizza();
- } else {
- throw new UnsupportedOperationException();
- }
- pizza.make();
- return pizza;
- }
-}
-```
-```java
-public class ChicagoPizzaStore implements PizzaStore {
- @Override
- public Pizza orderPizza(String item) {
- Pizza pizza = null;
- if (item.equals("cheese")) {
- pizza = new ChicagoStyleCheesePizza();
- } else if (item.equals("veggie")) {
- pizza = new ChicagoStyleVeggiePizza();
- } else {
- throw new UnsupportedOperationException();
- }
- pizza.make();
- return pizza;
- }
-}
-```
-```java
-public class PizzaTestDrive {
- public static void main(String[] args) {
- PizzaStore nyStore = new NYPizzaStore();
- nyStore.orderPizza("cheese");
- PizzaStore chicagoStore = new ChicagoPizzaStore();
- chicagoStore.orderPizza("cheese");
- }
-}
-```
-
-运行结果
-
-```html
-NYStyleCheesePizza is making..
-ChicagoStyleCheesePizza is making..
-```
-
-## 3. 抽象工厂模式
-
-**1. 设计原则**
-
-**依赖倒置原则** :要依赖抽象,不要依赖具体类。听起来像是针对接口编程,不针对实现编程,但是这个原则说明了:不能让高层组件依赖底层组件,而且,不管高层或底层组件,两者都应该依赖于抽象。例如,下图中 PizzaStore 属于高层组件,它依赖的是 Pizza 的抽象类,这样就可以不用关心 Pizza 的具体实现细节。
-
-
-
-**2. 模式定义**
-
-提供一个接口,用于创建相关或依赖对象的家族,而不需要明确指定具体类。
-
-**3. 模式类图**
-
-抽象工厂模式创建的是对象家族,也就是很多对象而不是一个对象,并且这些对象是相关的,也就是说必须一起创建出来。而工厂模式只是用于创建一个对象,这和抽象工厂模式有很大不同。并且,抽象工厂模式也用到了工厂模式来创建单一对象,在类图左部,AbstractFactory 中的 CreateProductA 和 CreateProductB 方法都是让子类来实现,这两个方法单独来看就是在创建一个对象,这符合工厂模式的定义。至于创建对象的家族这一概念是在 Client 体现,Client 要通过 AbstractFactory 同时调用两个方法来创建出两个对象,在这里这两个对象就有很大的相关性,Client 需要这两个对象的协作才能完成任务。从高层次来看,抽象工厂使用了组合,即 Cilent 组合了 AbstractFactory ,而工厂模式使用了继承。
-
-
-
-**4. 解决方案类图**
-
-
-
-**5. 代码实现**
-
-```java
-public interface Dough {
- public String doughType();
-}
-```
-```java
-public class ThickCrustDough implements Dough{
-
- @Override
- public String doughType() {
- return "ThickCrustDough";
- }
-}
-```
-```java
-public class ThinCrustDough implements Dough {
- @Override
- public String doughType() {
- return "ThinCrustDough";
- }
-}
-```
-```java
-public interface Sauce {
- public String sauceType();
-}
-```
-```java
-public class MarinaraSauce implements Sauce {
- @Override
- public String sauceType() {
- return "MarinaraSauce";
- }
-}
-```
-```java
-public class PlumTomatoSauce implements Sauce {
- @Override
- public String sauceType() {
- return "PlumTomatoSauce";
- }
-}
-```
-```java
-public interface PizzaIngredientFactory {
- public Dough createDough();
- public Sauce createSauce();
-}
-```
-```java
-public class NYPizzaIngredientFactory implements PizzaIngredientFactory{
- @Override
- public Dough createDough() {
- return new ThickCrustDough();
- }
-
- @Override
- public Sauce createSauce() {
- return new MarinaraSauce();
- }
-}
-```
-```java
-public class ChicagoPizzaIngredientFactory implements PizzaIngredientFactory{
- @Override
- public Dough createDough() {
- return new ThinCrustDough();
- }
-
- @Override
- public Sauce createSauce() {
- return new PlumTomatoSauce();
- }
-}
-```
-```java
-public class NYPizzaStore {
- private PizzaIngredientFactory ingredientFactory;
-
- public NYPizzaStore() {
- ingredientFactory = new NYPizzaIngredientFactory();
- }
-
- public void makePizza() {
- Dough dough = ingredientFactory.createDough();
- Sauce sauce = ingredientFactory.createSauce();
- System.out.println(dough.doughType());
- System.out.println(sauce.sauceType());
- }
-}
-```
-```java
-public class NYPizzaStoreTestDrive {
- public static void main(String[] args) {
- NYPizzaStore nyPizzaStore = new NYPizzaStore();
- nyPizzaStore.makePizza();
- }
-}
-```
-
-运行结果
-
-```html
-ThickCrustDough
-MarinaraSauce
-```
-
-# 第五章 单件模式
-
-**1. 模式定义**
+## 意图
确保一个类只有一个实例,并提供了一个全局访问点。
-**2. 模式类图**
+## 类图
-使用一个私有构造器、一个私有静态变量以及一个公有静态函数来实现。私有构造函数保证了不能通过构造函数来创建对象实例,只能通过公有静态函数返回唯一的私有静态变量。
+使用一个私有构造函数、一个私有静态变量以及一个公有静态函数来实现。
-
+私有构造函数保证了不能通过构造函数来创建对象实例,只能通过公有静态函数返回唯一的私有静态变量。
-**3. 懒汉式-线程不安全**
+
-以下实现中,私有静态变量被延迟化实例化,这样做的好处是,如果没有用到该类,那么就不会创建私有静态变量,从而节约资源。
+## 使用场景
-这个实现在多线程环境下是不安全的,如果多个线程能够同时进入 if(uniqueInstance == null) 内的语句块,那么就会多次实例化 uniqueInstance 私有静态变量。
+- Logger Classes
+- Configuration Classes
+- Accesing resources in shared mode
+- Factories implemented as Singletons
+
+## JDK 的使用
+
+- [java.lang.Runtime#getRuntime()](http://docs.oracle.com/javase/8/docs/api/java/lang/Runtime.html#getRuntime%28%29)
+- [java.awt.Desktop#getDesktop()](http://docs.oracle.com/javase/8/docs/api/java/awt/Desktop.html#getDesktop--)
+- [java.lang.System#getSecurityManager()](http://docs.oracle.com/javase/8/docs/api/java/lang/System.html#getSecurityManager--)
+
+## 实现
+
+### 懒汉式-线程不安全
+
+以下实现中,私有静态变量 uniqueInstance 被延迟化实例化,这样做的好处是,如果没有用到该类,那么就不会实例化 uniqueInstance,从而节约资源。
+
+这个实现在多线程环境下是不安全的,如果多个线程能够同时进入 if(uniqueInstance == null) ,那么就会多次实例化 uniqueInstance。
```java
public class Singleton {
@@ -771,33 +66,34 @@ public class Singleton {
}
```
-**4. 懒汉式-线程安全**
+### 懒汉式-线程安全
-只需要对 getUniqueInstance() 方法加锁,那么在一个时间点只能有一个线程能够进入该方法,从而避免了对 uniqueInstance 进行多次实例化的问题。但是这样有一个问题,就是当一个线程进入该方法之后,其它线程视图进入该方法都必须等待,因此性能上有一定的损耗。
+只需要对 getUniqueInstance() 方法加锁,那么在一个时间点只能有一个线程能够进入该方法,从而避免了对 uniqueInstance 进行多次实例化的问题。
+
+但是这样有一个问题,就是当一个线程进入该方法之后,其它线程试图进入该方法都必须等待,因此性能上有一定的损耗。
```java
- public static synchronized Singleton getUniqueInstance() {
- if (uniqueInstance == null) {
- uniqueInstance = new Singleton();
- }
- return uniqueInstance;
+public static synchronized Singleton getUniqueInstance() {
+ if (uniqueInstance == null) {
+ uniqueInstance = new Singleton();
}
+ return uniqueInstance;
+}
```
-**5. 饿汉式-线程安全**
+### 饿汉式-线程安全
-线程不安全问题主要是由于静态实例变量被初始化了多次,那么静态实例变量采用直接实例化就可以解决问题。但是直接初始化的方法也丢失了延迟初始化节约资源的优势。
+线程不安全问题主要是由于 uniqueInstance 被实例化了多次,如果 uniqueInstance 采用直接实例化的话,就不会被实例化多次,也就不会产生线程不安全问题。但是直接实例化的方式也丢失了延迟实例化带来的节约资源的优势。
```java
private static Singleton uniqueInstance = new Singleton();
```
-**6. 双重校验锁-线程安全**
+### 双重校验锁-线程安全
-因为 uniqueInstance 只需要被初始化一次,之后就可以直接使用了。加锁操作只需要对初始化那部分的代码进行,也就是说,只有当 uniqueInstance 没有被初始化时,才需要进行加锁。
-
-双重校验锁先判断 uniqueInstance 是否已经被初始化了,如果没有被初始化,那么才对初始化的语句进行加锁。如果只做一次判断,那么多个线程还是有可能同时进入实例化语句块的,因此需要仅此第二次的判断。
+uniqueInstance 只需要被实例化一次,之后就可以直接使用了。加锁操作只需要对实例化那部分的代码进行。也就是说,只有当 uniqueInstance 没有被实例化时,才需要进行加锁。
+双重校验锁先判断 uniqueInstance 是否已经被初始化了,如果没有被实例化,那么才对实例化语句进行加锁。
```java
public class Singleton {
@@ -820,965 +116,248 @@ public class Singleton {
}
```
-# 第六章 命令模式
-
-**1. 问题描述**
-
-设计一个遥控器,它有很多按钮,每个按钮可以发起一个命令,让一个家电完成相应操作。
-
-有非常多的家电,并且之后会增加家电。
-
-
-
-
-
-**2. 模式定义**
-
-将命令封装成对象,以便使用不同的命令来参数化其它对象。
-
-**3. 解决方案类图**
-
-- RemoteControl 是遥控器,它可以为每个按钮设置命令对象,并且调用命令对象的 execute() 方法。
-
-- Command 就是命令对象,命令模式正式将各种命令封装成 Commad 对象来实现的。
-
-- Light 是命令真正的执行者。可以注意到 LightOnCommand 和 LightOffCommand 类组合了一个 Light 对象,通过组合的方法,就可以将 excute() 方法委托给 Light 对象来执行。
-
-- RemoteLoader 是客户端,注意它与 RemoteControl 的区别。因为 RemoteControl 不能主动地调用自身的方法,因此也就不能当成是客户端。客户端好比人,只有人才能去真正去使用遥控器。
-
-
-
-**4. 模式类图**
-
-
-
-**5. 代码实现**
+考虑下面的实现,也就是只使用了一个 if 语句。在 uniqueInstance == null 的情况下,如果两个线程同时执行 if 语句,那么两个线程就会同时进入 if 语句块内。虽然在 if 语句块内有加锁操作,但是两个线程都会执行 uniqueInstance = new Singleton(); 这条语句,只是早晚的问题,也就是说会进行两次实例化,从而产生了两个实例。因此必须使用双重校验锁,也就是需要使用两个 if 判断。
```java
-public interface Command {
- public void execute();
-}
-```
-
-```java
-public class Light {
-
- public void on() {
- System.out.println("Light is on!");
- }
-
- public void off() {
- System.out.println("Light is off!");
+if (uniqueInstance == null) {
+ synchronized (Singleton.class) {
+ uniqueInstance = new Singleton();
}
}
```
-```java
-public class LightOnCommand implements Command{
- Light light;
+uniqueInstance 采用 volatile 关键字修饰也是很有必要的。
- public LightOnCommand(Light light) {
- this.light = light;
- }
+`uniqueInstance = new Singleton();` 这段代码其实是分为三步执行。
- @Override
- public void execute() {
- light.on();
- }
-}
-```
+1. 分配内存空间。
+2. 初始化对象。
+3. 将 uniqueInstance 指向分配的内存地址。
-```java
-/**
- * 遥控器类
- */
-public class SimpleRemoteControl {
- Command slot;
+但是由于 JVM 具有指令重排的特性,有可能执行顺序变为了 `1>3>2`,这在单线程情况下自然是没有问题。但如果是多线程就有可能 B 线程获得是一个还没有被初始化的对象以致于程序出错。
- public SimpleRemoteControl() {
+所以使用 volatile 修饰的目的是禁止 JVM 的指令重排,保证在多线程环境下也能正常运行。
- }
+# 三、简单工厂
- public void setCommand(Command command) {
- this.slot = command;
- }
+## 意图
- public void buttonWasPressed() {
- slot.execute();
- }
+在创建一个对象时不向客户暴露内部细节;
-}
-```
+## 类图
-```java
-/**
- * 客户端
- */
-public class RemoteLoader {
- public static void main(String[] args) {
- SimpleRemoteControl remote = new SimpleRemoteControl();
- Light light = new Light();
- LightOnCommand lightOnCommand = new LightOnCommand(light);
- remote.setCommand(lightOnCommand);
- remote.buttonWasPressed();
- }
-}
-```
+简单工厂不是设计模式,更像是一种编程习惯。它把实例化的操作单独放到一个类中,这个类就成为简单工厂类,让简单工厂类来决定应该用哪个子类来实例化。
-输出
+
-```html
-Light is on!
-```
+这样做能把客户类和具体子类的实现解耦,客户类不再需要知道有哪些子类以及应当实例化哪个子类。因为客户类往往有多个,如果不使用简单工厂,所有的客户类都要知道所有子类的细节。而且一旦子类发生改变,例如增加子类,那么所有的客户类都要进行修改。
-# 第七章 适配器模式与外观模式
-
-## 1. 适配器模式
-
-**1. 模式定义**
-
-将一个类的接口,转换为客户期望的另一个接口。适配器让原本不兼容的类可以合作无间。
-
-
-
-**2. 模式类图**
-
-有两种适配器模式的实现,一种是对象方式,一种是类方式。对象方式是通过组合的方法,让适配器类(Adapter)拥有一个待适配的对象(Adaptee),从而把相应的处理委托给待适配的对象。类方式用到多重继承,Adapter 继承 Target 和 Adaptee,先把 Adapter 当成 Adaptee 类型然后实例化一个对象,再把它当成 Target 类型的,这样 Client 就可以把这个对象当成 Target 的对象来处理,同时拥有 Adaptee 的方法。
-
-
-
-
-
-**3. 问题描述**
-
-鸭子(Duck)和火鸡(Turkey)拥有不同的叫声,Duck 调用的是 quack() 方法,而 Turkey 调用 gobble() 方法。
-
-要求将 Turkey 的 gobble() 方法适配成 Duck 的 quack() 方法。
-
-**4. 解决方案类图**
-
-
-
-**5. 代码实现**
-
-```java
-public interface Duck {
- public void quack();
-}
-```
-
-```java
-public interface Turkey {
- public void gobble();
-}
-```
-
-```java
-public class WildTurkey implements Turkey{
- @Override
- public void gobble() {
- System.out.println("gobble!");
- }
-}
-```
-
-```java
-public class TurkeyAdapter implements Duck{
- Turkey turkey;
-
- public TurkeyAdapter(Turkey turkey) {
- this.turkey = turkey;
- }
-
- @Override
- public void quack() {
- turkey.gobble();
- }
-}
-```
-
-```java
-public class DuckTestDrive {
- public static void main(String[] args) {
- Turkey turkey = new WildTurkey();
- Duck duck = new TurkeyAdapter(turkey);
- duck.quack();
- }
-}
-```
-
-运行结果
-
-```html
-gobble!
-```
-
-## 2. 外观模式
-
-**1. 模式定义**
-
-提供了一个统一的接口,用来访问子系统中的一群接口,从而让子系统更容易使用。
-
-**2. 模式类图**
-
-
-
-**3. 问题描述**
-
-家庭影院中有众多电器,当要进行观看电影时需要对很多电器进行操作。要求简化这些操作,使得家庭影院类只提供一个简化的接口,例如提供一个看电影相关的接口。
-
-
-
-**4. 解决方案类图**
-
-
-
-**5. 设计原则**
-
-**最少知识原则** :只和你的密友谈话。也就是应当使得客户对象所需要交互的对象尽可能少。
-
-**6. 代码实现**
-
-过于简单,无实现。
-
-# 第八章 模板方法模式
-
-**1. 模式定义**
-
-在一个方法中定义一个算法的骨架,而将一些步骤延迟到子类中。
-
-模板方法使得子类可以在不改变算法结构的情况下,重新定义算法中的某些步骤。
-
-**2. 模式类图**
-
-模板方法 templateMethod() 定义了算法的骨架,确定了 primitiveOperation1() 和 primitiveOperation2() 方法执行的顺序,而 primitiveOperation1() 和 primitiveOperation2() 让子类去实现。
-
-
-
-**3. 问题描述**
-
-冲咖啡和冲茶都有类似的流程,但是某些步骤会有点不一样,要求复用那些相同步骤的代码。
-
-
-
-**4. 解决方案类图**
-
-其中 prepareRecipe() 方法就是模板方法,它确定了其它四个方法的具体执行步骤。其中 brew() 和 addCondiments() 方法在子类中实现。
-
-
-
-**5. 设计原则**
-
-**好莱坞原则** :别调用(打电话给)我们,我们会调用(打电话给)你。这一原则可以防止依赖腐败,即防止高层组件依赖低层组件,低层组件又依赖高层组件。该原则在模板方法的体现为,只有父类会调用子类,子类不会调用父类。
-
-**6. 钩子**
-
-钩子(hock):某些步骤在不同实现中可有可无,可以先定义一个什么都不做的方法,把它加到模板方法中,如果子类需要它就覆盖默认实现并加上自己的实现。
-
-**7. 代码实现**
-
-```java
-public abstract class CaffeineBeverage {
-
- final void prepareRecipe(){
- boilWater();
- brew();
- pourInCup();
- addCondiments();
- }
-
- abstract void brew();
-
- abstract void addCondiments();
-
- void boilWater(){
- System.out.println("boilWater");
- }
-
- void pourInCup(){
- System.out.println("pourInCup");
- }
-}
-```
-
-```java
-public class Coffee extends CaffeineBeverage{
- @Override
- void brew() {
- System.out.println("Coffee.brew");
- }
-
- @Override
- void addCondiments() {
- System.out.println("Coffee.addCondiments");
- }
-}
-```
-
-```java
-public class Tea extends CaffeineBeverage{
- @Override
- void brew() {
- System.out.println("Tea.brew");
- }
-
- @Override
- void addCondiments() {
- System.out.println("Tea.addCondiments");
- }
-}
-```
-
-```java
-public class CaffeineBeverageTestDrive {
- public static void main(String[] args) {
- CaffeineBeverage caffeineBeverage = new Coffee();
- caffeineBeverage.prepareRecipe();
- System.out.println("-----------");
- caffeineBeverage = new Tea();
- caffeineBeverage.prepareRecipe();
- }
-}
-```
-
-运行结果
-
-```html
-boilWater
-Coffee.brew
-pourInCup
-Coffee.addCondiments
------------
-boilWater
-Tea.brew
-pourInCup
-Tea.addCondiments
-```
-
-# 第九章 迭代器和组合模式
-
-## 1. 迭代器模式
-
-**1. 模式定义**
-
-提供顺序访问一个聚合对象中的各个元素的方法,而又不暴露聚合对象内部的表示。
-
-**2. 模式类图**
-
-- Aggregate 是聚合类,其中 createIterator() 方法可以产生一个 Iterator 对象;
-
-- Iterator 主要定义了 hasNext() 和 next() 方法。
-
-- Client 需要拥有一个 Aggregate 对象,这是很明显的。为了迭代变量 Aggregate 对象,也需要拥有 Iterator 对象。
-
-
-
-**3. 代码实现**
-
-```java
-public class Aggregate {
-
- private int[] items;
-
- public Aggregate() {
- items = new int[10];
- for (int i = 0; i < items.length; i++) {
- items[i] = i;
- }
- }
-
- public Iterator createIterator() {
- return new ConcreteIterator(items);
- }
-
-}
-```
-
-```java
-public interface Iterator {
- boolean hasNext();
- int next();
-}
-```
-
-```java
-public class ConcreteIterator implements Iterator {
-
- private int[] items;
- private int position = 0;
-
- public ConcreteIterator(int[] items) {
- this.items = items;
- }
-
- @Override
- public boolean hasNext() {
- return position < items.length;
- }
-
- @Override
- public int next() {
- return items[position++];
- }
-}
-```
-```java
-public class Client {
- public static void main(String[] args) {
- Aggregate aggregate = new Aggregate();
- Iterator iterator = aggregate.createIterator();
- while(iterator.hasNext()){
- System.out.println(iterator.next());
- }
- }
-}
-```
-运行结果
-```html
-0
-1
-2
-3
-4
-5
-6
-7
-8
-9
-```
-
-## 2. Java 内置的迭代器
-
-**1. 实现接口**
-
-在使用 Java 的迭代器实现时,需要让聚合对象去实现 Iterable 接口,该接口有一个 iterator() 方法会返回一个 Iterator 对象。
-
-使用 Java 内置的迭代器实现,客户对象可以使用 foreach 循环来遍历聚合对象中的每个元素。
-
-Java 中的集合类基本都实现了 Iterable 接口。
-
-**2. 代码实现**
-
-```java
-import java.util.Iterator;
-
-public class Aggregate implements Iterable{
-
- private int[] items;
-
- public Aggregate() {
- items = new int[10];
- for (int i = 0; i < items.length; i++) {
- items[i] = i;
- }
- }
-
- @Override
- public Iterator iterator() {
- return new ConcreteIterator(items);
- }
-}
-```
-```java
-import java.util.Iterator;
-
-public class ConcreteIterator implements Iterator {
-
- private int[] items;
- private int position = 0;
-
- public ConcreteIterator(int[] items) {
- this.items = items;
- }
-
- @Override
- public boolean hasNext() {
- return position < items.length;
- }
-
- @Override
- public Integer next() {
- return items[position++];
- }
-}
-```
-```java
-public class Client {
- public static void main(String[] args) {
- Aggregate aggregate = new Aggregate();
- for (int item : aggregate) {
- System.out.println(item);
- }
- }
-}
-```
-
-## 3. 组合模式
-
-**1. 设计原则**
-
-一个类应该只有一个引起改变的原因。
-
-**2. 模式定义**
-
-允许将对象组合成树形结构来表现“整体/部分”层次结构。
-
-组合能让客户以一致的方式处理个别对象以及组合对象。
-
-**3. 模式类图**
-
-组件(Component)类是组合类(Composite)和叶子类(Leaf)的父类,可以把组合类看成是树的中间节点。
-
-组合类拥有一个组件对象,因此组合类的操作可以委托给组件对象去处理,而组件对象可以是另一个组合类或者叶子类。
-
-
-
-**4. 代码实现**
-
-```java
-public abstract class Component {
- protected String name;
-
- public Component(String name) {
- this.name = name;
- }
-
- abstract public void addChild(Component component);
-
- public void print() {
- print(0);
- }
-
- abstract protected void print(int level);
-}
-```
-
-```java
-public class Leaf extends Component {
- public Leaf(String name) {
- super(name);
- }
-
- @Override
- public void addChild(Component component) {
- throw new UnsupportedOperationException(); // 牺牲透明性换取单一职责原则,这样就不用考虑是叶子节点还是组合节点
- }
-
- @Override
- protected void print(int level) {
- for (int i = 0; i < level; i++) {
- System.out.print("--");
- }
- System.out.println("left:" + name);
- }
-}
-```
-
-```java
-public class Composite extends Component {
-
- private List childs;
-
- public Composite(String name) {
- super(name);
- childs = new ArrayList<>();
- }
-
- @Override
- public void addChild(Component component) {
- childs.add(component);
- }
-
- @Override
- protected void print(int level) {
- for (int i = 0; i < level; i++) {
- System.out.print("--");
- }
- System.out.println("Composite:" + name);
- for (Component component : childs) {
- component.print(level + 1);
- }
- }
-}
-```
+如果存在下面这种代码,就需要使用简单工厂将对象实例化的部分放到简单工厂中。
```java
public class Client {
public static void main(String[] args) {
- Composite root = new Composite("root");
- Component node1 = new Leaf("1");
- Component node2 = new Composite("2");
- Component node3 = new Leaf("3");
- root.addChild(node1);
- root.addChild(node2);
- root.addChild(node3);
- Component node21 = new Leaf("21");
- Component node22 = new Composite("22");
- node2.addChild(node21);
- node2.addChild(node22);
- Component node221 = new Leaf("221");
- node22.addChild(node221);
- root.print();
- }
-}
-```
-运行结果
-
-```html
-Composite:root
---left:1
---Composite:2
-----left:21
-----Composite:22
-------left:221
---left:3
-```
-
-# 第十章 状态模式
-
-**1. 模式定义**
-
-允许对象在内部状态改变时改变它的行为,对象看起来好像修改了它所属的类。
-
-**2. 模式类图**
-
-Context 的 request() 方法委托给 State 对象去处理。当 Context 组合的 State 对象发生改变时,它的行为也就发生了改变。
-
-
-
-**3. 与策略模式的比较**
-
-状态模式的类图和策略模式一样,并且都是能够动态改变对象的行为。
-
-但是状态模式是通过状态转移来改变 Context 所组合的 State 对象,而策略模式是通过 Context 本身的决策来改变组合的 Strategy 对象。
-
-所谓的状态转移,是指 Context 在运行过程中由于一些条件发生改变而使得 State 对象发生改变,主要必须要是在运行过程中。
-
-状态模式主要是用来解决状态转移的问题,当状态发生庄毅了,那么 Context 对象就会改变它的行为;而策略模式主要是用来封装一组可以互相替代的算法族,并且可以根据需要动态地去替换 Context 需要使用哪个算法。
-
-**4. 问题描述**
-
-糖果销售机有多种状态,每种状态下销售机有不同的行为,状态可以发生转移,使得销售机的行为也发生改变。
-
-
-
-**5. 直接解决方案**
-
-在糖果机的每个操作函数里面,判断当前的状态,根据不同的状态进行不同的处理,并且发生不同的状态转移。
-
-这种解决方案在需要增加状态的时候,必须对每个操作的代码都进行修改。
-
-
-
-**6 代码实现**
-
-糖果销售机即 Context。
-
-下面的实现中每个 State 都组合了 Context 对象,这是因为状态转移的操作在 State 对象中,而状态转移过程又必须改变 Context 对象的 state 对象,因此 State 必须拥有 Context 对象。
-
-```java
-public interface State {
- /**
- * 投入25 分钱
- */
- void insertQuarter();
-
- /**
- * 退回25 分钱
- */
- void ejectQuarter();
-
- /**
- * 转动曲柄
- */
- void turnCrank();
-
- /**
- * 发放糖果
- */
- void dispense();
-}
-```
-```java
-public class HasQuarterState implements State{
- private GumballMachine gumballMachine;
-
- public HasQuarterState(GumballMachine gumballMachine){
- this.gumballMachine = gumballMachine;
- }
-
- @Override
- public void insertQuarter() {
- System.out.println("You can't insert another quarter");
- }
-
- @Override
- public void ejectQuarter() {
- System.out.println("Quarter returned");
- gumballMachine.setState(gumballMachine.getNoQuarterState());
- }
-
- @Override
- public void turnCrank() {
- System.out.println("You turned...");
- gumballMachine.setState(gumballMachine.getSoldState());
- }
-
- @Override
- public void dispense() {
- System.out.println("No gumball dispensed");
- }
-}
-```
-```java
-public class NoQuarterState implements State {
- GumballMachine gumballMachine;
-
- public NoQuarterState(GumballMachine gumballMachine) {
- this.gumballMachine = gumballMachine;
- }
-
- @Override
- public void insertQuarter() {
- System.out.println("You insert a quarter");
- gumballMachine.setState(gumballMachine.getHasQuarterState());
- }
-
- @Override
- public void ejectQuarter() {
- System.out.println("You haven't insert a quarter");
- }
-
- @Override
- public void turnCrank() {
- System.out.println("You turned, but there's no quarter");
- }
-
- @Override
- public void dispense() {
- System.out.println("You need to pay first");
- }
-}
-```
-```java
-public class SoldOutState implements State {
-
- GumballMachine gumballMachine;
-
- public SoldOutState(GumballMachine gumballMachine) {
- this.gumballMachine = gumballMachine;
- }
-
- @Override
- public void insertQuarter() {
- System.out.println("You can't insert a quarter, the machine is sold out");
- }
-
- @Override
- public void ejectQuarter() {
- System.out.println("You can't eject, you haven't inserted a quarter yet");
- }
-
- @Override
- public void turnCrank() {
- System.out.println("You turned, but there are no gumballs");
- }
-
- @Override
- public void dispense() {
- System.out.println("No gumball dispensed");
- }
-}
-```
-```java
-public class SoldState implements State {
- GumballMachine gumballMachine;
-
- public SoldState(GumballMachine gumballMachine) {
- this.gumballMachine = gumballMachine;
- }
-
- @Override
- public void insertQuarter() {
- System.out.println("Please wait, we're already giving you a gumball");
- }
-
- @Override
- public void ejectQuarter() {
- System.out.println("Sorry, you already turned the crank");
- }
-
- @Override
- public void turnCrank() {
- System.out.println("Turning twice doesn't get you another gumball!");
- }
-
- @Override
- public void dispense() {
- gumballMachine.releaseBall();
- if(gumballMachine.getCount()>0){
- gumballMachine.setState(gumballMachine.getNoQuarterState());
- } else{
- System.out.println("Oops, out of gumballs");
- gumballMachine.setState(gumballMachine.getSoldOutState());
- }
- }
-}
-```
-```java
-public class GumballMachine {
- private State soldOutState;
- private State noQuarterState;
- private State hasQuarterState;
- private State soldState;
-
- private State state;
- private int count = 0;
-
- public GumballMachine(int numberGumballs) {
- count = numberGumballs;
- soldOutState = new SoldOutState(this);
- noQuarterState = new NoQuarterState(this);
- hasQuarterState = new HasQuarterState(this);
- soldState = new SoldState(this);
-
- if (numberGumballs > 0) {
- state = noQuarterState;
+ int type = 1;
+ Product product;
+ if (type == 1) {
+ product = new ConcreteProduct1();
+ } else if (type == 2) {
+ product = new ConcreteProduct2();
} else {
- state = soldOutState;
+ product = new ConcreteProduct();
}
}
-
- public void insertQuarter() {
- state.insertQuarter();
- }
-
- public void ejectQuarter() {
- state.ejectQuarter();
- }
-
- public void turnCrank() {
- state.turnCrank();
- state.dispense();
- }
-
- public void setState(State state) {
- this.state = state;
- }
-
- public void releaseBall() {
- System.out.println("A gumball comes rolling out the slot...");
- if (count != 0) {
- count -= 1;
- }
- }
-
- public State getSoldOutState() {
- return soldOutState;
- }
-
- public State getNoQuarterState() {
- return noQuarterState;
- }
-
- public State getHasQuarterState() {
- return hasQuarterState;
- }
-
- public State getSoldState() {
- return soldState;
- }
-
- public int getCount() {
- return count;
- }
}
```
+
+## 实现
+
```java
-public class GumballMachineTestDrive {
- public static void main(String[] args) {
- GumballMachine gumballMachine = new GumballMachine(5);
+public interface Product {
+}
+```
- gumballMachine.insertQuarter();
- gumballMachine.turnCrank();
+```java
+public class ConcreteProduct implements Product{
+}
+```
- gumballMachine.insertQuarter();
- gumballMachine.ejectQuarter();
- gumballMachine.turnCrank();
+```java
+public class ConcreteProduct1 implements Product{
+}
+```
- gumballMachine.insertQuarter();
- gumballMachine.turnCrank();
- gumballMachine.insertQuarter();
- gumballMachine.turnCrank();
- gumballMachine.ejectQuarter();
+```java
+public class ConcreteProduct2 implements Product{
+}
+```
- gumballMachine.insertQuarter();
- gumballMachine.insertQuarter();
- gumballMachine.turnCrank();
- gumballMachine.insertQuarter();
- gumballMachine.turnCrank();
- gumballMachine.insertQuarter();
- gumballMachine.turnCrank();
+```java
+public class SimpleFactory {
+ public Product createProduct(int type) {
+ if (type == 1) {
+ return new ConcreteProduct1();
+ } else if (type == 2) {
+ return new ConcreteProduct2();
+ }
+ return new ConcreteProduct();
}
}
```
-运行结果
-```html
-You insert a quarter
-You turned...
-A gumball comes rolling out the slot...
-You insert a quarter
-Quarter returned
-You turned, but there's no quarter
-You need to pay first
-You insert a quarter
-You turned...
-A gumball comes rolling out the slot...
-You insert a quarter
-You turned...
-A gumball comes rolling out the slot...
-You haven't insert a quarter
-You insert a quarter
-You can't insert another quarter
-You turned...
-A gumball comes rolling out the slot...
-You insert a quarter
-You turned...
-A gumball comes rolling out the slot...
-Oops, out of gumballs
-You can't insert a quarter, the machine is sold out
-You turned, but there are no gumballs
-No gumball dispensed
+
+```java
+public class Client {
+ public static void main(String[] args) {
+ SimpleFactory simpleFactory = new SimpleFactory();
+ Product product = simpleFactory.createProduct(1);
+ }
+}
```
-# 第十一章 代理模式
+# 四、工厂方法模式
-# 第十二章 复合模式
+## 意图
-## MVC
+定义了一个创建对象的接口,但由子类决定要实例化哪个类。工厂方法把实例化推迟到子类。
-**传统 MVC**
+## 类图
-视图使用组合模式,模型使用了观察者模式,控制器使用了策略模式。
+在简单工厂中,创建对象的是另一个类,而在工厂方法中,是由子类来创建对象。
-
+下图中,Factory 有一个 doSomethind() 方法,这个方法需要用到一组产品对象,这组产品对象由 factoryMethod() 方法创建。该方法是抽象的,需要由子类去实现。
-**Web 中的 MVC**
+
-模式不再使用观察者模式。
+## 实现
-
+```java
+public abstract class Factory {
+ abstract public Product factoryMethod();
+ public void doSomethind() {
+ Product product = factoryMethod();
+ // do something with the product
+ }
+}
+```
-# 第十三章 与设计模式相处
+```java
+public class ConcreteFactory extends Factory {
+ public Product factoryMethod() {
+ return new ConcreteProduct();
+ }
+}
+```
-定义:在某 **情境** 下,针对某 **问题** 的某种 **解决方案**。
+```java
+public class ConcreteFactory1 extends Factory{
+ public Product factoryMethod() {
+ return new ConcreteProduct1();
+ }
+}
+```
-过度使用设计模式可能导致代码被过度工程化,应该总是用最简单的解决方案完成工作,并在真正需要模式的地方才使用它。
+```java
+public class ConcreteFactory2 extends Factory {
+ public Product factoryMethod() {
+ return new ConcreteProduct2();
+ }
+}
+```
-反模式:不好的解决方案来解决一个问题。主要作用是为了警告不要使用这些解决方案。
+# 五、抽象工厂模式
-模式分类:
+## 意图
-
+提供一个接口,用于创建 **相关的对象家族** 。
-# 第十四章 剩下的模式
+## 类图
+
+
+
+抽象工厂模式创建的是对象家族,也就是很多对象而不是一个对象,并且这些对象是相关的,也就是说必须一起创建出来。而工厂模式只是用于创建一个对象,这和抽象工厂模式有很大不同。
+
+抽象工厂模式用到了工厂模式来创建单一对象,AbstractFactory 中的 createProductA 和 createProductB 方法都是让子类来实现,这两个方法单独来看就是在创建一个对象,这符合工厂模式的定义。
+
+至于创建对象的家族这一概念是在 Client 体现,Client 要通过 AbstractFactory 同时调用两个方法来创建出两个对象,在这里这两个对象就有很大的相关性,Client 需要同时创建出这两个对象。
+
+从高层次来看,抽象工厂使用了组合,即 Cilent 组合了 AbstractFactory,而工厂模式使用了继承。
+
+## 代码实现
+
+```java
+public class AbstractProductA {
+}
+```
+
+```java
+public class AbstractProductB {
+}
+```
+
+```java
+public class ProductA1 extends AbstractProductA {
+}
+```
+
+```java
+public class ProductA2 extends AbstractProductA {
+}
+```
+
+```java
+public class ProductB1 extends AbstractProductB{
+}
+```
+
+```java
+public class ProductB2 extends AbstractProductB{
+}
+```
+
+```java
+public abstract class AbstractFactory {
+ abstract AbstractProductA createProductA();
+ abstract AbstractProductB createProductB();
+}
+```
+
+```java
+public class ConcreteFactory1 extends AbstractFactory{
+ AbstractProductA createProductA() {
+ return new ProductA1();
+ }
+
+ AbstractProductB createProductB() {
+ return new ProductB1();
+ }
+}
+```
+
+```java
+public class ConcreteFactory2 extends AbstractFactory {
+ AbstractProductA createProductA() {
+ return new ProductA2();
+ }
+
+ AbstractProductB createProductB() {
+ return new ProductB2();
+ }
+}
+```
+
+```java
+public class Client {
+ public static void main(String[] args) {
+ AbstractFactory abstractFactory = new ConcreteFactory1();
+ AbstractProductA productA = abstractFactory.createProductA();
+ abstractFactory = new ConcreteFactory2();
+ productA = abstractFactory.createProductA();
+ }
+}
+```
+
+# 参考资料
+
+- 弗里曼. Head First 设计模式 [M]. 中国电力出版社, 2007.
+- [Design Patterns](http://www.oodesign.com/)
+- [Design patterns implemented in Java](http://java-design-patterns.com/)
diff --git a/notes/重构.md b/notes/重构.md
index 8cf08472..e74d3cc5 100644
--- a/notes/重构.md
+++ b/notes/重构.md
@@ -1,6 +1,6 @@
-* [第一个案例](#第一个案例)
-* [重构原则](#重构原则)
+* [一、第一个案例](#一第一个案例)
+* [二、重构原则](#二重构原则)
* [定义](#定义)
* [为何重构](#为何重构)
* [三次法则](#三次法则)
@@ -9,108 +9,109 @@
* [何时不该重构](#何时不该重构)
* [重构与设计](#重构与设计)
* [重构与性能](#重构与性能)
-* [代码的坏味道](#代码的坏味道)
- * [1. Duplicated Code(重复代码)](#1-duplicated-code重复代码)
- * [2. Long Method(过长函数)](#2-long-method过长函数)
- * [3. Large Class(过大的类)](#3-large-class过大的类)
- * [4. Long Parameter List(过长的参数列表)](#4-long-parameter-list过长的参数列表)
- * [5. Divergent Change(发散式变化)](#5-divergent-change发散式变化)
- * [6. Shotgun Surgery(散弹式修改)](#6-shotgun-surgery散弹式修改)
- * [7. Feature Envy(依恋情结)](#7-feature-envy依恋情结)
- * [8. Data Clumps(数据泥团)](#8-data-clumps数据泥团)
- * [9. Primitive Obsession(基本类型偏执)](#9-primitive-obsession基本类型偏执)
- * [10. Switch Statements(switch 惊悚现身)](#10-switch-statementsswitch-惊悚现身)
- * [11. Parallel Inheritance Hierarchies(平行继承体系)](#11-parallel-inheritance-hierarchies平行继承体系)
- * [12. Lazy Class(冗余类)](#12-lazy-class冗余类)
- * [13. Speculative Generality(夸夸其谈未来性)](#13-speculative-generality夸夸其谈未来性)
- * [14. Temporary Field(令人迷惑的暂时字段)](#14-temporary-field令人迷惑的暂时字段)
- * [15. Message Chains(过度耦合的消息链)](#15-message-chains过度耦合的消息链)
- * [16. Middle Man(中间人)](#16-middle-man中间人)
- * [17. Inappropriate Intimacy(狎昵关系)](#17-inappropriate-intimacy狎昵关系)
- * [18. Alernative Classes with Different Interfaces(异曲同工的类)](#18-alernative-classes-with-different-interfaces异曲同工的类)
- * [19. Incomplete Library Class(不完美的类库)](#19-incomplete-library-class不完美的类库)
- * [20. Data Class(幼稚的数据类)](#20-data-class幼稚的数据类)
- * [21. Refused Bequest(被拒绝的馈赠)](#21-refused-bequest被拒绝的馈赠)
- * [22. Comments(过多的注释)](#22-comments过多的注释)
-* [构筑测试体系](#构筑测试体系)
-* [重新组织函数](#重新组织函数)
- * [1. Extract Method(提炼函数)](#1-extract-method提炼函数)
- * [2. Inline Method(内联函数)](#2-inline-method内联函数)
- * [3. Inline Temp(内联临时变量)](#3-inline-temp内联临时变量)
- * [4. Replace Temp with Query(以查询取代临时变量)](#4-replace-temp-with-query以查询取代临时变量)
- * [5. Introduce Explaining Variable(引起解释变量)](#5-introduce-explaining-variable引起解释变量)
- * [6. Split Temporary Variable(分解临时变量)](#6-split-temporary-variable分解临时变量)
- * [7. Remove Assigments to Parameters(移除对参数的赋值)](#7-remove-assigments-to-parameters移除对参数的赋值)
- * [8. Replace Method with Method Object(以函数对象取代函数)](#8-replace-method-with-method-object以函数对象取代函数)
- * [9. Subsititute Algorithn(替换算法)](#9-subsititute-algorithn替换算法)
-* [在对象之间搬移特性](#在对象之间搬移特性)
- * [1. Move Method(搬移函数)](#1-move-method搬移函数)
- * [2. Move Field(搬移字段)](#2-move-field搬移字段)
- * [3. Extract Class(提炼类)](#3-extract-class提炼类)
- * [4. Inline Class(将类内联化)](#4-inline-class将类内联化)
- * [5. Hide Delegate(隐藏“委托关系”)](#5-hide-delegate隐藏“委托关系”)
- * [6. Remove Middle Man(移除中间人)](#6-remove-middle-man移除中间人)
- * [7. Introduce Foreign Method(引入外加函数)](#7-introduce-foreign-method引入外加函数)
- * [8. Introduce Local Extension(引入本地扩展)](#8-introduce-local-extension引入本地扩展)
-* [重新组织数据](#重新组织数据)
- * [1. Self Encapsulate Field(自封装字段)](#1-self-encapsulate-field自封装字段)
- * [2. Replace Data Value with Object(以对象取代数据值)](#2-replace-data-value-with-object以对象取代数据值)
- * [3. Change Value to Reference(将值对象改成引用对象)](#3-change-value-to-reference将值对象改成引用对象)
- * [4. Change Reference to value(将引用对象改为值对象)](#4-change-reference-to-value将引用对象改为值对象)
- * [5. Replace Array with Object(以对象取代数组)](#5-replace-array-with-object以对象取代数组)
- * [6. Duplicate Observed Data(赋值“被监视数据”)](#6-duplicate-observed-data赋值“被监视数据”)
- * [7. Change Unidirectional Association to Bidirectional(将单向关联改为双向关联)](#7-change-unidirectional-association-to-bidirectional将单向关联改为双向关联)
- * [8. Change Bidirectional Association to Unidirectional(将双向关联改为单向关联)](#8-change-bidirectional-association-to-unidirectional将双向关联改为单向关联)
- * [9. Replace Magic Number with Symbolic Constant(以字面常量取代魔法数)](#9-replace-magic-number-with-symbolic-constant以字面常量取代魔法数)
- * [10. Encapsulate Field(封装字段)](#10-encapsulate-field封装字段)
- * [11. Encapsulate Collection(封装集合)](#11-encapsulate-collection封装集合)
- * [12. Replace Record with Data Class(以数据类取代记录)](#12-replace-record-with-data-class以数据类取代记录)
- * [13. Replace Type Code with Class(以类取代类型码)](#13-replace-type-code-with-class以类取代类型码)
- * [14. Replace Type Code with Subcalsses(以子类取代类型码)](#14-replace-type-code-with-subcalsses以子类取代类型码)
- * [15. Replace Type Code with State/Strategy (以 State/Strategy 取代类型码)](#15-replace-type-code-with-statestrategy-以-statestrategy-取代类型码)
- * [16. Replace Subclass with Fields(以字段取代子类)](#16-replace-subclass-with-fields以字段取代子类)
-* [简化条件表达式](#简化条件表达式)
- * [1. Decompose Conditional(分解条件表达式)](#1-decompose-conditional分解条件表达式)
- * [2. Consolidate Conditional Expression(合并条件表达式)](#2-consolidate-conditional-expression合并条件表达式)
- * [3. Consolidate Duplicate Conditional Fragments (合并重复的条件片段)](#3-consolidate-duplicate-conditional-fragments-合并重复的条件片段)
- * [4. Remove Control Flag(移除控制标记)](#4-remove-control-flag移除控制标记)
- * [5. Replace Nested Conditional with Guard Clauses (以卫语句取代嵌套条件表达式)](#5-replace-nested-conditional-with-guard-clauses-以卫语句取代嵌套条件表达式)
- * [6. Replace Conditional with Polymorphism (以多态取代条件表达式)](#6-replace-conditional-with-polymorphism-以多态取代条件表达式)
- * [7. Introduce Null Object(引入Null对象)](#7-introduce-null-object引入null对象)
- * [8. Introduce Assertion(引入断言)](#8-introduce-assertion引入断言)
-* [简化函数调用](#简化函数调用)
- * [1. Rename Method(函数改名)](#1-rename-method函数改名)
- * [2. Add Parameter(添加参数)](#2-add-parameter添加参数)
- * [3. Remove Parameter(移除参数)](#3-remove-parameter移除参数)
- * [4. Separate Query from Modifier(将查询函数和修改函数分离)](#4-separate-query-from-modifier将查询函数和修改函数分离)
- * [5. Parameterize Method(令函数携带参数)](#5-parameterize-method令函数携带参数)
- * [6. Replace Parameter with Explicit Methods(以明确函数取代参数)](#6-replace-parameter-with-explicit-methods以明确函数取代参数)
- * [7. Preserve Whole Object(保持对象完整)](#7-preserve-whole-object保持对象完整)
- * [8. Replace Parameter with Methods(以函数取代参数)](#8-replace-parameter-with-methods以函数取代参数)
- * [9. Introduce Parameter Object(引入参数对象)](#9-introduce-parameter-object引入参数对象)
- * [10. Remove Setting Method(移除设值函数)](#10-remove-setting-method移除设值函数)
- * [11. Hide Method(隐藏函数)](#11-hide-method隐藏函数)
- * [12. Replace Constructor with Factory Method (以工厂函数取代构造函数)](#12-replace-constructor-with-factory-method-以工厂函数取代构造函数)
- * [13. Encapsulate Downcast(封装向下转型)](#13-encapsulate-downcast封装向下转型)
- * [14. Replace Error Code with Exception (以异常取代错误码)](#14-replace-error-code-with-exception-以异常取代错误码)
- * [15. Replace Exception with Test(以测试取代异常)](#15-replace-exception-with-test以测试取代异常)
-* [处理概括关系](#处理概括关系)
- * [1. Pull Up Field(字段上移)](#1-pull-up-field字段上移)
- * [2. Pull Up Method(函数上移)](#2-pull-up-method函数上移)
- * [3. Pull Up Constructor Body(构造函数本体上移)](#3-pull-up-constructor-body构造函数本体上移)
- * [4. Push Down Method(函数下移)](#4-push-down-method函数下移)
- * [5. Push Down Field(字段下移)](#5-push-down-field字段下移)
- * [6. Extract Subclass(提炼子类)](#6-extract-subclass提炼子类)
- * [7. Extract Superclass(提炼超类)](#7-extract-superclass提炼超类)
- * [8. Extract Interface(提炼接口)](#8-extract-interface提炼接口)
- * [9. Collapse Hierarchy(折叠继承体系)](#9-collapse-hierarchy折叠继承体系)
- * [10. Form Template Method(塑造模板函数)](#10-form-template-method塑造模板函数)
- * [11. Replace Inheritance with Delegation (以委托取代继承)](#11-replace-inheritance-with-delegation-以委托取代继承)
- * [12. Replace Delegation with Inheritance (以继承取代委托)](#12-replace-delegation-with-inheritance-以继承取代委托)
+* [三、代码的坏味道](#三代码的坏味道)
+ * [1. 重复代码](#1-重复代码)
+ * [2. 过长函数](#2-过长函数)
+ * [3. 过大的类](#3-过大的类)
+ * [4. 过长的参数列表](#4-过长的参数列表)
+ * [5. 发散式变化](#5-发散式变化)
+ * [6. 散弹式修改](#6-散弹式修改)
+ * [7. 依恋情结](#7-依恋情结)
+ * [8. 数据泥团](#8-数据泥团)
+ * [9. 基本类型偏执](#9-基本类型偏执)
+ * [10. switch 惊悚现身](#10-switch-惊悚现身)
+ * [11. 平行继承体系](#11-平行继承体系)
+ * [12. 冗余类](#12-冗余类)
+ * [13. 夸夸其谈未来性](#13-夸夸其谈未来性)
+ * [14. 令人迷惑的暂时字段](#14-令人迷惑的暂时字段)
+ * [15. 过度耦合的消息链](#15-过度耦合的消息链)
+ * [16. 中间人](#16-中间人)
+ * [17. 狎昵关系](#17-狎昵关系)
+ * [18. 异曲同工的类](#18-异曲同工的类)
+ * [19. 不完美的类库](#19-不完美的类库)
+ * [20. 幼稚的数据类](#20-幼稚的数据类)
+ * [21. 被拒绝的馈赠](#21-被拒绝的馈赠)
+ * [22. 过多的注释](#22-过多的注释)
+* [四、构筑测试体系](#四构筑测试体系)
+* [五、重新组织函数](#五重新组织函数)
+ * [1. 提炼函数](#1-提炼函数)
+ * [2. 内联函数](#2-内联函数)
+ * [3. 内联临时变量](#3-内联临时变量)
+ * [4. 以查询取代临时变量](#4-以查询取代临时变量)
+ * [5. 引起解释变量](#5-引起解释变量)
+ * [6. 分解临时变量](#6-分解临时变量)
+ * [7. 移除对参数的赋值](#7-移除对参数的赋值)
+ * [8. 以函数对象取代函数](#8-以函数对象取代函数)
+ * [9. 替换算法](#9-替换算法)
+* [六、在对象之间搬移特性](#六在对象之间搬移特性)
+ * [1. 搬移函数](#1-搬移函数)
+ * [2. 搬移字段](#2-搬移字段)
+ * [3. 提炼类](#3-提炼类)
+ * [4. 将类内联化](#4-将类内联化)
+ * [5. 隐藏委托关系](#5-隐藏委托关系)
+ * [6. 移除中间人](#6-移除中间人)
+ * [7. 引入外加函数](#7-引入外加函数)
+ * [8. 引入本地扩展](#8-引入本地扩展)
+* [七、重新组织数据](#七重新组织数据)
+ * [1. 自封装字段](#1-自封装字段)
+ * [2. 以对象取代数据值](#2-以对象取代数据值)
+ * [3. 将值对象改成引用对象](#3-将值对象改成引用对象)
+ * [4. 将引用对象改为值对象](#4-将引用对象改为值对象)
+ * [5. 以对象取代数组](#5-以对象取代数组)
+ * [6. 赋值被监视数据](#6-赋值被监视数据)
+ * [7. 将单向关联改为双向关联](#7-将单向关联改为双向关联)
+ * [8. 将双向关联改为单向关联](#8-将双向关联改为单向关联)
+ * [9. 以字面常量取代魔法数](#9-以字面常量取代魔法数)
+ * [10. 封装字段](#10-封装字段)
+ * [11. 封装集合](#11-封装集合)
+ * [12. 以数据类取代记录](#12-以数据类取代记录)
+ * [13. 以类取代类型码](#13-以类取代类型码)
+ * [14. 以子类取代类型码](#14-以子类取代类型码)
+ * [15. 以 State/Strategy 取代类型码](#15-以-statestrategy-取代类型码)
+ * [16. 以字段取代子类](#16-以字段取代子类)
+* [八、简化条件表达式](#八简化条件表达式)
+ * [1. 分解条件表达式](#1-分解条件表达式)
+ * [2. 合并条件表达式](#2-合并条件表达式)
+ * [3. 合并重复的条件片段](#3-合并重复的条件片段)
+ * [4. 移除控制标记](#4-移除控制标记)
+ * [5. 以卫语句取代嵌套条件表达式](#5-以卫语句取代嵌套条件表达式)
+ * [6. 以多态取代条件表达式](#6-以多态取代条件表达式)
+ * [7. 引入 Null 对象](#7-引入-null-对象)
+ * [8. 引入断言](#8-引入断言)
+* [九、简化函数调用](#九简化函数调用)
+ * [1. 函数改名](#1-函数改名)
+ * [2. 添加参数](#2-添加参数)
+ * [3. 移除参数](#3-移除参数)
+ * [4. 将查询函数和修改函数分离](#4-将查询函数和修改函数分离)
+ * [5. 令函数携带参数](#5-令函数携带参数)
+ * [6. 以明确函数取代参数](#6-以明确函数取代参数)
+ * [7. 保持对象完整](#7-保持对象完整)
+ * [8. 以函数取代参数](#8-以函数取代参数)
+ * [9. 引入参数对象](#9-引入参数对象)
+ * [10. 移除设值函数](#10-移除设值函数)
+ * [11. 隐藏函数](#11-隐藏函数)
+ * [12. 以工厂函数取代构造函数](#12-以工厂函数取代构造函数)
+ * [13. 封装向下转型](#13-封装向下转型)
+ * [14. 以异常取代错误码](#14-以异常取代错误码)
+ * [15. 以测试取代异常](#15-以测试取代异常)
+* [十、处理概括关系](#十处理概括关系)
+ * [1. 字段上移](#1-字段上移)
+ * [2. 函数上移](#2-函数上移)
+ * [3. 构造函数本体上移](#3-构造函数本体上移)
+ * [4. 函数下移](#4-函数下移)
+ * [5. 字段下移](#5-字段下移)
+ * [6. 提炼子类](#6-提炼子类)
+ * [7. 提炼超类](#7-提炼超类)
+ * [8. 提炼接口](#8-提炼接口)
+ * [9. 折叠继承体系](#9-折叠继承体系)
+ * [10. 塑造模板函数](#10-塑造模板函数)
+ * [11. 以委托取代继承](#11-以委托取代继承)
+ * [12. 以继承取代委托](#12-以继承取代委托)
+* [参考资料](#参考资料)
-# 第一个案例
+# 一、第一个案例
如果你发现自己需要为程序添加一个特性,而代码结构使你无法很方便地达成目的,那就先重构这个程序。
@@ -130,7 +131,6 @@
可以发现,该代码没有使用 Customer 类中的任何信息,更多的是使用 Rental 类的信息,因此第一个可以重构的点就是把具体计费的代码移到 Rental 类中,然后 Customer 类的 getTotalCharge() 方法只需要调用 Rental 类中的计费方法即可。
-
```java
class Customer...
double getTotalCharge() {
@@ -173,7 +173,7 @@ double getTotalCharge() {
-# 重构原则
+# 二、重构原则
## 定义
@@ -229,13 +229,15 @@ double getTotalCharge() {
应当只关注关键代码的性能,因为只有一小部分的代码是关键代码。
-# 代码的坏味道
+# 三、代码的坏味道
本章主要介绍一些不好的代码,也就是说这些代码应该被重构。
文中提到的具体重构原则可以先忽略。
-## 1. Duplicated Code(重复代码)
+## 1. 重复代码
+
+> Duplicated Code
同一个类的两个函数有相同表达式,则用 Extract Method 提取出重复代码;
@@ -245,7 +247,9 @@ double getTotalCharge() {
如果两个毫不相关的类出现重复代码,则使用 Extract Class 方法将重复代码提取到一个独立类中。
-## 2. Long Method(过长函数)
+## 2. 过长函数
+
+> Long Method
函数应该尽可能小,因为小函数具有解释能力、共享能力、选择能力。
@@ -255,111 +259,151 @@ Extract Method 会把很多参数和临时变量都当做参数,可以用 Repl
条件和循环语句往往也需要提取到新的函数中。
-## 3. Large Class(过大的类)
+## 3. 过大的类
+
+> Large Class
应该尽可能让一个类只做一件事,而过大的类做了过多事情,需要使用 Extract Class 或 Extract Subclass。
先确定客户端如何使用该类,然后运用 Extract Interface 为每一种使用方式提取出一个接口。
-## 4. Long Parameter List(过长的参数列表)
+## 4. 过长的参数列表
+
+> Long Parameter List
太长的参数列表往往会造成前后不一致,不易使用。
面向对象程序中,函数所需要的数据通常内在宿主类中找到。
-## 5. Divergent Change(发散式变化)
+## 5. 发散式变化
+
+> Divergent Change
设计原则:一个类应该只有一个引起改变的原因。也就是说,针对某一外界变化所有相应的修改,都只应该发生在单一类中。
针对某种原因的变化,使用 Extract Class 将它提炼到一个类中。
-## 6. Shotgun Surgery(散弹式修改)
+## 6. 散弹式修改
+
+> Shotgun Surgery
一个变化引起多个类修改;
使用 Move Method 和 Move Field 把所有需要修改的代码放到同一个类中。
-## 7. Feature Envy(依恋情结)
+## 7. 依恋情结
+
+> Feature Envy
一个函数对某个类的兴趣高于对自己所处类的兴趣,通常是过多访问其它类的数据,
使用 Move Method 将它移到该去的地方,如果对多个类都有 Feature Envy,先用 Extract Method 提取出多个函数。
-## 8. Data Clumps(数据泥团)
+## 8. 数据泥团
+
+> Data Clumps
有些数据经常一起出现,比如两个类具有相同的字段、许多函数有相同的参数,这些绑定在一起出现的数据应该拥有属于它们自己的对象。
使用 Extract Class 将它们放在一起。
-## 9. Primitive Obsession(基本类型偏执)
+## 9. 基本类型偏执
+
+> Primitive Obsession
使用类往往比使用基本类型更好,使用 Replace Data Value with Object 将数据值替换为对象。
-## 10. Switch Statements(switch 惊悚现身)
+## 10. switch 惊悚现身
+
+> Switch Statements
具体参见第一章的案例。
-## 11. Parallel Inheritance Hierarchies(平行继承体系)
+## 11. 平行继承体系
+
+> Parallel Inheritance Hierarchies
每当为某个类增加一个子类,必须也为另一个类相应增加一个子类。
这种结果会带来一些重复性,消除重复性的一般策略:让一个继承体系的实例引用另一个继承体系的实例。
-## 12. Lazy Class(冗余类)
+## 12. 冗余类
+
+> Lazy Class
如果一个类没有做足够多的工作,就应该消失。
-## 13. Speculative Generality(夸夸其谈未来性)
+## 13. 夸夸其谈未来性
+
+> Speculative Generality
有些内容是用来处理未来可能发生的变化,但是往往会造成系统难以理解和维护,并且预测未来可能发生的改变很可能和最开始的设想相反。因此,如果不是必要,就不要这么做。
-## 14. Temporary Field(令人迷惑的暂时字段)
+## 14. 令人迷惑的暂时字段
+
+> Temporary Field
某个字段仅为某种特定情况而设,这样的代码不易理解,因为通常认为对象在所有时候都需要它的所有字段。
把这种字段和特定情况的处理操作使用 Extract Class 提炼到一个独立类中。
-## 15. Message Chains(过度耦合的消息链)
+## 15. 过度耦合的消息链
+
+> Message Chains
一个对象请求另一个对象,然后再向后者请求另一个对象,然后...,这就是消息链。采用这种方式,意味着客户代码将与对象间的关系紧密耦合。
改用函数链,用函数委托另一个对象来处理。
-## 16. Middle Man(中间人)
+## 16. 中间人
+
+> Middle Man
中间人负责处理委托给它的操作,如果一个类中有过多的函数都委托给其它类,那就是过度运用委托,应当 Remove Middle Man,直接与负责的对象打交道。
-## 17. Inappropriate Intimacy(狎昵关系)
+## 17. 狎昵关系
+
+> Inappropriate Intimacy
两个类多于亲密,花费太多时间去探讨彼此的 private 成分。
-## 18. Alernative Classes with Different Interfaces(异曲同工的类)
+## 18. 异曲同工的类
+
+> Alernative Classes with Different Interfaces
两个函数做同一件事,却有着不同的签名。
使用 Rename Method 根据它们的用途重新命名。
-## 19. Incomplete Library Class(不完美的类库)
+## 19. 不完美的类库
+
+> Incomplete Library Class
类库的设计者不可能设计出完美的类库,当我们需要对类库进行一些修改时,可以使用以下两种方法:如果只是修改一两个函数,使用 Introduce Foreign Method;如果要添加一大堆额外行为,使用 Introduce Local Extension。
-## 20. Data Class(幼稚的数据类)
+## 20. 幼稚的数据类
+
+> Data Class
它只拥有一些数据字段,以及用于访问这些字段的函数,除此之外一无长物。
找出字段使用的地方,然后把相应的操作移到 Data Class 中。
-## 21. Refused Bequest(被拒绝的馈赠)
+## 21. 被拒绝的馈赠
+
+> Refused Bequest
子类不想继承超类的所有函数和数据。
为子类新建一个兄弟类,不需要的函数或数据使用 Push Down Method 和 Push Down Field 下推给那个兄弟。
-## 22. Comments(过多的注释)
+## 22. 过多的注释
+
+> Comments
使用 Extract Method 提炼出需要注释的部分,然后用函数名来解释函数的行为。
-# 构筑测试体系
+# 四、构筑测试体系
Java 可以使用 Junit 进行单元测试。
@@ -371,19 +415,25 @@ Java 可以使用 Junit 进行单元测试。
应当集中测试可能出错的边界条件。
-# 重新组织函数
+# 五、重新组织函数
-## 1. Extract Method(提炼函数)
+## 1. 提炼函数
+
+> Extract Method
将这段代码放进一个独立函数中,并让函数名称解释该函数的用途。
-## 2. Inline Method(内联函数)
+## 2. 内联函数
+
+> Inline Method
一个函数的本体与名称同样清楚易懂。
在函数调用点插入函数本体,然后移除该函数。
-## 3. Inline Temp(内联临时变量)
+## 3. 内联临时变量
+
+> Inline Temp
一个临时变量,只被简单表达式赋值一次,而它妨碍了其它重构手法。
@@ -398,7 +448,9 @@ return basePrice > 1000;
return anOrder.basePrice() > 1000;
```
-## 4. Replace Temp with Query(以查询取代临时变量)
+## 4. 以查询取代临时变量
+
+> Replace Temp with Query
以临时变量保存某一表达式的运算结果,将这个表达式提炼到一个独立函数中,将所有对临时变量的引用点替换为对新函数的调用。Replace Temp with Query 往往是 Extract Method 之前必不可少的一个步骤,因为局部变量会使代码难以提炼。
@@ -422,7 +474,9 @@ double basePrice(){
}
```
-## 5. Introduce Explaining Variable(引起解释变量)
+## 5. 引起解释变量
+
+> Introduce Explaining Variable
将复杂表达式(或其中一部分)的结果放进一个临时变量,以此变量名称来解释表达式用途。
@@ -444,13 +498,17 @@ if(isMacOS && isIEBrower && wasInitialized() && wasResized) {
}
```
-## 6. Split Temporary Variable(分解临时变量)
+## 6. 分解临时变量
+
+> Split Temporary Variable
某个临时变量被赋值超过一次,它既不是循环变量,也不是用于收集计算结果。
针对每次赋值,创造一个独立、对应的临时变量,每个临时变量只承担一个责任。
-## 7. Remove Assigments to Parameters(移除对参数的赋值)
+## 7. 移除对参数的赋值
+
+> Remove Assigments to Parameters
以一个临时变量取代对该参数的赋值。
@@ -465,37 +523,51 @@ int discount (int inputVal, int quentity, int yearToDate){
if (inputVal > 50) result -= 2;
```
-## 8. Replace Method with Method Object(以函数对象取代函数)
+## 8. 以函数对象取代函数
+
+> Replace Method with Method Object
当对一个大型函数采用 Extract Method 时,由于包含了局部变量使得很难进行该操作。
将这个函数放进一个单独对象中,如此一来局部变量就成了对象内的字段。然后可以在同一个对象中将这个大型函数分解为多个小型函数。
-## 9. Subsititute Algorithn(替换算法)
+## 9. 替换算法
-# 在对象之间搬移特性
+> Subsititute Algorithn
-## 1. Move Method(搬移函数)
+# 六、在对象之间搬移特性
+
+## 1. 搬移函数
+
+> Move Method
类中的某个函数与另一个类进行更多交流:调用后者或者被后者调用。
将这个函数搬移到另一个类中。
-## 2. Move Field(搬移字段)
+## 2. 搬移字段
+
+> Move Field
类中的某个字段被另一个类更多地用到,这里的用到是指调用取值设值函数,应当把该字段移到另一个类中。
-## 3. Extract Class(提炼类)
+## 3. 提炼类
+
+> Extract Class
某个类做了应当由两个类做的事。
应当建立一个新类,将相关的字段和函数从旧类搬移到新类。
-## 4. Inline Class(将类内联化)
+## 4. 将类内联化
+
+> Inline Class
与 Extract Class 相反。
-## 5. Hide Delegate(隐藏“委托关系”)
+## 5. 隐藏委托关系
+
+> Hide Delegate
建立所需的函数,隐藏委托关系。
@@ -531,57 +603,77 @@ public Person getManager(){
}
```
-## 6. Remove Middle Man(移除中间人)
+## 6. 移除中间人
+
+> Remove Middle Man
与 Hide Delegate 相反,本方法需要移除委托函数,让客户直接调用委托类。
Hide Delegate 有很大好处,但是它的代价是:每当客户要使用受托类的新特性时,就必须在服务器端添加一个简单的委托函数。随着受委托的特性越来越多,服务器类完全变成了一个“中间人”。
-## 7. Introduce Foreign Method(引入外加函数)
+## 7. 引入外加函数
+
+> Introduce Foreign Method
需要为提供服务的类添加一个函数,但是无法修改这个类。
可以在客户类中建立一个函数,并以第一参数形式传入一个服务类的实例,让客户类组合服务器实例。
-## 8. Introduce Local Extension(引入本地扩展)
+## 8. 引入本地扩展
+
+> Introduce Local Extension
和 Introduce Foreign Method 目的一样,但是 Introduce Local Extension 通过建立新的类来实现。有两种方式:子类或者包装类,子类就是通过继承实现,包装类就是通过组合实现。
-# 重新组织数据
+# 七、重新组织数据
-## 1. Self Encapsulate Field(自封装字段)
+## 1. 自封装字段
+
+> Self Encapsulate Field
为字段建立取值/设值函数,并用这些函数来访问字段。只有当子类想访问超类的一个字段,又想在子类中将对这个字段访问改为一个计算后的值,才使用这种方式,否则直接访问字段的方式简洁明了。
-## 2. Replace Data Value with Object(以对象取代数据值)
+## 2. 以对象取代数据值
+
+> Replace Data Value with Object
在开发初期,往往会用简单的数据项表示简单的情况,但是随着开发的进行,一些简单数据项会具有一些特殊行为。比如一开始会把电话号码存成字符串,但是随后发现电话号码需要“格式化”、“抽取区号”之类的特殊行为。
-## 3. Change Value to Reference(将值对象改成引用对象)
+## 3. 将值对象改成引用对象
+
+> Change Value to Reference
将彼此相等的实例替换为同一个对象。这就要用一个工厂来创建这种唯一对象,工厂类中需要保留一份已经创建对象的列表,当要创建一个对象时,先查找这份列表中是否已经存在该对象,如果存在,则返回列表中的这个对象;否则,新建一个对象,添加到列表中,并返回该对象。
-## 4. Change Reference to value(将引用对象改为值对象)
+## 4. 将引用对象改为值对象
+
+> Change Reference to value
以 Change Value to Reference 相反。值对象有个非常重要的特性:它是不可变的,不可变表示如果要改变这个对象,必须用一个新的对象来替换旧对象,而不是修改旧对象。
需要为值对象实现 equals() 和 hashCode() 方法
-## 5. Replace Array with Object(以对象取代数组)
+## 5. 以对象取代数组
+
+> Replace Array with Object
有一个数组,其中的元素各自代表不同的东西。
以对象替换数组,对于数组中的每个元素,以一个字段来表示,这样方便操作,也更容易理解。
-## 6. Duplicate Observed Data(赋值“被监视数据”)
+## 6. 赋值被监视数据
+
+> Duplicate Observed Data
一些领域数据置身于 GUI 控件中,而领域函数需要访问这些数据。
将该数据赋值到一个领域对象中,建立一个 Oberver 模式,用以同步领域对象和 GUI 对象内的重复数据。
-
+
-## 7. Change Unidirectional Association to Bidirectional(将单向关联改为双向关联)
+## 7. 将单向关联改为双向关联
+
+> Change Unidirectional Association to Bidirectional
当两个类都需要对方的特性时,可以使用双向关联。
@@ -612,55 +704,75 @@ class Curstomer{
注意到,这里让 Curstomer 类来控制关联关系。有以下原则来决定哪个类来控制关联关系:如果某个对象是组成另一个对象的部件,那么由后者负责控制关联关系;如果是一对多关系,则由单一引用那一方来控制关联关系。
-## 8. Change Bidirectional Association to Unidirectional(将双向关联改为单向关联)
+## 8. 将双向关联改为单向关联
+
+> Change Bidirectional Association to Unidirectional
和 Change Unidirectional Association to Bidirectiona 为反操作。
双向关联维护成本高,并且也不易于理解。大量的双向连接很容易造成“僵尸对象”:某个对象本身已经死亡了,却保留在系统中,因为它的引用还没有全部完全清除。
-## 9. Replace Magic Number with Symbolic Constant(以字面常量取代魔法数)
+## 9. 以字面常量取代魔法数
+
+> Replace Magic Number with Symbolic Constant
创建一个常量,根据其意义为它命名,并将字面常量换位这个常量。
-## 10. Encapsulate Field(封装字段)
+## 10. 封装字段
+
+> Encapsulate Field
public 字段应当改为 private,并提供相应的访问函数。
-## 11. Encapsulate Collection(封装集合)
+## 11. 封装集合
+
+> Encapsulate Collection
函数返回集合的一个只读副本,并在这个类中提供添加/移除集合元素的函数。如果函数返回集合自身,会让用户得以修改集合内容而集合拥有者却一无所知。
-## 12. Replace Record with Data Class(以数据类取代记录)
+## 12. 以数据类取代记录
-## 13. Replace Type Code with Class(以类取代类型码)
+> Replace Record with Data Class
+
+## 13. 以类取代类型码
+
+> Replace Type Code with Class
类中有一个数值类型码,但它并不影响类的行为,就用一个新类替换该数值类型码。如果类型码出现在 switch 语句中,需要使用 Replace Conditional with Polymorphism 去掉 switch,首先必须运用 Replace Type Code with Subcalss 或 Replace Type Code with State/Strategy 去掉类型码。
-
+
-## 14. Replace Type Code with Subcalsses(以子类取代类型码)
+## 14. 以子类取代类型码
+
+> Replace Type Code with Subcalsses
有一个不可变的类型码,它会影响类的行为,以子类取代这个类型码。
-
+
-## 15. Replace Type Code with State/Strategy (以 State/Strategy 取代类型码)
+## 15. 以 State/Strategy 取代类型码
+
+> Replace Type Code with State/Strategy
有一个可变的类型码,它会影响类的行为,以状态对象取代类型码。
和 Replace Type Code with Subcalsses 的区别是 Replace Type Code with State/Strategy 的类型码是动态可变的,前者通过继承的方式来实现,后者通过组合的方式来实现。因为类型码可变,如果通过继承的方式,一旦一个对象的类型码改变,那么就要改变用新的对象来取代旧对象,而客户端难以改变新的对象。但是通过组合的方式,改变引用的状态类是很容易的。
-
+
-## 16. Replace Subclass with Fields(以字段取代子类)
+## 16. 以字段取代子类
+
+> Replace Subclass with Fields
各个子类的唯一差别只在“返回常量数据”的函数上。
-
+
-# 简化条件表达式
+# 八、简化条件表达式
-## 1. Decompose Conditional(分解条件表达式)
+## 1. 分解条件表达式
+
+> Decompose Conditional
对于一个复杂的条件语句,可以从 if、then、else 三个段落中分别提炼出独立函数。
@@ -676,7 +788,9 @@ if(notSummer(date))
else charge = summerCharge(quantity);
```
-## 2. Consolidate Conditional Expression(合并条件表达式)
+## 2. 合并条件表达式
+
+> Consolidate Conditional Expression
有一系列条件测试,都得到相同结果。
@@ -697,7 +811,9 @@ double disabilityAmount(){
}
```
-## 3. Consolidate Duplicate Conditional Fragments (合并重复的条件片段)
+## 3. 合并重复的条件片段
+
+> Consolidate Duplicate Conditional Fragments
在条件表达式的每个分支上有着相同的一段代码。
@@ -722,13 +838,17 @@ if (isSpecialDeal()) {
send();
```
-## 4. Remove Control Flag(移除控制标记)
+## 4. 移除控制标记
+
+> Remove Control Flag
在一系列布尔表达式中,某个变量带有“控制标记”的作用。
-用 break语 句或 return 语句来取代控制标记。
+用 break 语 句或 return 语句来取代控制标记。
-## 5. Replace Nested Conditional with Guard Clauses (以卫语句取代嵌套条件表达式)
+## 5. 以卫语句取代嵌套条件表达式
+
+> Replace Nested Conditional with Guard Clauses
如果某个条件极其罕见,就应该单独检查该条件,并在该条件为真时立刻从函数中返回,这样的单独检查常常被称为“卫语句”(guard clauses)。
@@ -758,7 +878,9 @@ double getPayAmount() {
};
```
-## 6. Replace Conditional with Polymorphism (以多态取代条件表达式)
+## 6. 以多态取代条件表达式
+
+> Replace Conditional with Polymorphism
将这个条件表达式的每个分支放进一个子类内的覆写函数中,然后将原始函数声明为抽象函数。需要先使用 Replace Type Code with Subclass 或 Replace Type Code with State/Strategy 来建立继承结果。
@@ -776,9 +898,12 @@ double getSpeed() {
}
```
-
+
-## 7. Introduce Null Object(引入Null对象)
+
+## 7. 引入 Null 对象
+
+> Introduce Null Object
将 null 值替换为 null 对象。这样做的好处在于,不需要询问对象是否为空,直接调用就行。
@@ -787,7 +912,9 @@ if (customer == null) plan = BillingPlan.basic();
else plan = customer.getPlan();
```
-## 8. Introduce Assertion(引入断言)
+## 8. 引入断言
+
+> Introduce Assertion
以断言明确表现某种假设。断言只能用于开发过程中,产品代码中不会有断言。
@@ -805,21 +932,29 @@ double getExpenseLimit() {
}
```
-# 简化函数调用
+# 九、简化函数调用
-## 1. Rename Method(函数改名)
+## 1. 函数改名
+
+> Rename Method
使函数名能解释函数的用途。
-## 2. Add Parameter(添加参数)
+## 2. 添加参数
+
+> Add Parameter
使函数不需要通过调用获得某个信息。
-## 3. Remove Parameter(移除参数)
+## 3. 移除参数
+
+> Remove Parameter
与 Add Parameter 相反,改用调用的方式来获得某个信息。
-## 4. Separate Query from Modifier(将查询函数和修改函数分离)
+## 4. 将查询函数和修改函数分离
+
+> Separate Query from Modifier
某个函数即返回对象状态值,又修改对象状态。
@@ -834,7 +969,9 @@ getTotalOutstanding();
setReadyForSummaries();
```
-## 5. Parameterize Method(令函数携带参数)
+## 5. 令函数携带参数
+
+> Parameterize Method
若干函数做了类似的工作,但在函数本体中却包含了不同的值。
@@ -848,7 +985,9 @@ tenPercentRaise();
raise(percentage);
```
-## 6. Replace Parameter with Explicit Methods(以明确函数取代参数)
+## 6. 以明确函数取代参数
+
+> Replace Parameter with Explicit Methods
有一个函数,完全取决于参数值而采取不同行为。
@@ -877,7 +1016,9 @@ void setWidth(int arg){
}
```
-## 7. Preserve Whole Object(保持对象完整)
+## 7. 保持对象完整
+
+> Preserve Whole Object
从某个对象中取出若干值,将它们作为某一次函数调用时的参数。
@@ -893,7 +1034,9 @@ withinPlan = plan.withinRange(low,high);
withinPlan = plan.withinRange(daysTempRange());
```
-## 8. Replace Parameter with Methods(以函数取代参数)
+## 8. 以函数取代参数
+
+> Replace Parameter with Methods
对象调用某个函数,并将所得结果作为参数,传递给另一个函数。而接受该参数的函数本身也能够调用前一个函数。
@@ -910,33 +1053,43 @@ int basePrice = _quantity * _itemPrice;
double finalPrice = discountedPrice (basePrice);
```
-## 9. Introduce Parameter Object(引入参数对象)
+## 9. 引入参数对象
+
+> Introduce Parameter Object
某些参数总是很自然地同时出现,这些参数就是 Data Clumps。
以一个对象取代这些参数。
-
+
-## 10. Remove Setting Method(移除设值函数)
+## 10. 移除设值函数
+
+> Remove Setting Method
类中的某个字段应该在对象创建时被设值,然后就不再改变。
去掉该字段的所有设值函数,并将该字段设为 final。
-## 11. Hide Method(隐藏函数)
+## 11. 隐藏函数
+
+> Hide Method
有一个函数,从来没有被其他任何类用到。
将这个函数修改为 private。
-## 12. Replace Constructor with Factory Method (以工厂函数取代构造函数)
+## 12. 以工厂函数取代构造函数
+
+> Replace Constructor with Factory Method
希望在创建对象时不仅仅是做简单的建构动作。
将构造函数替换为工厂函数。
-## 13. Encapsulate Downcast(封装向下转型)
+## 13. 封装向下转型
+
+> Encapsulate Downcast
某个函数返回的对象,需要由函数调用者执行向下转型(downcast)。
@@ -953,13 +1106,17 @@ Reading lastReading(){
}
```
-## 14. Replace Error Code with Exception (以异常取代错误码)
+## 14. 以异常取代错误码
+
+> Replace Error Code with Exception
某个函数返回一个特定的代码,用以表示某种错误情况。
改用异常,异常将普通程序和错误处理分开,使代码更容易理解。
-## 15. Replace Exception with Test(以测试取代异常)
+## 15. 以测试取代异常
+
+> Replace Exception with Test
面对一个调用者可以预先检查的条件,你抛出了一个异常。
@@ -980,21 +1137,27 @@ double getValueForPeriod(int periodNumber) {
return values[periodNumber];
```
-# 处理概括关系
+# 十、处理概括关系
-## 1. Pull Up Field(字段上移)
+## 1. 字段上移
+
+> Pull Up Field
两个子类拥有相同的字段。
将该字段移至超类。
-## 2. Pull Up Method(函数上移)
+## 2. 函数上移
+
+> Pull Up Method
有些函数,在各个子类中产生完全相同的结果。
将该函数移至超类。
-## 3. Pull Up Constructor Body(构造函数本体上移)
+## 3. 构造函数本体上移
+
+> Pull Up Constructor Body
你在各个子类中拥有一些构造函数,它们的本体几乎完全一致。
@@ -1017,56 +1180,78 @@ public Manager(String name, String id, int grade) {
}
```
-## 4. Push Down Method(函数下移)
+## 4. 函数下移
+
+> Push Down Method
超类中的某个函数只与部分子类有关。
将这个函数移到相关的那些子类去。
-## 5. Push Down Field(字段下移)
+## 5. 字段下移
+
+> Push Down Field
超类中的某个字段只被部分子类用到。
将这个字段移到需要它的那些子类去。
-## 6. Extract Subclass(提炼子类)
+## 6. 提炼子类
+
+> Extract Subclass
类中的某些特性只被某些实例用到。
新建一个子类,将上面所说的那一部分特性移到子类中。
-## 7. Extract Superclass(提炼超类)
+## 7. 提炼超类
+
+> Extract Superclass
两个类有相似特性。
为这两个类建立一个超类,将相同特性移至超类。
-## 8. Extract Interface(提炼接口)
+## 8. 提炼接口
+
+> Extract Interface
若干客户使用类接口中的同一子集,或者两个类的接口有部分相同。
将相同的子集提炼到一个独立接口中。
-## 9. Collapse Hierarchy(折叠继承体系)
+## 9. 折叠继承体系
+
+> Collapse Hierarchy
超类和子类之间无太大区别。
将它们合为一体。
-## 10. Form Template Method(塑造模板函数)
+## 10. 塑造模板函数
+
+> Form Template Method
你有一些子类,其中相应的某些函数以相同顺序执行类似的操作,但各个操作的细节上有所不同。
将这些操作分别放进独立函数中,并保持它们都有相同的签名,于是原函数也就变得相同了。然后将原函数上移至超类。(模板方法模式)
-## 11. Replace Inheritance with Delegation (以委托取代继承)
+## 11. 以委托取代继承
+
+> Replace Inheritance with Delegation
某个子类只使用超类接口中的一部分,或是根本不需要继承而来的数据。
在子类中新建一个字段用以保存超类,调整子类函数,令它改而委托超类,然后去掉两者之间的继承关系。
-## 12. Replace Delegation with Inheritance (以继承取代委托)
+## 12. 以继承取代委托
+
+> Replace Delegation with Inheritance
你在两个类之间使用委托关系,并经常为整个接口编写许多极简单的委托函数。
让委托类继承受托类。
+
+# 参考资料
+
+- MartinFowler, 福勒, 贝克, 等. 重构: 改善既有代码的设计 [M]. 电子工业出版社, 2011.
diff --git a/notes/面向对象思想.md b/notes/面向对象思想.md
index d079bb2c..5ab84adf 100644
--- a/notes/面向对象思想.md
+++ b/notes/面向对象思想.md
@@ -1,111 +1,125 @@
-* [S.O.L.I.D](#solid)
- * [1. 单一责任原则](#1-单一责任原则)
- * [2. 开放封闭原则](#2-开放封闭原则)
- * [3. 里氏替换原则](#3-里氏替换原则)
- * [4. 接口分离原则](#4-接口分离原则)
- * [5. 依赖倒置原则](#5-依赖倒置原则)
-* [其他常见原则](#其他常见原则)
- * [1. 迪米特法则](#1-迪米特法则)
- * [2. 合成复用原则](#2-合成复用原则)
- * [3. 共同封闭原则](#3-共同封闭原则)
- * [4. 稳定抽象原则](#4-稳定抽象原则)
- * [5. 稳定依赖原则](#5-稳定依赖原则)
-* [封装、继承、多态](#封装继承多态)
- * [1. 封装](#1-封装)
- * [2. 继承](#2-继承)
- * [3. 多态](#3-多态)
-* [UML](#uml)
- * [1. 类图](#1-类图)
- * [2. 时序图](#2-时序图)
+* [一、设计原则](#一设计原则)
+ * [S.O.L.I.D](#solid)
+ * [其他常见原则](#其他常见原则)
+* [二、三大特性](#二三大特性)
+ * [封装](#封装)
+ * [继承](#继承)
+ * [多态](#多态)
+* [三、类图](#三类图)
+ * [泛化关系 (Generalization)](#泛化关系-generalization)
+ * [实现关系 (Realization)](#实现关系-realization)
+ * [聚合关系 (Aggregation)](#聚合关系-aggregation)
+ * [组合关系 (Composition)](#组合关系-composition)
+ * [关联关系 (Association)](#关联关系-association)
+ * [依赖关系 (Dependency)](#依赖关系-dependency)
* [参考资料](#参考资料)
-# S.O.L.I.D
+# 一、设计原则
-S.O.L.I.D 是面向对象设计和编程 (OOD&OOP) 中几个重要编码原则 (Programming Priciple) 的首字母缩写。
+## S.O.L.I.D
-| 简写 | 全拼 | 中文翻译 |
-| -- | -- | -- |
-|SRP| The Single Responsibility Principle | 单一责任原则 |
-|OCP| The Open Closed Principle | 开放封闭原则 |
-|LSP| The Liskov Substitution Principle | 里氏替换原则 |
-|ISP| The Interface Segregation Principle | 接口分离原则 |
-|DIP| The Dependency Inversion Principle | 依赖倒置原则 |
+| 简写 | 全拼 | 中文翻译 |
+| :--: | :--: | :--: |
+| SRP | The Single Responsibility Principle | 单一责任原则 |
+| OCP | The Open Closed Principle | 开放封闭原则 |
+| LSP | The Liskov Substitution Principle | 里氏替换原则 |
+| ISP | The Interface Segregation Principle | 接口分离原则 |
+| DIP | The Dependency Inversion Principle | 依赖倒置原则 |
+### 1. 单一责任原则
-## 1. 单一责任原则
+> 修改一个类的原因应该只有一个。
-当需要修改某个类的时候原因有且只有一个。换句话说就是让一个类只做一种类型责任,当这个类需要承当其他类型的责任的时候,就需要分解这个类。
+换句话说就是让一个类只负责一件事,当这个类需要做过多事情的时候,就需要分解这个类。
-## 2. 开放封闭原则
+如果一个类承担的职责过多,就等于把这些职责耦合在了一起,一个职责的变化可能会削弱这个类完成其它职责的能力。
-软件实体应该是可扩展,而不可修改的。也就是说,对扩展是开放的,而对修改是封闭的。
+### 2. 开放封闭原则
-## 3. 里氏替换原则
+> 类应该对扩展开放,对修改关闭。
-当一个子类的实例应该能够替换任何其超类的实例时,它们之间才具有 is-a 关系。
+扩展就是添加新功能的意思,因此该原则要求在添加新功能时不需要修改代码。
-## 4. 接口分离原则
+符合开闭原则最典型的设计模式是装饰者模式,它可以动态地将责任附加到对象上,而不用去修改类的代码。
-不能强迫用户去依赖那些他们不使用的接口。换句话说,使用多个专门的接口比使用单一的总接口总要好。
+### 3. 里氏替换原则
-## 5. 依赖倒置原则
+> 子类对象必须能够替换掉所有父类对象。
-1. 高层模块不应该依赖于低层模块,二者都应该依赖于抽象
-2. 抽象不应该依赖于细节,细节应该依赖于抽象
+继承是一种 IS-A 关系,子类需要能够当成父类来使用,并且需要比父类更特殊。
-# 其他常见原则
+如果不满足这个原则,那么各个子类的行为上就会有很大差异,增加继承体系的复杂度。
+
+### 4. 接口分离原则
+
+> 不应该强迫客户依赖于它们不用的方法。
+
+因此使用多个专门的接口比使用单一的总接口要好。
+
+### 5. 依赖倒置原则
+
+> 高层模块不应该依赖于低层模块,二者都应该依赖于抽象;抽象不应该依赖于细节,细节应该依赖于抽象。
+
+高层模块包含一个应用程序中重要的策略选择和业务模块,如果高层模块依赖于低层模块,那么低层模块的改动就会直接影响到高层模块,从而迫使高层模块也需要改动。
+
+依赖于抽象意味着:
+
+- 任何变量都不应该持有一个指向具体类的指针或者引用;
+- 任何类都不应该从具体类派生;
+- 任何方法都不应该覆写它的任何基类中的已经实现的方法。
+
+## 其他常见原则
除了上述的经典原则,在实际开发中还有下面这些常见的设计原则。
| 简写 | 全拼 | 中文翻译 |
-| -- | -- | -- |
-|LoD| The Law of Demeter | 迪米特法则 |
-|CRP| The Composite Reuse Principle | 合成复用原则 |
-|CCP| The Common Closure Principle | 共同封闭原则 |
+| :--: | :--: | :--: |
+|LOD| The Law of Demeter | 迪米特法则 |
+|CRP| The Composite Reuse Principle | 合成复用原则 |
+|CCP| The Common Closure Principle | 共同封闭原则 |
|SAP| The Stable Abstractions Principle | 稳定抽象原则 |
|SDP| The Stable Dependencies Principle | 稳定依赖原则 |
-## 1. 迪米特法则
+### 1. 迪米特法则
-迪米特法则又叫作最少知道原则(Least Knowledge Principle 简写LKP),就是说一个对象应当对其他对象有尽可能少的了解,不和陌生人说话。
+迪米特法则又叫作最少知识原则(Least Knowledge Principle,简写 LKP),就是说一个对象应当对其他对象有尽可能少的了解,不和陌生人说话。
-## 2. 合成复用原则
+### 2. 合成复用原则
尽量使用对象组合,而不是继承来达到复用的目的。
-## 3. 共同封闭原则
+### 3. 共同封闭原则
一起修改的类,应该组合在一起(同一个包里)。如果必须修改应用程序里的代码,我们希望所有的修改都发生在一个包里(修改关闭),而不是遍布在很多包里。
-## 4. 稳定抽象原则
+### 4. 稳定抽象原则
最稳定的包应该是最抽象的包,不稳定的包应该是具体的包,即包的抽象程度跟它的稳定性成正比。
-## 5. 稳定依赖原则
+### 5. 稳定依赖原则
包之间的依赖关系都应该是稳定方向依赖的,包要依赖的包要比自己更具有稳定性。
-# 封装、继承、多态
+# 二、三大特性
-封装、继承、多态是面向对象的三大特性。
+## 封装
-## 1. 封装
+利用抽象数据类型将数据和基于数据的操作封装在一起,使其构成一个不可分割的独立实体。数据被保护在抽象数据类型的内部,尽可能地隐藏内部的细节,只保留一些对外接口使之与外部发生联系。用户无需知道对象内部的细节,但可以通过对象对外提供的接口来访问该对象。
-利用抽象数据类型将数据和基于数据的操作封装在一起,使其构成一个不可分割的独立实体,数据被保护在抽象数据类型的内部,尽可能地隐藏内部的细节,只保留一些对外接口使之与外部发生联系。用户是无需知道对象内部的细节,但可以通过该对象对外的提供的接口来访问该对象。
+优点:
-封装有三大好处:
-
-1. 良好的封装能够减少耦合。
-2. 类内部的结构可以自由修改。
-3. 可以对成员进行更精确的控制。
-4. 隐藏信息,实现细节。
+- 减少耦合:可以独立地开发、测试、优化、使用、理解和修改
+- 减轻维护的负担:可以更容易被程序员理解,并且在调试的时候可以不影响其他模块
+- 有效地调节性能:可以通过剖析确定哪些模块影响了系统的性能
+- 提高软件的可重用性
+- 减低了构建大型系统的风险:即使整个系统不可用,但是这些独立的模块却有可能是可用的
以下 Person 类封装 name、gender、age 等属性,外界只能通过 get() 方法获取一个 Person 对象的 name 属性和 gender 属性,而无法获取 age 属性,但是 age 属性可以供 work() 方法使用。
-注意到 gender 属性使用 int 数据类型进行存储,封装使得用户注意不到这种实现细节。并且在需要修改使用的数据类型时,也可以在不影响客户端代码的情况下进行。
+注意到 gender 属性使用 int 数据类型进行存储,封装使得用户注意不到这种实现细节。并且在需要修改 gender 属性使用的数据类型时,也可以在不影响客户端代码的情况下进行。
```java
public class Person {
@@ -125,31 +139,35 @@ public class Person {
if(18 <= age && age <= 50) {
System.out.println(name + " is working very hard!");
} else {
- System.out.println(name + " can't work!");
+ System.out.println(name + " can't work any more!");
}
}
}
```
-## 2. 继承
+## 继承
-继承实现了 **is-a** 关系,例如 Cat 和 Animal 就是一种 is-a 关系,因此可以将 Cat 继承自 Animal,从而获得 Animal 非 private 的属性和方法。
+继承实现了 **IS-A** 关系,例如 Cat 和 Animal 就是一种 IS-A 关系,因此 Cat 可以继承自 Animal,从而获得 Animal 非 private 的属性和方法。
-Cat 可以当做 Animal 来使用,也就是可以使用 Animal 引用 Cat 对象,这种子类转换为父类称为 **向上转型** 。
-
-继承应该遵循里氏替换原则:当一个子类的实例应该能够替换任何其超类的实例时,它们之间才具有 is-a 关系。
+Cat 可以当做 Animal 来使用,也就是说可以使用 Animal 引用 Cat 对象。父类引用指向子类对象称为 **向上转型** 。
```java
Animal animal = new Cat();
```
-## 3. 多态
+继承应该遵循里氏替换原则,子类对象必须能够替换掉所有父类对象。
-多态分为编译时多态和运行时多态。编译时多态主要指方法的重装,运行时多态指程序中定义的对象引用所指向的具体类型在运行期间才确定。
+## 多态
-多态有三个条件:1. 继承;2. 覆盖父类方法;3. 向上转型。
+多态分为编译时多态和运行时多态。编译时多态主要指方法的重载,运行时多态指程序中定义的对象引用所指向的具体类型在运行期间才确定。
-下面的代码中,乐器类(Instrument)有两个子类:Wind 和 Percussion,它们都覆盖了 play() 方法,并且在 main() 方法中使用父类 Instrument 来引用 Wind 和 Percussion 对象。在 Instrument 引用调用 play() 方法时,会执行实际引用对象所在类的 play() 方法,而不是 Instrument 类的方法。
+运行时多态有三个条件:
+
+1. 继承
+2. 覆盖
+3. 向上转型
+
+下面的代码中,乐器类(Instrument)有两个子类:Wind 和 Percussion,它们都覆盖了父类的 play() 方法,并且在 main() 方法中使用父类 Instrument 来引用 Wind 和 Percussion 对象。在 Instrument 引用调用 play() 方法时,会执行实际引用对象所在类的 play() 方法,而不是 Instrument 类的方法。
```java
public class Instrument {
@@ -180,160 +198,55 @@ public class Music {
}
}
}
-
```
+# 三、类图
+## 泛化关系 (Generalization)
-# UML
+用来描述继承关系,在 Java 中使用 extends 关键字。
-## 1. 类图
+
-**1.1 继承相关**
+## 实现关系 (Realization)
-继承有两种形式 : 泛化(generalize)和实现(realize),表现为 is-a 关系。
+用来实现一个接口,在 Java 中使用 implement 关键字。
-① 泛化关系 (generalization)
+
-从具体类中继承
+## 聚合关系 (Aggregation)
-
+表示整体由部分组成,但是整体和部分不是强依赖的,整体不存在了部分还是会存在。
-② 实现关系 (realize)
+
-从抽象类或者接口中继承
-
-
-
-**1.2 整体和部分**
-
-① 聚合关系 (aggregation)
-
-表示整体由部分组成,但是整体和部分不是强依赖的,整体不存在了部分还是会存在。以下表示 B 由 A 组成:
-
-
-
-② 组合关系 (composition)
+## 组合关系 (Composition)
和聚合不同,组合中整体和部分是强依赖的,整体不存在了部分也不存在了。比如公司和部门,公司没了部门就不存在了。但是公司和员工就属于聚合关系了,因为公司没了员工还在。
-
+
-**1.3 相互联系**
-
-① 关联关系 (association)
+## 关联关系 (Association)
表示不同类对象之间有关联,这是一种静态关系,与运行过程的状态无关,在最开始就可以确定。因此也可以用 1 对 1、多对 1、多对多这种关联关系来表示。比如学生和学校就是一种关联关系,一个学校可以有很多学生,但是一个学生只属于一个学校,因此这是一种多对一的关系,在运行开始之前就可以确定。
-
+
-② 依赖关系 (dependency)
+## 依赖关系 (Dependency)
-和关联关系不同的是 , 依赖关系是在运行过程中起作用的。一般依赖作为类的构造器或者方法的参数传入。双向依赖时一种不好的设计。
+和关联关系不同的是,依赖关系是在运行过程中起作用的。A 类和 B 类是依赖关系主要有三种形式:
-
-
-## 2. 时序图
-
-http://www.cnblogs.com/wolf-sun/p/UML-Sequence-diagram.html
-
-**2.1 定义**
-
-时序图描述了对象之间传递消息的时间顺序,它用来表示用例的行为顺序。它的主要作用是通过对象间的交互来描述用例(注意是对象),从而寻找类的操作。
-
-**2.2 赤壁之战时序图**
-
-从虚线从上往下表示时间的推进。
-
-
-
-可见,通过时序图可以知道每个类具有以下操作:
-
-```java
-publc class 刘备 {
- public void 应战 ();
-}
-
-publc class 孔明 {
- public void 拟定策略 ();
- public void 联合孙权 ();
- private void 借东风火攻 ();
-}
-
-public class 关羽 {
- public void 防守荊州 ();
-}
-
-public class 张飞 {
- public void 防守荆州前线 ();
-}
-
-public class 孙权 {
- public void 领兵相助 ();
-}
-```
-
-**2.3 活动图、时序图之间的关系**
-
-活动图示从用户的角度来描述用例;
-
-时序图是从计算机的角度(对象间的交互)描述用例。
-
-**2.4 类图与时序图的关系**
-
-类图描述系统的静态结构,时序图描述系统的动态行为。
-
-**2.5 时序图的组成**
-
-① 对象
-
-有三种表现形式
-
-
-
-在画图时,应该遵循以下原则:
-
-1. 把交互频繁的对象尽可能地靠拢。
-
-2. 把初始化整个交互活动的对象(有时是一个参与者)放置在最左边。
-
-② 生命线
-
-生命线从对象的创建开始到对象销毁时终止
-
-
-
-③ 消息
-
-对象之间的交互式通过发送消息来实现的。
-
-消息有 4 种类型:
-
-1\. 简单消息,不区分同步异步。
-
-
-
-2\. 同步消息,发送消息之后需要暂停活动来等待回应。
-
-
-
-3\. 异步消息,发送消息之后不需要等待。
-
-
-
-4\. 返回消息,可选。
-
-④ 激活
-
-生命线上的方框表示激活状态,其它时间处于休眠状态。
-
-
+1. A 类是 B 类中的(某中方法的)局部变量;
+2. A 类是 B 类方法当中的一个参数;
+3. A 类向 B 类发送消息,从而影响 B 类发生变化;
+
# 参考资料
- Java 编程思想
-- [ 面向对象设计的 SOLID 原则 ](http://www.cnblogs.com/shanyou/archive/2009/09/21/1570716.html)
-- [ 看懂 UML 类图和时序图 ](http://design-patterns.readthedocs.io/zh_CN/latest/read_uml.html#generalization)
+- 敏捷软件开发:原则、模式与实践
+- [面向对象设计的 SOLID 原则](http://www.cnblogs.com/shanyou/archive/2009/09/21/1570716.html)
+- [看懂 UML 类图和时序图](http://design-patterns.readthedocs.io/zh_CN/latest/read_uml.html#generalization)
- [UML 系列——时序图(顺序图)sequence diagram](http://www.cnblogs.com/wolf-sun/p/UML-Sequence-diagram.html)
-- [ 面向对象编程三大特性 ------ 封装、继承、多态 ](http://blog.csdn.net/jianyuerensheng/article/details/51602015)
+- [面向对象编程三大特性 ------ 封装、继承、多态](http://blog.csdn.net/jianyuerensheng/article/details/51602015)
diff --git a/other/alipay.jpg b/other/alipay.jpg
deleted file mode 100644
index 211df347..00000000
Binary files a/other/alipay.jpg and /dev/null differ
diff --git a/other/alipay.md b/other/alipay.md
deleted file mode 100644
index 3513ee8f..00000000
--- a/other/alipay.md
+++ /dev/null
@@ -1,3 +0,0 @@
-
-
-
diff --git a/other/handbook.png b/other/handbook.png
deleted file mode 100644
index 68deb247..00000000
Binary files a/other/handbook.png and /dev/null differ
diff --git a/pics/0042edad-8e3b-4279-bd93-6906fcd1b640.jpg b/pics/0042edad-8e3b-4279-bd93-6906fcd1b640.jpg
deleted file mode 100644
index 9cabf716..00000000
Binary files a/pics/0042edad-8e3b-4279-bd93-6906fcd1b640.jpg and /dev/null differ
diff --git a/pics/005d83c2-e64a-41f0-bbdd-51c71d494a18.jpg b/pics/005d83c2-e64a-41f0-bbdd-51c71d494a18.jpg
deleted file mode 100644
index 00ac9106..00000000
Binary files a/pics/005d83c2-e64a-41f0-bbdd-51c71d494a18.jpg and /dev/null differ
diff --git a/pics/00d8d345-cd4a-48af-919e-209d2788eca7.jpg b/pics/00d8d345-cd4a-48af-919e-209d2788eca7.jpg
deleted file mode 100644
index f20c6cdc..00000000
Binary files a/pics/00d8d345-cd4a-48af-919e-209d2788eca7.jpg and /dev/null differ
diff --git a/pics/01658047-0d86-4a7a-a8ca-7ea20fa1fdde.png b/pics/01658047-0d86-4a7a-a8ca-7ea20fa1fdde.png
deleted file mode 100644
index e459e145..00000000
Binary files a/pics/01658047-0d86-4a7a-a8ca-7ea20fa1fdde.png and /dev/null differ
diff --git a/pics/02986f62-c641-44a8-a55f-983581490e0c.png b/pics/02986f62-c641-44a8-a55f-983581490e0c.png
new file mode 100644
index 00000000..96726ebc
Binary files /dev/null and b/pics/02986f62-c641-44a8-a55f-983581490e0c.png differ
diff --git a/pics/037c3a0b-332d-434d-a374-f343ef72c8e1.jpg b/pics/037c3a0b-332d-434d-a374-f343ef72c8e1.jpg
new file mode 100644
index 00000000..68950556
Binary files /dev/null and b/pics/037c3a0b-332d-434d-a374-f343ef72c8e1.jpg differ
diff --git a/pics/04ff7ae6-7bee-4cf8-82f8-dfe2ba1f3616.jpg b/pics/04ff7ae6-7bee-4cf8-82f8-dfe2ba1f3616.jpg
deleted file mode 100644
index 44529eab..00000000
Binary files a/pics/04ff7ae6-7bee-4cf8-82f8-dfe2ba1f3616.jpg and /dev/null differ
diff --git a/pics/061c29ce-e2ed-425a-911e-56fbba1efce3.jpg b/pics/061c29ce-e2ed-425a-911e-56fbba1efce3.jpg
new file mode 100644
index 00000000..29d32e72
Binary files /dev/null and b/pics/061c29ce-e2ed-425a-911e-56fbba1efce3.jpg differ
diff --git a/pics/0635cbe8.png b/pics/0635cbe8.png
new file mode 100644
index 00000000..849c9eaf
Binary files /dev/null and b/pics/0635cbe8.png differ
diff --git a/pics/065c3bbb-3ea0-4dbf-8f26-01d0e0ba7db7.png b/pics/065c3bbb-3ea0-4dbf-8f26-01d0e0ba7db7.png
deleted file mode 100644
index cf80e5ae..00000000
Binary files a/pics/065c3bbb-3ea0-4dbf-8f26-01d0e0ba7db7.png and /dev/null differ
diff --git a/pics/07717718-1230-4347-aa18-2041c315e670.jpg b/pics/07717718-1230-4347-aa18-2041c315e670.jpg
new file mode 100644
index 00000000..456bd952
Binary files /dev/null and b/pics/07717718-1230-4347-aa18-2041c315e670.jpg differ
diff --git a/pics/07903a31-0fb3-45fc-86f5-26f0b28fa4e7.png b/pics/07903a31-0fb3-45fc-86f5-26f0b28fa4e7.png
new file mode 100644
index 00000000..40573164
Binary files /dev/null and b/pics/07903a31-0fb3-45fc-86f5-26f0b28fa4e7.png differ
diff --git a/pics/08427d38-8df1-49a1-8990-e0ce5ee36ca2.png b/pics/08427d38-8df1-49a1-8990-e0ce5ee36ca2.png
new file mode 100644
index 00000000..13d49836
Binary files /dev/null and b/pics/08427d38-8df1-49a1-8990-e0ce5ee36ca2.png differ
diff --git a/pics/086871db-5871-460f-97b7-126cd738bb0e.jpg b/pics/086871db-5871-460f-97b7-126cd738bb0e.jpg
deleted file mode 100644
index 75ce9011..00000000
Binary files a/pics/086871db-5871-460f-97b7-126cd738bb0e.jpg and /dev/null differ
diff --git a/pics/08738dd0-ae8e-404a-ba78-a6b1b7d225b3.jpg b/pics/08738dd0-ae8e-404a-ba78-a6b1b7d225b3.jpg
deleted file mode 100644
index 7aadcbe1..00000000
Binary files a/pics/08738dd0-ae8e-404a-ba78-a6b1b7d225b3.jpg and /dev/null differ
diff --git a/pics/09e398d8-9c6e-48f6-b48b-8b4f9de61d1d.png b/pics/09e398d8-9c6e-48f6-b48b-8b4f9de61d1d.png
deleted file mode 100644
index 908ebe92..00000000
Binary files a/pics/09e398d8-9c6e-48f6-b48b-8b4f9de61d1d.png and /dev/null differ
diff --git a/pics/09e6e8d4-4d83-4949-b908-6d6b4c2fd064.png b/pics/09e6e8d4-4d83-4949-b908-6d6b4c2fd064.png
new file mode 100644
index 00000000..f76e5b5f
Binary files /dev/null and b/pics/09e6e8d4-4d83-4949-b908-6d6b4c2fd064.png differ
diff --git a/pics/0a9f4125-b6ab-4e94-a807-fd7070ae726a.png b/pics/0a9f4125-b6ab-4e94-a807-fd7070ae726a.png
new file mode 100644
index 00000000..395d9201
Binary files /dev/null and b/pics/0a9f4125-b6ab-4e94-a807-fd7070ae726a.png differ
diff --git a/pics/0aaf4630-d2a2-4783-b3f7-a2b6a7dfc01b.jpg b/pics/0aaf4630-d2a2-4783-b3f7-a2b6a7dfc01b.jpg
new file mode 100644
index 00000000..bee1bae5
Binary files /dev/null and b/pics/0aaf4630-d2a2-4783-b3f7-a2b6a7dfc01b.jpg differ
diff --git a/pics/0c55e11c-d3ce-4cd8-b139-028aea6f40e3.png b/pics/0c55e11c-d3ce-4cd8-b139-028aea6f40e3.png
new file mode 100644
index 00000000..5892c1f9
Binary files /dev/null and b/pics/0c55e11c-d3ce-4cd8-b139-028aea6f40e3.png differ
diff --git a/pics/0ddebc5c-7c24-46b1-98db-4fa5e54db16b.png b/pics/0ddebc5c-7c24-46b1-98db-4fa5e54db16b.png
deleted file mode 100644
index 7fb917af..00000000
Binary files a/pics/0ddebc5c-7c24-46b1-98db-4fa5e54db16b.png and /dev/null differ
diff --git a/pics/0e34263d-7287-4ffe-a716-37c53d1a2526.png b/pics/0e34263d-7287-4ffe-a716-37c53d1a2526.png
new file mode 100644
index 00000000..c5f0bbe3
Binary files /dev/null and b/pics/0e34263d-7287-4ffe-a716-37c53d1a2526.png differ
diff --git a/pics/0ee0f61b-c782-441e-bf34-665650198ae0.jpg b/pics/0ee0f61b-c782-441e-bf34-665650198ae0.jpg
new file mode 100644
index 00000000..f3e7163b
Binary files /dev/null and b/pics/0ee0f61b-c782-441e-bf34-665650198ae0.jpg differ
diff --git a/pics/0f31bc7a-d60b-48a6-8e3f-597708369e52.png b/pics/0f31bc7a-d60b-48a6-8e3f-597708369e52.png
deleted file mode 100644
index df191c91..00000000
Binary files a/pics/0f31bc7a-d60b-48a6-8e3f-597708369e52.png and /dev/null differ
diff --git a/pics/0f373947-c68f-45b4-a59e-086154745ac5.png b/pics/0f373947-c68f-45b4-a59e-086154745ac5.png
new file mode 100644
index 00000000..6b474d85
Binary files /dev/null and b/pics/0f373947-c68f-45b4-a59e-086154745ac5.png differ
diff --git a/pics/0f39c274-b79c-4e83-8c7c-94fc2747832d.jpg b/pics/0f39c274-b79c-4e83-8c7c-94fc2747832d.jpg
deleted file mode 100644
index 8cdf9af3..00000000
Binary files a/pics/0f39c274-b79c-4e83-8c7c-94fc2747832d.jpg and /dev/null differ
diff --git a/pics/0f8c0a60-d4c6-47f4-978d-1a5c393fedac.jpg b/pics/0f8c0a60-d4c6-47f4-978d-1a5c393fedac.jpg
deleted file mode 100644
index 4719d5ae..00000000
Binary files a/pics/0f8c0a60-d4c6-47f4-978d-1a5c393fedac.jpg and /dev/null differ
diff --git a/pics/10.gif b/pics/10.gif
new file mode 100644
index 00000000..d52a911e
Binary files /dev/null and b/pics/10.gif differ
diff --git a/pics/1005dc9d-9049-4b06-9524-6171e56ebd8c.png b/pics/1005dc9d-9049-4b06-9524-6171e56ebd8c.png
deleted file mode 100644
index 04142289..00000000
Binary files a/pics/1005dc9d-9049-4b06-9524-6171e56ebd8c.png and /dev/null differ
diff --git a/pics/106f5585-b2e7-4718-be5d-3b322d1ef42a.jpg b/pics/106f5585-b2e7-4718-be5d-3b322d1ef42a.jpg
deleted file mode 100644
index b70e5806..00000000
Binary files a/pics/106f5585-b2e7-4718-be5d-3b322d1ef42a.jpg and /dev/null differ
diff --git a/pics/107a6a2b-f15b-4cad-bced-b7fb95258c9c.png b/pics/107a6a2b-f15b-4cad-bced-b7fb95258c9c.png
new file mode 100644
index 00000000..35c2ddb1
Binary files /dev/null and b/pics/107a6a2b-f15b-4cad-bced-b7fb95258c9c.png differ
diff --git a/pics/10bdf7bf-0daa-4a26-b927-f142b3f8e72b.png b/pics/10bdf7bf-0daa-4a26-b927-f142b3f8e72b.png
new file mode 100644
index 00000000..cb78d856
Binary files /dev/null and b/pics/10bdf7bf-0daa-4a26-b927-f142b3f8e72b.png differ
diff --git a/pics/10f5e35b-1c71-4717-9e80-47f259702642.jpg b/pics/10f5e35b-1c71-4717-9e80-47f259702642.jpg
new file mode 100644
index 00000000..7041fd5f
Binary files /dev/null and b/pics/10f5e35b-1c71-4717-9e80-47f259702642.jpg differ
diff --git a/pics/11.gif b/pics/11.gif
new file mode 100644
index 00000000..5d9c6f00
Binary files /dev/null and b/pics/11.gif differ
diff --git a/pics/110b1a9b-87cd-45c3-a21d-824623715b33.jpg b/pics/110b1a9b-87cd-45c3-a21d-824623715b33.jpg
deleted file mode 100644
index 7b7359ee..00000000
Binary files a/pics/110b1a9b-87cd-45c3-a21d-824623715b33.jpg and /dev/null differ
diff --git a/pics/111521118015898.gif b/pics/111521118015898.gif
new file mode 100644
index 00000000..5c31da1d
Binary files /dev/null and b/pics/111521118015898.gif differ
diff --git a/pics/111521118445538.gif b/pics/111521118445538.gif
new file mode 100644
index 00000000..323d129c
Binary files /dev/null and b/pics/111521118445538.gif differ
diff --git a/pics/111521118483039.gif b/pics/111521118483039.gif
new file mode 100644
index 00000000..a81124dd
Binary files /dev/null and b/pics/111521118483039.gif differ
diff --git a/pics/111521118640738.gif b/pics/111521118640738.gif
new file mode 100644
index 00000000..7a7b05a9
Binary files /dev/null and b/pics/111521118640738.gif differ
diff --git a/pics/111521119203347.gif b/pics/111521119203347.gif
new file mode 100644
index 00000000..37cdb5a5
Binary files /dev/null and b/pics/111521119203347.gif differ
diff --git a/pics/111521119368714.gif b/pics/111521119368714.gif
new file mode 100644
index 00000000..216c3033
Binary files /dev/null and b/pics/111521119368714.gif differ
diff --git a/pics/114c49a6-72e3-4264-ae07-c564127094ac.png b/pics/114c49a6-72e3-4264-ae07-c564127094ac.png
deleted file mode 100644
index 8cb30058..00000000
Binary files a/pics/114c49a6-72e3-4264-ae07-c564127094ac.png and /dev/null differ
diff --git a/pics/1164a71f-413d-494a-9cc8-679fb6a2613d.jpg b/pics/1164a71f-413d-494a-9cc8-679fb6a2613d.jpg
new file mode 100644
index 00000000..1804cdb1
Binary files /dev/null and b/pics/1164a71f-413d-494a-9cc8-679fb6a2613d.jpg differ
diff --git a/pics/1202b2d6-9469-4251-bd47-ca6034fb6116.png b/pics/1202b2d6-9469-4251-bd47-ca6034fb6116.png
new file mode 100644
index 00000000..b44fa996
Binary files /dev/null and b/pics/1202b2d6-9469-4251-bd47-ca6034fb6116.png differ
diff --git a/pics/123bdf81-1ef5-48a9-a08c-2db97057b4d2.png b/pics/123bdf81-1ef5-48a9-a08c-2db97057b4d2.png
new file mode 100644
index 00000000..3c248d37
Binary files /dev/null and b/pics/123bdf81-1ef5-48a9-a08c-2db97057b4d2.png differ
diff --git a/pics/144d28a0-1dc5-4aba-8961-ced5bc88428a.jpg b/pics/144d28a0-1dc5-4aba-8961-ced5bc88428a.jpg
deleted file mode 100644
index e3b4b466..00000000
Binary files a/pics/144d28a0-1dc5-4aba-8961-ced5bc88428a.jpg and /dev/null differ
diff --git a/pics/1556770b-8c01-4681-af10-46f1df69202c.jpg b/pics/1556770b-8c01-4681-af10-46f1df69202c.jpg
new file mode 100644
index 00000000..94d1adc2
Binary files /dev/null and b/pics/1556770b-8c01-4681-af10-46f1df69202c.jpg differ
diff --git a/pics/1582217a-ed46-4cac-811e-90d13a65163b.png b/pics/1582217a-ed46-4cac-811e-90d13a65163b.png
new file mode 100644
index 00000000..c89551d1
Binary files /dev/null and b/pics/1582217a-ed46-4cac-811e-90d13a65163b.png differ
diff --git a/pics/15b45dc6-27aa-4519-9194-f4acfa2b077f.jpg b/pics/15b45dc6-27aa-4519-9194-f4acfa2b077f.jpg
new file mode 100644
index 00000000..a61ab275
Binary files /dev/null and b/pics/15b45dc6-27aa-4519-9194-f4acfa2b077f.jpg differ
diff --git a/pics/163cf8b4-5f30-46c9-af00-316a71b3c890.jpg b/pics/163cf8b4-5f30-46c9-af00-316a71b3c890.jpg
deleted file mode 100644
index b00c7858..00000000
Binary files a/pics/163cf8b4-5f30-46c9-af00-316a71b3c890.jpg and /dev/null differ
diff --git a/pics/1706ce58-a081-4fed-9b36-c3c0d7e22b3a.jpg b/pics/1706ce58-a081-4fed-9b36-c3c0d7e22b3a.jpg
deleted file mode 100644
index 4f24f5bd..00000000
Binary files a/pics/1706ce58-a081-4fed-9b36-c3c0d7e22b3a.jpg and /dev/null differ
diff --git a/pics/17976404-95f5-480e-9cb4-250e6aa1d55f.png b/pics/17976404-95f5-480e-9cb4-250e6aa1d55f.png
new file mode 100644
index 00000000..276969a7
Binary files /dev/null and b/pics/17976404-95f5-480e-9cb4-250e6aa1d55f.png differ
diff --git a/pics/17d807ef-03bf-4824-a97c-ea5fb58ec61d.jpg b/pics/17d807ef-03bf-4824-a97c-ea5fb58ec61d.jpg
deleted file mode 100644
index 2f9426b0..00000000
Binary files a/pics/17d807ef-03bf-4824-a97c-ea5fb58ec61d.jpg and /dev/null differ
diff --git a/pics/185b9c49-4c13-4241-a848-fbff85c03a64.png b/pics/185b9c49-4c13-4241-a848-fbff85c03a64.png
new file mode 100644
index 00000000..6f56bc5e
Binary files /dev/null and b/pics/185b9c49-4c13-4241-a848-fbff85c03a64.png differ
diff --git a/pics/1974a836-aa6b-4fb8-bce1-6eb11969284a.jpg b/pics/1974a836-aa6b-4fb8-bce1-6eb11969284a.jpg
deleted file mode 100644
index bf7776c4..00000000
Binary files a/pics/1974a836-aa6b-4fb8-bce1-6eb11969284a.jpg and /dev/null differ
diff --git a/pics/19f2c9ef-6739-4a95-8e9d-aa3f7654e028.jpg b/pics/19f2c9ef-6739-4a95-8e9d-aa3f7654e028.jpg
deleted file mode 100644
index 905b16ba..00000000
Binary files a/pics/19f2c9ef-6739-4a95-8e9d-aa3f7654e028.jpg and /dev/null differ
diff --git a/pics/1a511c76-bb6b-40ab-b8aa-39eeb619d673.jpg b/pics/1a511c76-bb6b-40ab-b8aa-39eeb619d673.jpg
deleted file mode 100644
index 4d753f51..00000000
Binary files a/pics/1a511c76-bb6b-40ab-b8aa-39eeb619d673.jpg and /dev/null differ
diff --git a/pics/1a851e90-0d5c-4d4f-ac54-34c20ecfb903.jpg b/pics/1a851e90-0d5c-4d4f-ac54-34c20ecfb903.jpg
new file mode 100644
index 00000000..4809984f
Binary files /dev/null and b/pics/1a851e90-0d5c-4d4f-ac54-34c20ecfb903.jpg differ
diff --git a/pics/1ab49e39-012b-4383-8284-26570987e3c4.jpg b/pics/1ab49e39-012b-4383-8284-26570987e3c4.jpg
new file mode 100644
index 00000000..48a91211
Binary files /dev/null and b/pics/1ab49e39-012b-4383-8284-26570987e3c4.jpg differ
diff --git a/pics/1b7f180e-7fee-4eaf-8ebb-164b68ae2b29.png b/pics/1b7f180e-7fee-4eaf-8ebb-164b68ae2b29.png
new file mode 100644
index 00000000..081bf5aa
Binary files /dev/null and b/pics/1b7f180e-7fee-4eaf-8ebb-164b68ae2b29.png differ
diff --git a/pics/1be8b4b0-cc7a-44d7-9c77-85be37b76f7d.png b/pics/1be8b4b0-cc7a-44d7-9c77-85be37b76f7d.png
new file mode 100644
index 00000000..6ba8797b
Binary files /dev/null and b/pics/1be8b4b0-cc7a-44d7-9c77-85be37b76f7d.png differ
diff --git a/pics/1bfa3118-f3cd-4480-a950-cf6d646015db.png b/pics/1bfa3118-f3cd-4480-a950-cf6d646015db.png
new file mode 100644
index 00000000..c28fe627
Binary files /dev/null and b/pics/1bfa3118-f3cd-4480-a950-cf6d646015db.png differ
diff --git a/pics/1c8432c8-2552-457f-b117-1da36c697221.jpg b/pics/1c8432c8-2552-457f-b117-1da36c697221.jpg
deleted file mode 100644
index f92174fb..00000000
Binary files a/pics/1c8432c8-2552-457f-b117-1da36c697221.jpg and /dev/null differ
diff --git a/pics/1c8ccf5c-7ecd-4b8a-b160-3f72a510ce26.png b/pics/1c8ccf5c-7ecd-4b8a-b160-3f72a510ce26.png
deleted file mode 100644
index 69ce5de1..00000000
Binary files a/pics/1c8ccf5c-7ecd-4b8a-b160-3f72a510ce26.png and /dev/null differ
diff --git a/pics/1dc481cc-99f6-4fa8-8f68-fbd563995bf5.png b/pics/1dc481cc-99f6-4fa8-8f68-fbd563995bf5.png
deleted file mode 100644
index 4723269f..00000000
Binary files a/pics/1dc481cc-99f6-4fa8-8f68-fbd563995bf5.png and /dev/null differ
diff --git a/pics/1dd56e61-2970-4d27-97c2-6e81cee86978.jpg b/pics/1dd56e61-2970-4d27-97c2-6e81cee86978.jpg
deleted file mode 100644
index 8e56f63a..00000000
Binary files a/pics/1dd56e61-2970-4d27-97c2-6e81cee86978.jpg and /dev/null differ
diff --git a/pics/1e09d75f-6268-4425-acf8-8ecd1b4a0ef3.jpg b/pics/1e09d75f-6268-4425-acf8-8ecd1b4a0ef3.jpg
deleted file mode 100644
index 95e3b7ee..00000000
Binary files a/pics/1e09d75f-6268-4425-acf8-8ecd1b4a0ef3.jpg and /dev/null differ
diff --git a/pics/1ea4dc9a-c4dd-46b5-bb11-49f98d57ded1.png b/pics/1ea4dc9a-c4dd-46b5-bb11-49f98d57ded1.png
deleted file mode 100644
index 71ed45ee..00000000
Binary files a/pics/1ea4dc9a-c4dd-46b5-bb11-49f98d57ded1.png and /dev/null differ
diff --git a/pics/1f039a45-6b91-4f31-a2c2-6c63eb8bdb56.png b/pics/1f039a45-6b91-4f31-a2c2-6c63eb8bdb56.png
new file mode 100644
index 00000000..eb4daeab
Binary files /dev/null and b/pics/1f039a45-6b91-4f31-a2c2-6c63eb8bdb56.png differ
diff --git a/pics/1f4a7f10-52b2-4bd7-a67d-a9581d66dc62.jpg b/pics/1f4a7f10-52b2-4bd7-a67d-a9581d66dc62.jpg
new file mode 100644
index 00000000..e4f0f3ac
Binary files /dev/null and b/pics/1f4a7f10-52b2-4bd7-a67d-a9581d66dc62.jpg differ
diff --git a/pics/20150928140509757.png b/pics/20150928140509757.png
new file mode 100644
index 00000000..60684377
Binary files /dev/null and b/pics/20150928140509757.png differ
diff --git a/pics/2017511e-22f0-4d74-873d-1261b71cf5a4.png b/pics/2017511e-22f0-4d74-873d-1261b71cf5a4.png
new file mode 100644
index 00000000..81633b51
Binary files /dev/null and b/pics/2017511e-22f0-4d74-873d-1261b71cf5a4.png differ
diff --git a/pics/2018040302.jpg b/pics/2018040302.jpg
new file mode 100644
index 00000000..27daefae
Binary files /dev/null and b/pics/2018040302.jpg differ
diff --git a/pics/207c1801-2335-4b1b-b65c-126a0ba966cb.png b/pics/207c1801-2335-4b1b-b65c-126a0ba966cb.png
new file mode 100644
index 00000000..cace90b3
Binary files /dev/null and b/pics/207c1801-2335-4b1b-b65c-126a0ba966cb.png differ
diff --git a/pics/21041ec2-babb-483f-bf47-8b8148eec162.png b/pics/21041ec2-babb-483f-bf47-8b8148eec162.png
new file mode 100644
index 00000000..7de9b57a
Binary files /dev/null and b/pics/21041ec2-babb-483f-bf47-8b8148eec162.png differ
diff --git a/pics/211c60d4-75ca-4acd-8a4f-171458ed58b4.jpg b/pics/211c60d4-75ca-4acd-8a4f-171458ed58b4.jpg
new file mode 100644
index 00000000..efb47ffc
Binary files /dev/null and b/pics/211c60d4-75ca-4acd-8a4f-171458ed58b4.jpg differ
diff --git a/pics/21a00b02-c0a6-4bcd-9af0-5ec6bb66e34c.jpg b/pics/21a00b02-c0a6-4bcd-9af0-5ec6bb66e34c.jpg
deleted file mode 100644
index 002ded06..00000000
Binary files a/pics/21a00b02-c0a6-4bcd-9af0-5ec6bb66e34c.jpg and /dev/null differ
diff --git a/pics/222768a7-914f-4d64-b874-d98f3b926fb6.jpg b/pics/222768a7-914f-4d64-b874-d98f3b926fb6.jpg
deleted file mode 100644
index 412af6de..00000000
Binary files a/pics/222768a7-914f-4d64-b874-d98f3b926fb6.jpg and /dev/null differ
diff --git a/pics/223fc26e-2fd6-484c-bcb7-443cac134f15.jpg b/pics/223fc26e-2fd6-484c-bcb7-443cac134f15.jpg
deleted file mode 100644
index e93bbd01..00000000
Binary files a/pics/223fc26e-2fd6-484c-bcb7-443cac134f15.jpg and /dev/null differ
diff --git a/pics/2279cc60-9714-4e0e-aac9-4c348e0c2165.png b/pics/2279cc60-9714-4e0e-aac9-4c348e0c2165.png
deleted file mode 100644
index 4bf0a826..00000000
Binary files a/pics/2279cc60-9714-4e0e-aac9-4c348e0c2165.png and /dev/null differ
diff --git a/pics/22b39f77-ac47-4978-91ed-84aaf457644c.jpg b/pics/22b39f77-ac47-4978-91ed-84aaf457644c.jpg
deleted file mode 100644
index 0bbf51f4..00000000
Binary files a/pics/22b39f77-ac47-4978-91ed-84aaf457644c.jpg and /dev/null differ
diff --git a/pics/2366c2ad-5859-4d4e-805f-7e2b88061cd8.jpg b/pics/2366c2ad-5859-4d4e-805f-7e2b88061cd8.jpg
deleted file mode 100644
index 744d2c42..00000000
Binary files a/pics/2366c2ad-5859-4d4e-805f-7e2b88061cd8.jpg and /dev/null differ
diff --git a/pics/23ba890e-e11c-45e2-a20c-64d217f83430.png b/pics/23ba890e-e11c-45e2-a20c-64d217f83430.png
new file mode 100644
index 00000000..5fccbd1c
Binary files /dev/null and b/pics/23ba890e-e11c-45e2-a20c-64d217f83430.png differ
diff --git a/pics/245fd2fb-209c-4ad5-bc5e-eb5664966a0e.jpg b/pics/245fd2fb-209c-4ad5-bc5e-eb5664966a0e.jpg
new file mode 100644
index 00000000..92c11bfd
Binary files /dev/null and b/pics/245fd2fb-209c-4ad5-bc5e-eb5664966a0e.jpg differ
diff --git a/pics/25226bb2-92cc-40cb-9e7f-c44e79fbb64a.jpg b/pics/25226bb2-92cc-40cb-9e7f-c44e79fbb64a.jpg
deleted file mode 100644
index ff506c8b..00000000
Binary files a/pics/25226bb2-92cc-40cb-9e7f-c44e79fbb64a.jpg and /dev/null differ
diff --git a/pics/25387681-89f8-4365-a2fa-83b86449ee84.jpg b/pics/25387681-89f8-4365-a2fa-83b86449ee84.jpg
deleted file mode 100644
index 89259924..00000000
Binary files a/pics/25387681-89f8-4365-a2fa-83b86449ee84.jpg and /dev/null differ
diff --git a/pics/253bd869-ea48-4092-9aed-6906ccb2f3b0.jpg b/pics/253bd869-ea48-4092-9aed-6906ccb2f3b0.jpg
deleted file mode 100644
index 42d8c373..00000000
Binary files a/pics/253bd869-ea48-4092-9aed-6906ccb2f3b0.jpg and /dev/null differ
diff --git a/pics/2548f2ec-7b00-4ec7-b286-20fc3022e084.jpg b/pics/2548f2ec-7b00-4ec7-b286-20fc3022e084.jpg
deleted file mode 100644
index c6d9cd68..00000000
Binary files a/pics/2548f2ec-7b00-4ec7-b286-20fc3022e084.jpg and /dev/null differ
diff --git a/pics/25b8adad-2ef6-4f30-9012-c306b4e49897.png b/pics/25b8adad-2ef6-4f30-9012-c306b4e49897.png
deleted file mode 100644
index 1d4c7e2f..00000000
Binary files a/pics/25b8adad-2ef6-4f30-9012-c306b4e49897.png and /dev/null differ
diff --git a/pics/265a355d-aead-48aa-b455-f33b62fe729f.png b/pics/265a355d-aead-48aa-b455-f33b62fe729f.png
new file mode 100644
index 00000000..bb564f6e
Binary files /dev/null and b/pics/265a355d-aead-48aa-b455-f33b62fe729f.png differ
diff --git a/pics/26772ecc-a3e3-4ab7-a46f-8b4656990c27.jpg b/pics/26772ecc-a3e3-4ab7-a46f-8b4656990c27.jpg
deleted file mode 100644
index 38c0da24..00000000
Binary files a/pics/26772ecc-a3e3-4ab7-a46f-8b4656990c27.jpg and /dev/null differ
diff --git a/pics/26cb5e7e-6fa3-44ad-854e-fe24d1a5278c.jpg b/pics/26cb5e7e-6fa3-44ad-854e-fe24d1a5278c.jpg
deleted file mode 100644
index 2f7b925b..00000000
Binary files a/pics/26cb5e7e-6fa3-44ad-854e-fe24d1a5278c.jpg and /dev/null differ
diff --git a/pics/2719067e-b299-4639-9065-bed6729dbf0b.png b/pics/2719067e-b299-4639-9065-bed6729dbf0b.png
new file mode 100644
index 00000000..95057e05
Binary files /dev/null and b/pics/2719067e-b299-4639-9065-bed6729dbf0b.png differ
diff --git a/pics/2766d04f-7dad-42e4-99d1-60682c9d5c61.jpg b/pics/2766d04f-7dad-42e4-99d1-60682c9d5c61.jpg
new file mode 100644
index 00000000..f9a9489b
Binary files /dev/null and b/pics/2766d04f-7dad-42e4-99d1-60682c9d5c61.jpg differ
diff --git a/pics/276c31df-3b28-4ac2-b006-1e80fc86a64f.jpg b/pics/276c31df-3b28-4ac2-b006-1e80fc86a64f.jpg
new file mode 100644
index 00000000..8676c440
Binary files /dev/null and b/pics/276c31df-3b28-4ac2-b006-1e80fc86a64f.jpg differ
diff --git a/pics/27ace615-558f-4dfb-8ad4-7ac769c10118.jpg b/pics/27ace615-558f-4dfb-8ad4-7ac769c10118.jpg
deleted file mode 100644
index 1ae556ba..00000000
Binary files a/pics/27ace615-558f-4dfb-8ad4-7ac769c10118.jpg and /dev/null differ
diff --git a/pics/27c2e0b3-8f95-453d-bedc-6398a8566ce9.jpg b/pics/27c2e0b3-8f95-453d-bedc-6398a8566ce9.jpg
deleted file mode 100644
index 428a8fd6..00000000
Binary files a/pics/27c2e0b3-8f95-453d-bedc-6398a8566ce9.jpg and /dev/null differ
diff --git a/pics/27cd6f0c-f581-45da-b8c9-fed026830560.png b/pics/27cd6f0c-f581-45da-b8c9-fed026830560.png
new file mode 100644
index 00000000..40775ac8
Binary files /dev/null and b/pics/27cd6f0c-f581-45da-b8c9-fed026830560.png differ
diff --git a/pics/29058e09-bb72-4040-a73d-4c497895e9ce.jpg b/pics/29058e09-bb72-4040-a73d-4c497895e9ce.jpg
deleted file mode 100644
index eefed273..00000000
Binary files a/pics/29058e09-bb72-4040-a73d-4c497895e9ce.jpg and /dev/null differ
diff --git a/pics/293b9326-02fc-4ad8-8c79-b4a7b5ba60d3.png b/pics/293b9326-02fc-4ad8-8c79-b4a7b5ba60d3.png
deleted file mode 100644
index 50900e44..00000000
Binary files a/pics/293b9326-02fc-4ad8-8c79-b4a7b5ba60d3.png and /dev/null differ
diff --git a/pics/293d2af9-de1d-403e-bed0-85d029383528.png b/pics/293d2af9-de1d-403e-bed0-85d029383528.png
new file mode 100644
index 00000000..36765e32
Binary files /dev/null and b/pics/293d2af9-de1d-403e-bed0-85d029383528.png differ
diff --git a/pics/29574e6f-295c-444e-83c7-b162e8a73a83.jpg b/pics/29574e6f-295c-444e-83c7-b162e8a73a83.jpg
deleted file mode 100644
index 11ab909d..00000000
Binary files a/pics/29574e6f-295c-444e-83c7-b162e8a73a83.jpg and /dev/null differ
diff --git a/pics/29badd92-109f-4f29-abb9-9857f5973928.png b/pics/29badd92-109f-4f29-abb9-9857f5973928.png
deleted file mode 100644
index 3581cad4..00000000
Binary files a/pics/29badd92-109f-4f29-abb9-9857f5973928.png and /dev/null differ
diff --git a/pics/2_14_microkernelArchitecture.jpg b/pics/2_14_microkernelArchitecture.jpg
new file mode 100644
index 00000000..21c2a58f
Binary files /dev/null and b/pics/2_14_microkernelArchitecture.jpg differ
diff --git a/pics/2a40042a-03c8-4556-ad1f-72d89f8c555c.jpg b/pics/2a40042a-03c8-4556-ad1f-72d89f8c555c.jpg
deleted file mode 100644
index 605a1f0a..00000000
Binary files a/pics/2a40042a-03c8-4556-ad1f-72d89f8c555c.jpg and /dev/null differ
diff --git a/pics/2a502516-5d34-4eef-8f39-916298a60035.png b/pics/2a502516-5d34-4eef-8f39-916298a60035.png
deleted file mode 100644
index 3bc4ccb9..00000000
Binary files a/pics/2a502516-5d34-4eef-8f39-916298a60035.png and /dev/null differ
diff --git a/pics/2ad244f5-939c-49fa-9385-69bc688677ab.jpg b/pics/2ad244f5-939c-49fa-9385-69bc688677ab.jpg
new file mode 100644
index 00000000..5c1e0af9
Binary files /dev/null and b/pics/2ad244f5-939c-49fa-9385-69bc688677ab.jpg differ
diff --git a/pics/2b3410f1-9559-4dd1-bc3d-e3e572247be2.png b/pics/2b3410f1-9559-4dd1-bc3d-e3e572247be2.png
deleted file mode 100644
index 9ba05bb5..00000000
Binary files a/pics/2b3410f1-9559-4dd1-bc3d-e3e572247be2.png and /dev/null differ
diff --git a/pics/2bf2fd8f-5ade-48ba-a2b3-74195ac77c4b.png b/pics/2bf2fd8f-5ade-48ba-a2b3-74195ac77c4b.png
new file mode 100644
index 00000000..590a4299
Binary files /dev/null and b/pics/2bf2fd8f-5ade-48ba-a2b3-74195ac77c4b.png differ
diff --git a/pics/2c4556e4-0751-4377-ab08-e7b89d697ca7.png b/pics/2c4556e4-0751-4377-ab08-e7b89d697ca7.png
new file mode 100644
index 00000000..c7f87138
Binary files /dev/null and b/pics/2c4556e4-0751-4377-ab08-e7b89d697ca7.png differ
diff --git a/pics/2c968ec5-0967-49ce-ac06-f8f5c9ab33bc.jpg b/pics/2c968ec5-0967-49ce-ac06-f8f5c9ab33bc.jpg
deleted file mode 100644
index c3475c0d..00000000
Binary files a/pics/2c968ec5-0967-49ce-ac06-f8f5c9ab33bc.jpg and /dev/null differ
diff --git a/pics/2cdc3ce2-fa82-4c22-baaa-000c07d10473.jpg b/pics/2cdc3ce2-fa82-4c22-baaa-000c07d10473.jpg
deleted file mode 100644
index 6e841182..00000000
Binary files a/pics/2cdc3ce2-fa82-4c22-baaa-000c07d10473.jpg and /dev/null differ
diff --git a/pics/2d09a847-b854-439c-9198-b29c65810944.png b/pics/2d09a847-b854-439c-9198-b29c65810944.png
new file mode 100644
index 00000000..384f7ef0
Binary files /dev/null and b/pics/2d09a847-b854-439c-9198-b29c65810944.png differ
diff --git a/pics/2ddd6132-60be-4a72-9daa-3d9756191f4a.png b/pics/2ddd6132-60be-4a72-9daa-3d9756191f4a.png
deleted file mode 100644
index 3590549f..00000000
Binary files a/pics/2ddd6132-60be-4a72-9daa-3d9756191f4a.png and /dev/null differ
diff --git a/pics/2e5620c4-b558-46fe-8f12-00c9dd597a61.png b/pics/2e5620c4-b558-46fe-8f12-00c9dd597a61.png
deleted file mode 100644
index 5a17489f..00000000
Binary files a/pics/2e5620c4-b558-46fe-8f12-00c9dd597a61.png and /dev/null differ
diff --git a/pics/2e6c72f5-3b8e-4e32-b87b-9491322628fe.png b/pics/2e6c72f5-3b8e-4e32-b87b-9491322628fe.png
new file mode 100644
index 00000000..701e37f2
Binary files /dev/null and b/pics/2e6c72f5-3b8e-4e32-b87b-9491322628fe.png differ
diff --git a/pics/30210b86-472d-4574-abb6-b74898cc17a4.jpg b/pics/30210b86-472d-4574-abb6-b74898cc17a4.jpg
new file mode 100644
index 00000000..7726cf8c
Binary files /dev/null and b/pics/30210b86-472d-4574-abb6-b74898cc17a4.jpg differ
diff --git a/pics/3086c248-b552-499e-b101-9cffe5c2773e.png b/pics/3086c248-b552-499e-b101-9cffe5c2773e.png
new file mode 100644
index 00000000..c23e85c9
Binary files /dev/null and b/pics/3086c248-b552-499e-b101-9cffe5c2773e.png differ
diff --git a/pics/31d99967-1171-448e-8531-bccf5c14cffe.jpg b/pics/31d99967-1171-448e-8531-bccf5c14cffe.jpg
new file mode 100644
index 00000000..61e00649
Binary files /dev/null and b/pics/31d99967-1171-448e-8531-bccf5c14cffe.jpg differ
diff --git a/pics/323ffd6c-8b54-4f3e-b361-555a6c8bf218.png b/pics/323ffd6c-8b54-4f3e-b361-555a6c8bf218.png
new file mode 100644
index 00000000..3316254e
Binary files /dev/null and b/pics/323ffd6c-8b54-4f3e-b361-555a6c8bf218.png differ
diff --git a/pics/3294ff06-f942-425e-aecc-ca04e45566d4.png b/pics/3294ff06-f942-425e-aecc-ca04e45566d4.png
deleted file mode 100644
index 52d4305b..00000000
Binary files a/pics/3294ff06-f942-425e-aecc-ca04e45566d4.png and /dev/null differ
diff --git a/pics/33821037-dc40-4266-901c-e5b38e618426.png b/pics/33821037-dc40-4266-901c-e5b38e618426.png
deleted file mode 100644
index ef726e15..00000000
Binary files a/pics/33821037-dc40-4266-901c-e5b38e618426.png and /dev/null differ
diff --git a/pics/33a4e822-2dd0-481e-ac89-7f6161034402.jpg b/pics/33a4e822-2dd0-481e-ac89-7f6161034402.jpg
deleted file mode 100644
index 86deea69..00000000
Binary files a/pics/33a4e822-2dd0-481e-ac89-7f6161034402.jpg and /dev/null differ
diff --git a/pics/33ac2b23-cb85-4e99-bc41-b7b7199fad1c.png b/pics/33ac2b23-cb85-4e99-bc41-b7b7199fad1c.png
new file mode 100644
index 00000000..6e8383f3
Binary files /dev/null and b/pics/33ac2b23-cb85-4e99-bc41-b7b7199fad1c.png differ
diff --git a/pics/3402d1c0-7020-4249-9a7f-12ea2ea6adf7.jpg b/pics/3402d1c0-7020-4249-9a7f-12ea2ea6adf7.jpg
deleted file mode 100644
index 8a1c0cc7..00000000
Binary files a/pics/3402d1c0-7020-4249-9a7f-12ea2ea6adf7.jpg and /dev/null differ
diff --git a/pics/341c632a-1fc1-4068-9b9f-bf7ef68ebb4c.jpg b/pics/341c632a-1fc1-4068-9b9f-bf7ef68ebb4c.jpg
deleted file mode 100644
index 8dbd591e..00000000
Binary files a/pics/341c632a-1fc1-4068-9b9f-bf7ef68ebb4c.jpg and /dev/null differ
diff --git a/pics/34259bb8-ca3a-4872-8771-9e946782d9c3.png b/pics/34259bb8-ca3a-4872-8771-9e946782d9c3.png
deleted file mode 100644
index db57712b..00000000
Binary files a/pics/34259bb8-ca3a-4872-8771-9e946782d9c3.png and /dev/null differ
diff --git a/pics/346244ff-98c1-4f12-9a87-d0832e8c04cf.jpg b/pics/346244ff-98c1-4f12-9a87-d0832e8c04cf.jpg
deleted file mode 100644
index a25da37c..00000000
Binary files a/pics/346244ff-98c1-4f12-9a87-d0832e8c04cf.jpg and /dev/null differ
diff --git a/pics/348bc2db-582e-4aca-9f88-38c40e9a0e69.png b/pics/348bc2db-582e-4aca-9f88-38c40e9a0e69.png
new file mode 100644
index 00000000..cb7e3681
Binary files /dev/null and b/pics/348bc2db-582e-4aca-9f88-38c40e9a0e69.png differ
diff --git a/pics/3646544a-cb57-451d-9e03-d3c4f5e4434a.png b/pics/3646544a-cb57-451d-9e03-d3c4f5e4434a.png
new file mode 100644
index 00000000..76d45e19
Binary files /dev/null and b/pics/3646544a-cb57-451d-9e03-d3c4f5e4434a.png differ
diff --git a/pics/37a72755-4890-4b42-9eab-b0084e0c54d9.png b/pics/37a72755-4890-4b42-9eab-b0084e0c54d9.png
new file mode 100644
index 00000000..c74c59ef
Binary files /dev/null and b/pics/37a72755-4890-4b42-9eab-b0084e0c54d9.png differ
diff --git a/pics/37b74a34-251c-45f8-88a4-614ec953f7e9.png b/pics/37b74a34-251c-45f8-88a4-614ec953f7e9.png
deleted file mode 100644
index 4bac48ab..00000000
Binary files a/pics/37b74a34-251c-45f8-88a4-614ec953f7e9.png and /dev/null differ
diff --git a/pics/386cd64f-7a9d-40e6-8c55-22b90ee2d258.png b/pics/386cd64f-7a9d-40e6-8c55-22b90ee2d258.png
new file mode 100644
index 00000000..5f5f5636
Binary files /dev/null and b/pics/386cd64f-7a9d-40e6-8c55-22b90ee2d258.png differ
diff --git a/pics/390c913b-5f31-444f-bbdb-2b88b688e7ce.jpg b/pics/390c913b-5f31-444f-bbdb-2b88b688e7ce.jpg
new file mode 100644
index 00000000..78eb732b
Binary files /dev/null and b/pics/390c913b-5f31-444f-bbdb-2b88b688e7ce.jpg differ
diff --git a/pics/3939369b-3a4a-48a0-b9eb-3efae26dd400.png b/pics/3939369b-3a4a-48a0-b9eb-3efae26dd400.png
new file mode 100644
index 00000000..c2cf9d1b
Binary files /dev/null and b/pics/3939369b-3a4a-48a0-b9eb-3efae26dd400.png differ
diff --git a/pics/395a9e83-b1a1-4a1d-b170-d081e7bb5bab.png b/pics/395a9e83-b1a1-4a1d-b170-d081e7bb5bab.png
new file mode 100644
index 00000000..b486ec02
Binary files /dev/null and b/pics/395a9e83-b1a1-4a1d-b170-d081e7bb5bab.png differ
diff --git a/pics/39ccb299-ee99-4dd1-b8b4-2f9ec9495cb4.png b/pics/39ccb299-ee99-4dd1-b8b4-2f9ec9495cb4.png
new file mode 100644
index 00000000..8e363e47
Binary files /dev/null and b/pics/39ccb299-ee99-4dd1-b8b4-2f9ec9495cb4.png differ
diff --git a/pics/3a676c54-b559-4466-9b21-eb10f1e25879.jpg b/pics/3a676c54-b559-4466-9b21-eb10f1e25879.jpg
deleted file mode 100644
index 9c7b10d5..00000000
Binary files a/pics/3a676c54-b559-4466-9b21-eb10f1e25879.jpg and /dev/null differ
diff --git a/pics/3b0d1aa8-d0e0-46c2-8fd1-736bf08a11aa.jpg b/pics/3b0d1aa8-d0e0-46c2-8fd1-736bf08a11aa.jpg
new file mode 100644
index 00000000..f7e9f145
Binary files /dev/null and b/pics/3b0d1aa8-d0e0-46c2-8fd1-736bf08a11aa.jpg differ
diff --git a/pics/3cd630ea-017c-488d-ad1d-732b4efeddf5.png b/pics/3cd630ea-017c-488d-ad1d-732b4efeddf5.png
new file mode 100644
index 00000000..9dc77733
Binary files /dev/null and b/pics/3cd630ea-017c-488d-ad1d-732b4efeddf5.png differ
diff --git a/pics/3dc454fb-efd4-4eb8-afde-785b2182caeb.jpg b/pics/3dc454fb-efd4-4eb8-afde-785b2182caeb.jpg
deleted file mode 100644
index 36df547d..00000000
Binary files a/pics/3dc454fb-efd4-4eb8-afde-785b2182caeb.jpg and /dev/null differ
diff --git a/pics/3e2200b3-1c18-4853-ae42-7788e8e1f939.png b/pics/3e2200b3-1c18-4853-ae42-7788e8e1f939.png
deleted file mode 100644
index 97b0d441..00000000
Binary files a/pics/3e2200b3-1c18-4853-ae42-7788e8e1f939.png and /dev/null differ
diff --git a/pics/3f5bba4b-7813-4aea-b578-970c7e3f6bf3.jpg b/pics/3f5bba4b-7813-4aea-b578-970c7e3f6bf3.jpg
new file mode 100644
index 00000000..c9702daf
Binary files /dev/null and b/pics/3f5bba4b-7813-4aea-b578-970c7e3f6bf3.jpg differ
diff --git a/pics/3f8d8c9d-a9a9-4d7a-813c-2de05ee5a97e.jpg b/pics/3f8d8c9d-a9a9-4d7a-813c-2de05ee5a97e.jpg
deleted file mode 100644
index af314b78..00000000
Binary files a/pics/3f8d8c9d-a9a9-4d7a-813c-2de05ee5a97e.jpg and /dev/null differ
diff --git a/pics/40639782-5df2-4e96-a4f3-f9dd664d0ca1.jpg b/pics/40639782-5df2-4e96-a4f3-f9dd664d0ca1.jpg
deleted file mode 100644
index c6a7115c..00000000
Binary files a/pics/40639782-5df2-4e96-a4f3-f9dd664d0ca1.jpg and /dev/null differ
diff --git a/pics/40c3f8e5-3a20-45b6-a60c-77b9b952e104.jpg b/pics/40c3f8e5-3a20-45b6-a60c-77b9b952e104.jpg
deleted file mode 100644
index dd57e354..00000000
Binary files a/pics/40c3f8e5-3a20-45b6-a60c-77b9b952e104.jpg and /dev/null differ
diff --git a/pics/4102b7d0-39b9-48d8-82ae-ac4addb7ebfb.jpg b/pics/4102b7d0-39b9-48d8-82ae-ac4addb7ebfb.jpg
deleted file mode 100644
index a319bbb8..00000000
Binary files a/pics/4102b7d0-39b9-48d8-82ae-ac4addb7ebfb.jpg and /dev/null differ
diff --git a/pics/41392d76-dd1d-4712-85d9-e8bb46b04a2d.png b/pics/41392d76-dd1d-4712-85d9-e8bb46b04a2d.png
new file mode 100644
index 00000000..d2fd2c63
Binary files /dev/null and b/pics/41392d76-dd1d-4712-85d9-e8bb46b04a2d.png differ
diff --git a/pics/4146e14b-56b9-433c-8e3d-74b1b325399c.jpg b/pics/4146e14b-56b9-433c-8e3d-74b1b325399c.jpg
deleted file mode 100644
index c683acde..00000000
Binary files a/pics/4146e14b-56b9-433c-8e3d-74b1b325399c.jpg and /dev/null differ
diff --git a/pics/41a4cb30-f393-4b3b-abe4-9941ccf8fa1f.jpg b/pics/41a4cb30-f393-4b3b-abe4-9941ccf8fa1f.jpg
deleted file mode 100644
index e6880589..00000000
Binary files a/pics/41a4cb30-f393-4b3b-abe4-9941ccf8fa1f.jpg and /dev/null differ
diff --git a/pics/426df589-6f97-4622-b74d-4a81fcb1da8e.png b/pics/426df589-6f97-4622-b74d-4a81fcb1da8e.png
new file mode 100644
index 00000000..98327bba
Binary files /dev/null and b/pics/426df589-6f97-4622-b74d-4a81fcb1da8e.png differ
diff --git a/pics/439deca7-fed0-4c89-87e5-7088d10f1fdb.jpg b/pics/439deca7-fed0-4c89-87e5-7088d10f1fdb.jpg
deleted file mode 100644
index 5a0e13e1..00000000
Binary files a/pics/439deca7-fed0-4c89-87e5-7088d10f1fdb.jpg and /dev/null differ
diff --git a/pics/43f2cafa-3568-4a89-a895-4725666b94a6.png b/pics/43f2cafa-3568-4a89-a895-4725666b94a6.png
new file mode 100644
index 00000000..010278a2
Binary files /dev/null and b/pics/43f2cafa-3568-4a89-a895-4725666b94a6.png differ
diff --git a/pics/4440ad24-625b-489a-96c1-e5ab1b06a30f.png b/pics/4440ad24-625b-489a-96c1-e5ab1b06a30f.png
deleted file mode 100644
index fb68a74b..00000000
Binary files a/pics/4440ad24-625b-489a-96c1-e5ab1b06a30f.png and /dev/null differ
diff --git a/pics/44e1d90e-3fe6-4dd6-8dce-6daab12e7663.jpg b/pics/44e1d90e-3fe6-4dd6-8dce-6daab12e7663.jpg
deleted file mode 100644
index b20afc33..00000000
Binary files a/pics/44e1d90e-3fe6-4dd6-8dce-6daab12e7663.jpg and /dev/null differ
diff --git a/pics/44e4a7ab-215c-41a1-8e34-f55f6c09e517.jpg b/pics/44e4a7ab-215c-41a1-8e34-f55f6c09e517.jpg
new file mode 100644
index 00000000..a14ecacf
Binary files /dev/null and b/pics/44e4a7ab-215c-41a1-8e34-f55f6c09e517.jpg differ
diff --git a/pics/44edefb7-4b58-4519-b8ee-4aca01697b78.jpg b/pics/44edefb7-4b58-4519-b8ee-4aca01697b78.jpg
new file mode 100644
index 00000000..32d0f3d2
Binary files /dev/null and b/pics/44edefb7-4b58-4519-b8ee-4aca01697b78.jpg differ
diff --git a/pics/450px-HTTP_persistent_connection.svg.png b/pics/450px-HTTP_persistent_connection.svg.png
new file mode 100644
index 00000000..e4fb8ba2
Binary files /dev/null and b/pics/450px-HTTP_persistent_connection.svg.png differ
diff --git a/pics/4583e24f-424b-4d50-8a14-2c38a1827d4a.png b/pics/4583e24f-424b-4d50-8a14-2c38a1827d4a.png
new file mode 100644
index 00000000..be98ce69
Binary files /dev/null and b/pics/4583e24f-424b-4d50-8a14-2c38a1827d4a.png differ
diff --git a/pics/45c86855-9b18-4cf4-a9a7-f8b6eb78d133.png b/pics/45c86855-9b18-4cf4-a9a7-f8b6eb78d133.png
deleted file mode 100644
index 9a3ff54c..00000000
Binary files a/pics/45c86855-9b18-4cf4-a9a7-f8b6eb78d133.png and /dev/null differ
diff --git a/pics/45e0e0bf-386d-4280-a341-a0b9496c7674.png b/pics/45e0e0bf-386d-4280-a341-a0b9496c7674.png
new file mode 100644
index 00000000..32fb94aa
Binary files /dev/null and b/pics/45e0e0bf-386d-4280-a341-a0b9496c7674.png differ
diff --git a/pics/46cec213-3048-4a80-aded-fdd577542801.jpg b/pics/46cec213-3048-4a80-aded-fdd577542801.jpg
new file mode 100644
index 00000000..32e1f3d5
Binary files /dev/null and b/pics/46cec213-3048-4a80-aded-fdd577542801.jpg differ
diff --git a/pics/46f66e88-e65a-4ad0-a060-3c63fe22947c.png b/pics/46f66e88-e65a-4ad0-a060-3c63fe22947c.png
new file mode 100644
index 00000000..0dcc8706
Binary files /dev/null and b/pics/46f66e88-e65a-4ad0-a060-3c63fe22947c.png differ
diff --git a/pics/47358f87-bc4c-496f-9a90-8d696de94cee.png b/pics/47358f87-bc4c-496f-9a90-8d696de94cee.png
new file mode 100644
index 00000000..83d59359
Binary files /dev/null and b/pics/47358f87-bc4c-496f-9a90-8d696de94cee.png differ
diff --git a/pics/47ca2614-509f-476e-98fc-50ec9f9d43c0.png b/pics/47ca2614-509f-476e-98fc-50ec9f9d43c0.png
deleted file mode 100644
index 2708dc43..00000000
Binary files a/pics/47ca2614-509f-476e-98fc-50ec9f9d43c0.png and /dev/null differ
diff --git a/pics/47d98583-8bb0-45cc-812d-47eefa0a4a40.jpg b/pics/47d98583-8bb0-45cc-812d-47eefa0a4a40.jpg
new file mode 100644
index 00000000..f6ddd2c2
Binary files /dev/null and b/pics/47d98583-8bb0-45cc-812d-47eefa0a4a40.jpg differ
diff --git a/pics/485fdf34-ccf8-4185-97c6-17374ee719a0.png b/pics/485fdf34-ccf8-4185-97c6-17374ee719a0.png
new file mode 100644
index 00000000..f9f73faf
Binary files /dev/null and b/pics/485fdf34-ccf8-4185-97c6-17374ee719a0.png differ
diff --git a/pics/4885d0bc-1441-460f-bd75-a2088aa7f2d4.png b/pics/4885d0bc-1441-460f-bd75-a2088aa7f2d4.png
new file mode 100644
index 00000000..2f8dd24f
Binary files /dev/null and b/pics/4885d0bc-1441-460f-bd75-a2088aa7f2d4.png differ
diff --git a/pics/4995b547-5620-45af-89d7-10f35c9621a1.jpg b/pics/4995b547-5620-45af-89d7-10f35c9621a1.jpg
deleted file mode 100644
index 180f8031..00000000
Binary files a/pics/4995b547-5620-45af-89d7-10f35c9621a1.jpg and /dev/null differ
diff --git a/pics/4b16e1d3-3a60-472c-9756-2f31b1c48abe.png b/pics/4b16e1d3-3a60-472c-9756-2f31b1c48abe.png
deleted file mode 100644
index 9c80a0ef..00000000
Binary files a/pics/4b16e1d3-3a60-472c-9756-2f31b1c48abe.png and /dev/null differ
diff --git a/pics/4bb7ed45-ec14-4d31-9da4-94024d9d3b05.png b/pics/4bb7ed45-ec14-4d31-9da4-94024d9d3b05.png
new file mode 100644
index 00000000..c5912612
Binary files /dev/null and b/pics/4bb7ed45-ec14-4d31-9da4-94024d9d3b05.png differ
diff --git a/pics/4ccd294c-d6b2-421b-839e-d88336ff5fb7.png b/pics/4ccd294c-d6b2-421b-839e-d88336ff5fb7.png
deleted file mode 100644
index 9a35106c..00000000
Binary files a/pics/4ccd294c-d6b2-421b-839e-d88336ff5fb7.png and /dev/null differ
diff --git a/pics/4d741402-344d-4b7c-be01-e57184bcad0e.png b/pics/4d741402-344d-4b7c-be01-e57184bcad0e.png
new file mode 100644
index 00000000..a4a5e7b0
Binary files /dev/null and b/pics/4d741402-344d-4b7c-be01-e57184bcad0e.png differ
diff --git a/pics/4e760981-a0c5-4dbf-9fbf-ce963e0629fb.png b/pics/4e760981-a0c5-4dbf-9fbf-ce963e0629fb.png
new file mode 100644
index 00000000..e5768980
Binary files /dev/null and b/pics/4e760981-a0c5-4dbf-9fbf-ce963e0629fb.png differ
diff --git a/pics/4f48e806-f90b-4c09-a55f-ac0cd641c047.png b/pics/4f48e806-f90b-4c09-a55f-ac0cd641c047.png
new file mode 100644
index 00000000..649d16cd
Binary files /dev/null and b/pics/4f48e806-f90b-4c09-a55f-ac0cd641c047.png differ
diff --git a/pics/4f67611d-492f-4958-9fa0-4948010e345f.jpg b/pics/4f67611d-492f-4958-9fa0-4948010e345f.jpg
deleted file mode 100644
index ed6a317a..00000000
Binary files a/pics/4f67611d-492f-4958-9fa0-4948010e345f.jpg and /dev/null differ
diff --git a/pics/4fc032e0-ac6f-4b42-9182-ee104a25e7a1.png b/pics/4fc032e0-ac6f-4b42-9182-ee104a25e7a1.png
new file mode 100644
index 00000000..4c61977c
Binary files /dev/null and b/pics/4fc032e0-ac6f-4b42-9182-ee104a25e7a1.png differ
diff --git a/pics/518f16f2-a9f7-499a-98e1-f1dbb37b5a9a.png b/pics/518f16f2-a9f7-499a-98e1-f1dbb37b5a9a.png
new file mode 100644
index 00000000..3a1010b9
Binary files /dev/null and b/pics/518f16f2-a9f7-499a-98e1-f1dbb37b5a9a.png differ
diff --git a/pics/51e2ed95-65b8-4ae9-8af3-65602d452a25.jpg b/pics/51e2ed95-65b8-4ae9-8af3-65602d452a25.jpg
new file mode 100644
index 00000000..595cdc58
Binary files /dev/null and b/pics/51e2ed95-65b8-4ae9-8af3-65602d452a25.jpg differ
diff --git a/pics/51fb761d-8ce0-4472-92ff-2f227ac7888a.png b/pics/51fb761d-8ce0-4472-92ff-2f227ac7888a.png
new file mode 100644
index 00000000..d49b1727
Binary files /dev/null and b/pics/51fb761d-8ce0-4472-92ff-2f227ac7888a.png differ
diff --git a/pics/524a237c-ffd7-426f-99c2-929a6bf4c847.jpg b/pics/524a237c-ffd7-426f-99c2-929a6bf4c847.jpg
deleted file mode 100644
index 9f0aa101..00000000
Binary files a/pics/524a237c-ffd7-426f-99c2-929a6bf4c847.jpg and /dev/null differ
diff --git a/pics/5292faa6-0141-4638-bf0f-bb95b081dcba.jpg b/pics/5292faa6-0141-4638-bf0f-bb95b081dcba.jpg
new file mode 100644
index 00000000..32e1f054
Binary files /dev/null and b/pics/5292faa6-0141-4638-bf0f-bb95b081dcba.jpg differ
diff --git a/pics/52e1af6f-3a7a-4bee-aa8f-fcb5dacebe40.jpg b/pics/52e1af6f-3a7a-4bee-aa8f-fcb5dacebe40.jpg
new file mode 100644
index 00000000..809abead
Binary files /dev/null and b/pics/52e1af6f-3a7a-4bee-aa8f-fcb5dacebe40.jpg differ
diff --git a/pics/5341d726-ffde-4d2a-a000-46597bcc9c5a.png b/pics/5341d726-ffde-4d2a-a000-46597bcc9c5a.png
new file mode 100644
index 00000000..61161099
Binary files /dev/null and b/pics/5341d726-ffde-4d2a-a000-46597bcc9c5a.png differ
diff --git a/pics/536c6dfd-305a-4b95-b12c-28ca5e8aa043.png b/pics/536c6dfd-305a-4b95-b12c-28ca5e8aa043.png
new file mode 100644
index 00000000..29bbc9de
Binary files /dev/null and b/pics/536c6dfd-305a-4b95-b12c-28ca5e8aa043.png differ
diff --git a/pics/540133af-aaaf-4208-8f7f-33cb89ac9621.png b/pics/540133af-aaaf-4208-8f7f-33cb89ac9621.png
deleted file mode 100644
index 48ad4cae..00000000
Binary files a/pics/540133af-aaaf-4208-8f7f-33cb89ac9621.png and /dev/null differ
diff --git a/pics/540631a4-6018-40a5-aed7-081e2eeeaeea.png b/pics/540631a4-6018-40a5-aed7-081e2eeeaeea.png
new file mode 100644
index 00000000..e22b2c83
Binary files /dev/null and b/pics/540631a4-6018-40a5-aed7-081e2eeeaeea.png differ
diff --git a/pics/543d47a1-f0dd-414f-b23c-0c142c814854.png b/pics/543d47a1-f0dd-414f-b23c-0c142c814854.png
deleted file mode 100644
index 5a3bbd02..00000000
Binary files a/pics/543d47a1-f0dd-414f-b23c-0c142c814854.png and /dev/null differ
diff --git a/pics/54cb3f21-485b-4159-8bf5-dcde1c4d4c36.png b/pics/54cb3f21-485b-4159-8bf5-dcde1c4d4c36.png
new file mode 100644
index 00000000..3a69d2ce
Binary files /dev/null and b/pics/54cb3f21-485b-4159-8bf5-dcde1c4d4c36.png differ
diff --git a/pics/5510045a-8f32-487f-a756-463e51a6dab0.png b/pics/5510045a-8f32-487f-a756-463e51a6dab0.png
deleted file mode 100644
index 597ba533..00000000
Binary files a/pics/5510045a-8f32-487f-a756-463e51a6dab0.png and /dev/null differ
diff --git a/pics/55dc4e84-573d-4c13-a765-52ed1dd251f9.png b/pics/55dc4e84-573d-4c13-a765-52ed1dd251f9.png
new file mode 100644
index 00000000..525b323a
Binary files /dev/null and b/pics/55dc4e84-573d-4c13-a765-52ed1dd251f9.png differ
diff --git a/pics/58633775-8584-4a01-ad3f-eee4d9a466e1.jpg b/pics/58633775-8584-4a01-ad3f-eee4d9a466e1.jpg
deleted file mode 100644
index 15f2d82c..00000000
Binary files a/pics/58633775-8584-4a01-ad3f-eee4d9a466e1.jpg and /dev/null differ
diff --git a/pics/58e57a21-6b6b-40b6-af85-956dd4e0f55a.jpg b/pics/58e57a21-6b6b-40b6-af85-956dd4e0f55a.jpg
new file mode 100644
index 00000000..a695be3a
Binary files /dev/null and b/pics/58e57a21-6b6b-40b6-af85-956dd4e0f55a.jpg differ
diff --git a/pics/5930aeb8-847d-4e9f-a168-9334d7dec744.png b/pics/5930aeb8-847d-4e9f-a168-9334d7dec744.png
new file mode 100644
index 00000000..6c9a572e
Binary files /dev/null and b/pics/5930aeb8-847d-4e9f-a168-9334d7dec744.png differ
diff --git a/pics/5942debd-fc00-477a-b390-7c5692cc8070.jpg b/pics/5942debd-fc00-477a-b390-7c5692cc8070.jpg
new file mode 100644
index 00000000..62b39e4b
Binary files /dev/null and b/pics/5942debd-fc00-477a-b390-7c5692cc8070.jpg differ
diff --git a/pics/5994928c-3d2d-45bd-abb1-adc4f5f4d775.jpg b/pics/5994928c-3d2d-45bd-abb1-adc4f5f4d775.jpg
deleted file mode 100644
index 7d64c8b8..00000000
Binary files a/pics/5994928c-3d2d-45bd-abb1-adc4f5f4d775.jpg and /dev/null differ
diff --git a/pics/5999e5de-7c16-4b52-b3aa-6dc7b58c7894.png b/pics/5999e5de-7c16-4b52-b3aa-6dc7b58c7894.png
new file mode 100644
index 00000000..a61a7dc4
Binary files /dev/null and b/pics/5999e5de-7c16-4b52-b3aa-6dc7b58c7894.png differ
diff --git a/pics/59aff6c1-8bc5-48e4-9e9c-082baeb2f274.jpg b/pics/59aff6c1-8bc5-48e4-9e9c-082baeb2f274.jpg
deleted file mode 100644
index 6c600949..00000000
Binary files a/pics/59aff6c1-8bc5-48e4-9e9c-082baeb2f274.jpg and /dev/null differ
diff --git a/pics/5aa82b89-f266-44da-887d-18f31f01d8ef.png b/pics/5aa82b89-f266-44da-887d-18f31f01d8ef.png
new file mode 100644
index 00000000..68310502
Binary files /dev/null and b/pics/5aa82b89-f266-44da-887d-18f31f01d8ef.png differ
diff --git a/pics/5aac64d3-2c7b-4f32-9e9a-1df2186f588b.png b/pics/5aac64d3-2c7b-4f32-9e9a-1df2186f588b.png
new file mode 100644
index 00000000..f457a99d
Binary files /dev/null and b/pics/5aac64d3-2c7b-4f32-9e9a-1df2186f588b.png differ
diff --git a/pics/5acf7550-86c5-4c5b-b912-8ce70ef9c34e.png b/pics/5acf7550-86c5-4c5b-b912-8ce70ef9c34e.png
new file mode 100644
index 00000000..e209482b
Binary files /dev/null and b/pics/5acf7550-86c5-4c5b-b912-8ce70ef9c34e.png differ
diff --git a/pics/5c558190-fccd-4b5e-98ed-1896653fc97f.jpg b/pics/5c558190-fccd-4b5e-98ed-1896653fc97f.jpg
deleted file mode 100644
index 5788fc85..00000000
Binary files a/pics/5c558190-fccd-4b5e-98ed-1896653fc97f.jpg and /dev/null differ
diff --git a/pics/5d387d02-6f96-44d6-b5d0-4538349f868e.png b/pics/5d387d02-6f96-44d6-b5d0-4538349f868e.png
deleted file mode 100644
index f37e74ef..00000000
Binary files a/pics/5d387d02-6f96-44d6-b5d0-4538349f868e.png and /dev/null differ
diff --git a/pics/5d4a5181-65fb-4bf2-a9c6-899cab534b44.png b/pics/5d4a5181-65fb-4bf2-a9c6-899cab534b44.png
new file mode 100644
index 00000000..110bea35
Binary files /dev/null and b/pics/5d4a5181-65fb-4bf2-a9c6-899cab534b44.png differ
diff --git a/pics/5e0cef33-4087-4f21-a428-16d5fddda671.jpg b/pics/5e0cef33-4087-4f21-a428-16d5fddda671.jpg
deleted file mode 100644
index 85d2d7b5..00000000
Binary files a/pics/5e0cef33-4087-4f21-a428-16d5fddda671.jpg and /dev/null differ
diff --git a/pics/5e9b10f3-9504-4483-9667-d4770adebf9f.png b/pics/5e9b10f3-9504-4483-9667-d4770adebf9f.png
new file mode 100644
index 00000000..4ac499de
Binary files /dev/null and b/pics/5e9b10f3-9504-4483-9667-d4770adebf9f.png differ
diff --git a/pics/5ef94f62-98ce-464d-a646-842d9c72c8b8.jpg b/pics/5ef94f62-98ce-464d-a646-842d9c72c8b8.jpg
deleted file mode 100644
index 6d083c11..00000000
Binary files a/pics/5ef94f62-98ce-464d-a646-842d9c72c8b8.jpg and /dev/null differ
diff --git a/pics/5f5ef0b6-98ea-497c-a007-f6c55288eab1.png b/pics/5f5ef0b6-98ea-497c-a007-f6c55288eab1.png
new file mode 100644
index 00000000..a3ea0a24
Binary files /dev/null and b/pics/5f5ef0b6-98ea-497c-a007-f6c55288eab1.png differ
diff --git a/pics/5f96e565-0693-47df-80f1-29e4271057b7.png b/pics/5f96e565-0693-47df-80f1-29e4271057b7.png
new file mode 100644
index 00000000..4da1e286
Binary files /dev/null and b/pics/5f96e565-0693-47df-80f1-29e4271057b7.png differ
diff --git a/pics/600px-Sharedmem.jpg b/pics/600px-Sharedmem.jpg
new file mode 100644
index 00000000..76cb2a48
Binary files /dev/null and b/pics/600px-Sharedmem.jpg differ
diff --git a/pics/6019b2db-bc3e-4408-b6d8-96025f4481d6.png b/pics/6019b2db-bc3e-4408-b6d8-96025f4481d6.png
new file mode 100644
index 00000000..900ee963
Binary files /dev/null and b/pics/6019b2db-bc3e-4408-b6d8-96025f4481d6.png differ
diff --git a/pics/61b4832d-71f3-413c-84b6-237e219b9fdc.png b/pics/61b4832d-71f3-413c-84b6-237e219b9fdc.png
deleted file mode 100644
index 70be236c..00000000
Binary files a/pics/61b4832d-71f3-413c-84b6-237e219b9fdc.png and /dev/null differ
diff --git a/pics/62077f5d-a06d-4129-9b43-78715b82cb03.png b/pics/62077f5d-a06d-4129-9b43-78715b82cb03.png
deleted file mode 100644
index 58ae8ca7..00000000
Binary files a/pics/62077f5d-a06d-4129-9b43-78715b82cb03.png and /dev/null differ
diff --git a/pics/62ebbb63-8fd7-4488-a866-76a9dc911662.png b/pics/62ebbb63-8fd7-4488-a866-76a9dc911662.png
deleted file mode 100644
index 805c69d0..00000000
Binary files a/pics/62ebbb63-8fd7-4488-a866-76a9dc911662.png and /dev/null differ
diff --git a/pics/6468a541-3a9a-4008-82b6-03a0fe941d2a.png b/pics/6468a541-3a9a-4008-82b6-03a0fe941d2a.png
deleted file mode 100644
index bd027b48..00000000
Binary files a/pics/6468a541-3a9a-4008-82b6-03a0fe941d2a.png and /dev/null differ
diff --git a/pics/64b95403-d976-421a-8b45-bac89c0b5185.jpg b/pics/64b95403-d976-421a-8b45-bac89c0b5185.jpg
deleted file mode 100644
index 5486d7c1..00000000
Binary files a/pics/64b95403-d976-421a-8b45-bac89c0b5185.jpg and /dev/null differ
diff --git a/pics/654acfed-a6a5-4fc7-8f40-3fdcae57bae8.jpg b/pics/654acfed-a6a5-4fc7-8f40-3fdcae57bae8.jpg
new file mode 100644
index 00000000..ed98e7e6
Binary files /dev/null and b/pics/654acfed-a6a5-4fc7-8f40-3fdcae57bae8.jpg differ
diff --git a/pics/66192382-558b-4b05-a35d-ac4a2b1a9811.jpg b/pics/66192382-558b-4b05-a35d-ac4a2b1a9811.jpg
new file mode 100644
index 00000000..eec226c5
Binary files /dev/null and b/pics/66192382-558b-4b05-a35d-ac4a2b1a9811.jpg differ
diff --git a/pics/664f8901-5dc7-4644-a072-dad88cc5133a.jpg b/pics/664f8901-5dc7-4644-a072-dad88cc5133a.jpg
deleted file mode 100644
index 7854f1bb..00000000
Binary files a/pics/664f8901-5dc7-4644-a072-dad88cc5133a.jpg and /dev/null differ
diff --git a/pics/68778c1b-15ab-4826-99c0-3b4fd38cb9e9.png b/pics/68778c1b-15ab-4826-99c0-3b4fd38cb9e9.png
new file mode 100644
index 00000000..39d7ec60
Binary files /dev/null and b/pics/68778c1b-15ab-4826-99c0-3b4fd38cb9e9.png differ
diff --git a/pics/688dacfe-1057-412f-b3a1-86abb5b0f914.png b/pics/688dacfe-1057-412f-b3a1-86abb5b0f914.png
new file mode 100644
index 00000000..21fa725f
Binary files /dev/null and b/pics/688dacfe-1057-412f-b3a1-86abb5b0f914.png differ
diff --git a/pics/6943e2af-5a70-4004-8bee-b33d60f39da3.jpg b/pics/6943e2af-5a70-4004-8bee-b33d60f39da3.jpg
deleted file mode 100644
index 6ab91a1a..00000000
Binary files a/pics/6943e2af-5a70-4004-8bee-b33d60f39da3.jpg and /dev/null differ
diff --git a/pics/699b4f96-d63f-46ea-a581-2b3d95eceb6a.jpg b/pics/699b4f96-d63f-46ea-a581-2b3d95eceb6a.jpg
new file mode 100644
index 00000000..313b0f92
Binary files /dev/null and b/pics/699b4f96-d63f-46ea-a581-2b3d95eceb6a.jpg differ
diff --git a/pics/69f16984-a66f-4288-82e4-79b4aa43e835.jpg b/pics/69f16984-a66f-4288-82e4-79b4aa43e835.jpg
new file mode 100644
index 00000000..03b7f3da
Binary files /dev/null and b/pics/69f16984-a66f-4288-82e4-79b4aa43e835.jpg differ
diff --git a/pics/6ab5de9b-1c1e-4118-b2c3-fb6c7ed7de6f.png b/pics/6ab5de9b-1c1e-4118-b2c3-fb6c7ed7de6f.png
deleted file mode 100644
index a1e6064e..00000000
Binary files a/pics/6ab5de9b-1c1e-4118-b2c3-fb6c7ed7de6f.png and /dev/null differ
diff --git a/pics/6bc61bb8-3b1c-4dc8-ac25-cef925ace0eb.jpg b/pics/6bc61bb8-3b1c-4dc8-ac25-cef925ace0eb.jpg
deleted file mode 100644
index 3ba4690b..00000000
Binary files a/pics/6bc61bb8-3b1c-4dc8-ac25-cef925ace0eb.jpg and /dev/null differ
diff --git a/pics/6c0f4afb-20ab-49fd-837d-8144f4e38bfd.png b/pics/6c0f4afb-20ab-49fd-837d-8144f4e38bfd.png
new file mode 100644
index 00000000..d86b4635
Binary files /dev/null and b/pics/6c0f4afb-20ab-49fd-837d-8144f4e38bfd.png differ
diff --git a/pics/6e11b122-95ce-4869-bf7d-3b0d7591707e.jpg b/pics/6e11b122-95ce-4869-bf7d-3b0d7591707e.jpg
deleted file mode 100644
index 1ebfdbdb..00000000
Binary files a/pics/6e11b122-95ce-4869-bf7d-3b0d7591707e.jpg and /dev/null differ
diff --git a/pics/6e2cb20a-8d2a-46fe-9ac7-68a2126b7bd5.jpg b/pics/6e2cb20a-8d2a-46fe-9ac7-68a2126b7bd5.jpg
deleted file mode 100644
index b2ce1e5e..00000000
Binary files a/pics/6e2cb20a-8d2a-46fe-9ac7-68a2126b7bd5.jpg and /dev/null differ
diff --git a/pics/6e9bd38c-0d23-4ce1-a1f1-8bc302165360.jpg b/pics/6e9bd38c-0d23-4ce1-a1f1-8bc302165360.jpg
deleted file mode 100644
index c0e0130b..00000000
Binary files a/pics/6e9bd38c-0d23-4ce1-a1f1-8bc302165360.jpg and /dev/null differ
diff --git a/pics/6f4abf41-3728-4a6b-9b94-85eed7ca8163.png b/pics/6f4abf41-3728-4a6b-9b94-85eed7ca8163.png
new file mode 100644
index 00000000..1fb8642f
Binary files /dev/null and b/pics/6f4abf41-3728-4a6b-9b94-85eed7ca8163.png differ
diff --git a/pics/6f4af159-8b03-4246-8d0e-222db65bb83c.jpg b/pics/6f4af159-8b03-4246-8d0e-222db65bb83c.jpg
deleted file mode 100644
index f0e0d700..00000000
Binary files a/pics/6f4af159-8b03-4246-8d0e-222db65bb83c.jpg and /dev/null differ
diff --git a/pics/6f5ed46f-86d7-4852-a34f-c1cf1b6343a0.png b/pics/6f5ed46f-86d7-4852-a34f-c1cf1b6343a0.png
deleted file mode 100644
index 7cf1c9cb..00000000
Binary files a/pics/6f5ed46f-86d7-4852-a34f-c1cf1b6343a0.png and /dev/null differ
diff --git a/pics/6fec7f56-a685-4232-b03e-c92a8dfba486.png b/pics/6fec7f56-a685-4232-b03e-c92a8dfba486.png
deleted file mode 100644
index b627cea5..00000000
Binary files a/pics/6fec7f56-a685-4232-b03e-c92a8dfba486.png and /dev/null differ
diff --git a/pics/7.gif b/pics/7.gif
new file mode 100644
index 00000000..fad88513
Binary files /dev/null and b/pics/7.gif differ
diff --git a/pics/7080a928-06ba-4e10-9792-b8dd190dc8e2.jpg b/pics/7080a928-06ba-4e10-9792-b8dd190dc8e2.jpg
deleted file mode 100644
index fb06a9c4..00000000
Binary files a/pics/7080a928-06ba-4e10-9792-b8dd190dc8e2.jpg and /dev/null differ
diff --git a/pics/70b66757-755c-4e17-a7b7-5ce808023643.png b/pics/70b66757-755c-4e17-a7b7-5ce808023643.png
deleted file mode 100644
index 0b834d6e..00000000
Binary files a/pics/70b66757-755c-4e17-a7b7-5ce808023643.png and /dev/null differ
diff --git a/pics/71363383-2d06-4c63-8b72-c01c2186707d.png b/pics/71363383-2d06-4c63-8b72-c01c2186707d.png
new file mode 100644
index 00000000..621820c3
Binary files /dev/null and b/pics/71363383-2d06-4c63-8b72-c01c2186707d.png differ
diff --git a/pics/72f0ff69-138d-4e54-b7ac-ebe025d978dc.png b/pics/72f0ff69-138d-4e54-b7ac-ebe025d978dc.png
new file mode 100644
index 00000000..5299728f
Binary files /dev/null and b/pics/72f0ff69-138d-4e54-b7ac-ebe025d978dc.png differ
diff --git a/pics/73a3983d-dd18-4373-897e-64b706a7e370.jpg b/pics/73a3983d-dd18-4373-897e-64b706a7e370.jpg
deleted file mode 100644
index 1fd0b8ef..00000000
Binary files a/pics/73a3983d-dd18-4373-897e-64b706a7e370.jpg and /dev/null differ
diff --git a/pics/73b73189-9e95-47e5-91d0-9378b8462e15.png b/pics/73b73189-9e95-47e5-91d0-9378b8462e15.png
new file mode 100644
index 00000000..3f3b32e5
Binary files /dev/null and b/pics/73b73189-9e95-47e5-91d0-9378b8462e15.png differ
diff --git a/pics/760a5d63-d96d-4dd9-bf9a-c3d126b2f401.jpg b/pics/760a5d63-d96d-4dd9-bf9a-c3d126b2f401.jpg
deleted file mode 100644
index 48eabcf6..00000000
Binary files a/pics/760a5d63-d96d-4dd9-bf9a-c3d126b2f401.jpg and /dev/null differ
diff --git a/pics/76a49594323247f21c9b3a69945445ee.png b/pics/76a49594323247f21c9b3a69945445ee.png
new file mode 100644
index 00000000..788ba0b1
Binary files /dev/null and b/pics/76a49594323247f21c9b3a69945445ee.png differ
diff --git a/pics/76dc7769-1aac-4888-9bea-064f1caa8e77.jpg b/pics/76dc7769-1aac-4888-9bea-064f1caa8e77.jpg
new file mode 100644
index 00000000..642aba6b
Binary files /dev/null and b/pics/76dc7769-1aac-4888-9bea-064f1caa8e77.jpg differ
diff --git a/pics/7779232-1e8ed39548081a1f.png b/pics/7779232-1e8ed39548081a1f.png
new file mode 100644
index 00000000..95e9027e
Binary files /dev/null and b/pics/7779232-1e8ed39548081a1f.png differ
diff --git a/pics/7779232-96822582feb08651.png b/pics/7779232-96822582feb08651.png
new file mode 100644
index 00000000..d45ba85c
Binary files /dev/null and b/pics/7779232-96822582feb08651.png differ
diff --git a/pics/77f81379-3987-4036-8d7c-93a4dcf7b05d.jpg b/pics/77f81379-3987-4036-8d7c-93a4dcf7b05d.jpg
new file mode 100644
index 00000000..bc5ddb98
Binary files /dev/null and b/pics/77f81379-3987-4036-8d7c-93a4dcf7b05d.jpg differ
diff --git a/pics/78534153-88d1-4f83-a6e0-59064dbdc43a.png b/pics/78534153-88d1-4f83-a6e0-59064dbdc43a.png
deleted file mode 100644
index e42a0de4..00000000
Binary files a/pics/78534153-88d1-4f83-a6e0-59064dbdc43a.png and /dev/null differ
diff --git a/pics/785806ed-c46b-4dca-b756-cebe7bf8ac3a.jpg b/pics/785806ed-c46b-4dca-b756-cebe7bf8ac3a.jpg
deleted file mode 100644
index a9469812..00000000
Binary files a/pics/785806ed-c46b-4dca-b756-cebe7bf8ac3a.jpg and /dev/null differ
diff --git a/pics/78f2314e-2643-41df-8f3d-b7e28294094b.jpg b/pics/78f2314e-2643-41df-8f3d-b7e28294094b.jpg
deleted file mode 100644
index 586f2ae4..00000000
Binary files a/pics/78f2314e-2643-41df-8f3d-b7e28294094b.jpg and /dev/null differ
diff --git a/pics/78f65456-666b-4044-b4ee-f7692dbbc0d3.jpg b/pics/78f65456-666b-4044-b4ee-f7692dbbc0d3.jpg
deleted file mode 100644
index 93439ff0..00000000
Binary files a/pics/78f65456-666b-4044-b4ee-f7692dbbc0d3.jpg and /dev/null differ
diff --git a/pics/79b12431-6d9d-4a7d-985b-1b79bc5bf5fb.png b/pics/79b12431-6d9d-4a7d-985b-1b79bc5bf5fb.png
new file mode 100644
index 00000000..d2a30004
Binary files /dev/null and b/pics/79b12431-6d9d-4a7d-985b-1b79bc5bf5fb.png differ
diff --git a/pics/7a29acce-f243-4914-9f00-f2988c528412.jpg b/pics/7a29acce-f243-4914-9f00-f2988c528412.jpg
new file mode 100644
index 00000000..a048d094
Binary files /dev/null and b/pics/7a29acce-f243-4914-9f00-f2988c528412.jpg differ
diff --git a/pics/7b038838-c75b-4538-ae84-6299386704e5.jpg b/pics/7b038838-c75b-4538-ae84-6299386704e5.jpg
new file mode 100644
index 00000000..919a0e58
Binary files /dev/null and b/pics/7b038838-c75b-4538-ae84-6299386704e5.jpg differ
diff --git a/pics/7b281b1e-0595-402b-ae35-8c91084c33c1.png b/pics/7b281b1e-0595-402b-ae35-8c91084c33c1.png
new file mode 100644
index 00000000..9308ecb6
Binary files /dev/null and b/pics/7b281b1e-0595-402b-ae35-8c91084c33c1.png differ
diff --git a/pics/7b3efa99-d306-4982-8cfb-e7153c33aab4.png b/pics/7b3efa99-d306-4982-8cfb-e7153c33aab4.png
new file mode 100644
index 00000000..21aafd0a
Binary files /dev/null and b/pics/7b3efa99-d306-4982-8cfb-e7153c33aab4.png differ
diff --git a/pics/7b68b142-9489-44f6-87b0-4cb5c6431e63.jpg b/pics/7b68b142-9489-44f6-87b0-4cb5c6431e63.jpg
new file mode 100644
index 00000000..d4380524
Binary files /dev/null and b/pics/7b68b142-9489-44f6-87b0-4cb5c6431e63.jpg differ
diff --git a/pics/7b877a2a-8fd1-40d8-a34c-c445827300b8.jpg b/pics/7b877a2a-8fd1-40d8-a34c-c445827300b8.jpg
deleted file mode 100644
index 5e5d19cc..00000000
Binary files a/pics/7b877a2a-8fd1-40d8-a34c-c445827300b8.jpg and /dev/null differ
diff --git a/pics/7b8f0d8e-a4fa-4c9d-b9a0-3e6a11cb3e33.jpg b/pics/7b8f0d8e-a4fa-4c9d-b9a0-3e6a11cb3e33.jpg
deleted file mode 100644
index 522090c6..00000000
Binary files a/pics/7b8f0d8e-a4fa-4c9d-b9a0-3e6a11cb3e33.jpg and /dev/null differ
diff --git a/pics/7bd202a7-93d4-4f3a-a878-af68ae25539a.png b/pics/7bd202a7-93d4-4f3a-a878-af68ae25539a.png
new file mode 100644
index 00000000..711fb45b
Binary files /dev/null and b/pics/7bd202a7-93d4-4f3a-a878-af68ae25539a.png differ
diff --git a/pics/7be0abf9-687c-4451-becd-626b0be7ec22.jpg b/pics/7be0abf9-687c-4451-becd-626b0be7ec22.jpg
new file mode 100644
index 00000000..002f50ac
Binary files /dev/null and b/pics/7be0abf9-687c-4451-becd-626b0be7ec22.jpg differ
diff --git a/pics/7c54de21-e2ff-402e-bc42-4037de1c1592.png b/pics/7c54de21-e2ff-402e-bc42-4037de1c1592.png
new file mode 100644
index 00000000..8b5ce204
Binary files /dev/null and b/pics/7c54de21-e2ff-402e-bc42-4037de1c1592.png differ
diff --git a/pics/7c98e1b6-c446-4cde-8513-5c11b9f52aea.jpg b/pics/7c98e1b6-c446-4cde-8513-5c11b9f52aea.jpg
new file mode 100644
index 00000000..7dcb6d1c
Binary files /dev/null and b/pics/7c98e1b6-c446-4cde-8513-5c11b9f52aea.jpg differ
diff --git a/pics/7dda050d-ac35-4f47-9f51-18f18ed6fa9a.png b/pics/7dda050d-ac35-4f47-9f51-18f18ed6fa9a.png
deleted file mode 100644
index a7f5e30e..00000000
Binary files a/pics/7dda050d-ac35-4f47-9f51-18f18ed6fa9a.png and /dev/null differ
diff --git a/pics/7f38a583-2f2e-4738-97af-510e6fb403a7.png b/pics/7f38a583-2f2e-4738-97af-510e6fb403a7.png
new file mode 100644
index 00000000..57e71b9c
Binary files /dev/null and b/pics/7f38a583-2f2e-4738-97af-510e6fb403a7.png differ
diff --git a/pics/7f642a65-b167-4c8f-b382-8322c6322b2c.jpg b/pics/7f642a65-b167-4c8f-b382-8322c6322b2c.jpg
new file mode 100644
index 00000000..27751a37
Binary files /dev/null and b/pics/7f642a65-b167-4c8f-b382-8322c6322b2c.jpg differ
diff --git a/pics/7f82fd18-7f16-4125-ada6-bb6b795b4fda.png b/pics/7f82fd18-7f16-4125-ada6-bb6b795b4fda.png
deleted file mode 100644
index 342fc5f0..00000000
Binary files a/pics/7f82fd18-7f16-4125-ada6-bb6b795b4fda.png and /dev/null differ
diff --git a/pics/7fcb2fb0-2cd9-4396-bc2d-282becf963c3.jpg b/pics/7fcb2fb0-2cd9-4396-bc2d-282becf963c3.jpg
deleted file mode 100644
index 8f1e02f0..00000000
Binary files a/pics/7fcb2fb0-2cd9-4396-bc2d-282becf963c3.jpg and /dev/null differ
diff --git a/pics/7fffa4b8-b36d-471f-ad0c-a88ee763bb76.png b/pics/7fffa4b8-b36d-471f-ad0c-a88ee763bb76.png
new file mode 100644
index 00000000..b636edf5
Binary files /dev/null and b/pics/7fffa4b8-b36d-471f-ad0c-a88ee763bb76.png differ
diff --git a/pics/8006a450-6c2f-498c-a928-c927f758b1d0.png b/pics/8006a450-6c2f-498c-a928-c927f758b1d0.png
new file mode 100644
index 00000000..ac453eeb
Binary files /dev/null and b/pics/8006a450-6c2f-498c-a928-c927f758b1d0.png differ
diff --git a/pics/80c5aff8-fc46-4810-aeaa-215b5c60a003.png b/pics/80c5aff8-fc46-4810-aeaa-215b5c60a003.png
deleted file mode 100644
index 2daf705a..00000000
Binary files a/pics/80c5aff8-fc46-4810-aeaa-215b5c60a003.png and /dev/null differ
diff --git a/pics/81375888-6be1-476f-9521-42eea3e3154f.jpg b/pics/81375888-6be1-476f-9521-42eea3e3154f.jpg
deleted file mode 100644
index 86f9e488..00000000
Binary files a/pics/81375888-6be1-476f-9521-42eea3e3154f.jpg and /dev/null differ
diff --git a/pics/8143787f-12eb-46ea-9bc3-c66d22d35285.jpg b/pics/8143787f-12eb-46ea-9bc3-c66d22d35285.jpg
deleted file mode 100644
index dbcffd88..00000000
Binary files a/pics/8143787f-12eb-46ea-9bc3-c66d22d35285.jpg and /dev/null differ
diff --git a/pics/81fd1d6f-a3b2-4160-9a0a-1f7cb50ba440.jpg b/pics/81fd1d6f-a3b2-4160-9a0a-1f7cb50ba440.jpg
deleted file mode 100644
index 0dfa5179..00000000
Binary files a/pics/81fd1d6f-a3b2-4160-9a0a-1f7cb50ba440.jpg and /dev/null differ
diff --git a/pics/8229e8e7-a183-4d29-94e6-e8d8537c6ce5.png b/pics/8229e8e7-a183-4d29-94e6-e8d8537c6ce5.png
new file mode 100644
index 00000000..3aaa6323
Binary files /dev/null and b/pics/8229e8e7-a183-4d29-94e6-e8d8537c6ce5.png differ
diff --git a/pics/823cdab7-3779-4e3a-a951-dc2d154e0ee6.jpg b/pics/823cdab7-3779-4e3a-a951-dc2d154e0ee6.jpg
new file mode 100644
index 00000000..48d9c267
Binary files /dev/null and b/pics/823cdab7-3779-4e3a-a951-dc2d154e0ee6.jpg differ
diff --git a/pics/8320bad6-3f91-4a15-8e3d-68e8f39649b5.png b/pics/8320bad6-3f91-4a15-8e3d-68e8f39649b5.png
deleted file mode 100644
index 00e82151..00000000
Binary files a/pics/8320bad6-3f91-4a15-8e3d-68e8f39649b5.png and /dev/null differ
diff --git a/pics/836a4eaf-4798-4e48-b52a-a3dab9435ace.png b/pics/836a4eaf-4798-4e48-b52a-a3dab9435ace.png
new file mode 100644
index 00000000..be8ab737
Binary files /dev/null and b/pics/836a4eaf-4798-4e48-b52a-a3dab9435ace.png differ
diff --git a/pics/8393f520-d824-44ea-a5f3-1c1a73d735fb.jpg b/pics/8393f520-d824-44ea-a5f3-1c1a73d735fb.jpg
deleted file mode 100644
index 3a56d4d2..00000000
Binary files a/pics/8393f520-d824-44ea-a5f3-1c1a73d735fb.jpg and /dev/null differ
diff --git a/pics/8433fbb2-c35c-45ef-831d-e3ca42aebd51.png b/pics/8433fbb2-c35c-45ef-831d-e3ca42aebd51.png
new file mode 100644
index 00000000..5c0b076b
Binary files /dev/null and b/pics/8433fbb2-c35c-45ef-831d-e3ca42aebd51.png differ
diff --git a/pics/85c05fb1-5546-4c50-9221-21f231cdc8c5.jpg b/pics/85c05fb1-5546-4c50-9221-21f231cdc8c5.jpg
new file mode 100644
index 00000000..4b911193
Binary files /dev/null and b/pics/85c05fb1-5546-4c50-9221-21f231cdc8c5.jpg differ
diff --git a/pics/8681db55-0873-434b-aa98-83d07e8392ae.jpg b/pics/8681db55-0873-434b-aa98-83d07e8392ae.jpg
deleted file mode 100644
index 3e6414b1..00000000
Binary files a/pics/8681db55-0873-434b-aa98-83d07e8392ae.jpg and /dev/null differ
diff --git a/pics/86b71296-0d1e-4a63-bcd9-54955b6b781b.jpg b/pics/86b71296-0d1e-4a63-bcd9-54955b6b781b.jpg
deleted file mode 100644
index c319156d..00000000
Binary files a/pics/86b71296-0d1e-4a63-bcd9-54955b6b781b.jpg and /dev/null differ
diff --git a/pics/8785dabd-1285-4bd0-b3aa-b05cc060a24a.jpg b/pics/8785dabd-1285-4bd0-b3aa-b05cc060a24a.jpg
deleted file mode 100644
index 7c7efcce..00000000
Binary files a/pics/8785dabd-1285-4bd0-b3aa-b05cc060a24a.jpg and /dev/null differ
diff --git a/pics/88ff46b3-028a-4dbb-a572-1f062b8b96d3.png b/pics/88ff46b3-028a-4dbb-a572-1f062b8b96d3.png
new file mode 100644
index 00000000..c6344ad2
Binary files /dev/null and b/pics/88ff46b3-028a-4dbb-a572-1f062b8b96d3.png differ
diff --git a/pics/89091427-7b2b-4923-aff6-44681319a8aa.jpg b/pics/89091427-7b2b-4923-aff6-44681319a8aa.jpg
deleted file mode 100644
index d2cc88f7..00000000
Binary files a/pics/89091427-7b2b-4923-aff6-44681319a8aa.jpg and /dev/null differ
diff --git a/pics/897a4f4e-2683-44e1-a26a-c0d0234dc576.jpg b/pics/897a4f4e-2683-44e1-a26a-c0d0234dc576.jpg
deleted file mode 100644
index c6f0a0b6..00000000
Binary files a/pics/897a4f4e-2683-44e1-a26a-c0d0234dc576.jpg and /dev/null differ
diff --git a/pics/8a4c6ad4-a816-47d1-b93f-7ca4f78ab67a.png b/pics/8a4c6ad4-a816-47d1-b93f-7ca4f78ab67a.png
new file mode 100644
index 00000000..360a984e
Binary files /dev/null and b/pics/8a4c6ad4-a816-47d1-b93f-7ca4f78ab67a.png differ
diff --git a/pics/8ab40d6d-bd7c-47d3-afe8-6a8bc9f5d04c.jpg b/pics/8ab40d6d-bd7c-47d3-afe8-6a8bc9f5d04c.jpg
deleted file mode 100644
index c4eb4d8d..00000000
Binary files a/pics/8ab40d6d-bd7c-47d3-afe8-6a8bc9f5d04c.jpg and /dev/null differ
diff --git a/pics/8adb2591-d3f1-4632-84cb-823fb9c5eb09.jpg b/pics/8adb2591-d3f1-4632-84cb-823fb9c5eb09.jpg
new file mode 100644
index 00000000..9083d22b
Binary files /dev/null and b/pics/8adb2591-d3f1-4632-84cb-823fb9c5eb09.jpg differ
diff --git a/pics/8ae4550b-f0cb-4e4d-9e2b-c550538bf230.png b/pics/8ae4550b-f0cb-4e4d-9e2b-c550538bf230.png
new file mode 100644
index 00000000..31ae5e0c
Binary files /dev/null and b/pics/8ae4550b-f0cb-4e4d-9e2b-c550538bf230.png differ
diff --git a/pics/8af348d0-4d72-4f76-b56c-0a440ed4673d.png b/pics/8af348d0-4d72-4f76-b56c-0a440ed4673d.png
new file mode 100644
index 00000000..9d575722
Binary files /dev/null and b/pics/8af348d0-4d72-4f76-b56c-0a440ed4673d.png differ
diff --git a/pics/8b15e36f-69b4-46b6-a07c-7234ac7c7927.jpg b/pics/8b15e36f-69b4-46b6-a07c-7234ac7c7927.jpg
deleted file mode 100644
index 58d6f04d..00000000
Binary files a/pics/8b15e36f-69b4-46b6-a07c-7234ac7c7927.jpg and /dev/null differ
diff --git a/pics/8b5bd2c8-8425-4a8b-89db-235c95800de9.jpg b/pics/8b5bd2c8-8425-4a8b-89db-235c95800de9.jpg
deleted file mode 100644
index 0d5e32aa..00000000
Binary files a/pics/8b5bd2c8-8425-4a8b-89db-235c95800de9.jpg and /dev/null differ
diff --git a/pics/8bc6fc2c-d198-4759-b06c-18d94d851e97.png b/pics/8bc6fc2c-d198-4759-b06c-18d94d851e97.png
deleted file mode 100644
index 92f32485..00000000
Binary files a/pics/8bc6fc2c-d198-4759-b06c-18d94d851e97.png and /dev/null differ
diff --git a/pics/8c139711-3500-4f71-8456-c1adaf429ad0.png b/pics/8c139711-3500-4f71-8456-c1adaf429ad0.png
deleted file mode 100644
index 4bd01d9c..00000000
Binary files a/pics/8c139711-3500-4f71-8456-c1adaf429ad0.png and /dev/null differ
diff --git a/pics/8cc671f0-7134-44b1-a7b5-6d24fe55e1c1.jpg b/pics/8cc671f0-7134-44b1-a7b5-6d24fe55e1c1.jpg
new file mode 100644
index 00000000..5e6608b2
Binary files /dev/null and b/pics/8cc671f0-7134-44b1-a7b5-6d24fe55e1c1.jpg differ
diff --git a/pics/8d211911-0e62-4190-ab00-d8610adec4a0.jpg b/pics/8d211911-0e62-4190-ab00-d8610adec4a0.jpg
deleted file mode 100644
index b6c65270..00000000
Binary files a/pics/8d211911-0e62-4190-ab00-d8610adec4a0.jpg and /dev/null differ
diff --git a/pics/8d6af5ac-74eb-4e07-99aa-654b9f21f1d3.jpg b/pics/8d6af5ac-74eb-4e07-99aa-654b9f21f1d3.jpg
deleted file mode 100644
index 65b9370e..00000000
Binary files a/pics/8d6af5ac-74eb-4e07-99aa-654b9f21f1d3.jpg and /dev/null differ
diff --git a/pics/8dfb4cc9-26da-45e7-b820-4576fa1cbb0e.png b/pics/8dfb4cc9-26da-45e7-b820-4576fa1cbb0e.png
new file mode 100644
index 00000000..7044dfbd
Binary files /dev/null and b/pics/8dfb4cc9-26da-45e7-b820-4576fa1cbb0e.png differ
diff --git a/pics/8e8ba824-7a9e-4934-a212-e6a41dcc1602.jpg b/pics/8e8ba824-7a9e-4934-a212-e6a41dcc1602.jpg
deleted file mode 100644
index b405a3b8..00000000
Binary files a/pics/8e8ba824-7a9e-4934-a212-e6a41dcc1602.jpg and /dev/null differ
diff --git a/pics/8ef22836-8800-4765-b4b8-ade80096b323.jpg b/pics/8ef22836-8800-4765-b4b8-ade80096b323.jpg
deleted file mode 100644
index b4d8268f..00000000
Binary files a/pics/8ef22836-8800-4765-b4b8-ade80096b323.jpg and /dev/null differ
diff --git a/pics/8f0cc500-5994-4c7a-91a9-62885d658662.png b/pics/8f0cc500-5994-4c7a-91a9-62885d658662.png
new file mode 100644
index 00000000..034205a8
Binary files /dev/null and b/pics/8f0cc500-5994-4c7a-91a9-62885d658662.png differ
diff --git a/pics/8f3b9519-d705-48fe-87ad-2e4052fc81d2.png b/pics/8f3b9519-d705-48fe-87ad-2e4052fc81d2.png
new file mode 100644
index 00000000..a8e36ee0
Binary files /dev/null and b/pics/8f3b9519-d705-48fe-87ad-2e4052fc81d2.png differ
diff --git a/pics/8f64e9c5-7682-4feb-9312-dea09514e160.jpg b/pics/8f64e9c5-7682-4feb-9312-dea09514e160.jpg
new file mode 100644
index 00000000..6f7e9e35
Binary files /dev/null and b/pics/8f64e9c5-7682-4feb-9312-dea09514e160.jpg differ
diff --git a/pics/8f6f9dc9-9ecd-47c8-b50e-2814f0219056.png b/pics/8f6f9dc9-9ecd-47c8-b50e-2814f0219056.png
new file mode 100644
index 00000000..21b8a930
Binary files /dev/null and b/pics/8f6f9dc9-9ecd-47c8-b50e-2814f0219056.png differ
diff --git a/pics/8fe838e3-ef77-4f63-bf45-417b6bc5c6bb.png b/pics/8fe838e3-ef77-4f63-bf45-417b6bc5c6bb.png
new file mode 100644
index 00000000..2e53fbe3
Binary files /dev/null and b/pics/8fe838e3-ef77-4f63-bf45-417b6bc5c6bb.png differ
diff --git a/pics/9.gif b/pics/9.gif
new file mode 100644
index 00000000..f81aa264
Binary files /dev/null and b/pics/9.gif differ
diff --git a/pics/903093ec-acc8-4f9b-bf2c-b990b9a5390c.jpg b/pics/903093ec-acc8-4f9b-bf2c-b990b9a5390c.jpg
deleted file mode 100644
index a031071b..00000000
Binary files a/pics/903093ec-acc8-4f9b-bf2c-b990b9a5390c.jpg and /dev/null differ
diff --git a/pics/910f613f-514f-4534-87dd-9b4699d59d31.png b/pics/910f613f-514f-4534-87dd-9b4699d59d31.png
new file mode 100644
index 00000000..28362d47
Binary files /dev/null and b/pics/910f613f-514f-4534-87dd-9b4699d59d31.png differ
diff --git a/pics/9110c1a0-8a54-4145-a814-2477d0128114.png b/pics/9110c1a0-8a54-4145-a814-2477d0128114.png
new file mode 100644
index 00000000..68aa9f98
Binary files /dev/null and b/pics/9110c1a0-8a54-4145-a814-2477d0128114.png differ
diff --git a/pics/912174d8-0786-4222-b7ef-a611d36e5db9.jpg b/pics/912174d8-0786-4222-b7ef-a611d36e5db9.jpg
deleted file mode 100644
index 5f9cb884..00000000
Binary files a/pics/912174d8-0786-4222-b7ef-a611d36e5db9.jpg and /dev/null differ
diff --git a/pics/9192dc0a-a7cd-4030-8df6-e388600644cf.jpg b/pics/9192dc0a-a7cd-4030-8df6-e388600644cf.jpg
deleted file mode 100644
index 1917c941..00000000
Binary files a/pics/9192dc0a-a7cd-4030-8df6-e388600644cf.jpg and /dev/null differ
diff --git a/pics/91ef04e4-923a-4277-99c0-6be4ce81e5ac.jpg b/pics/91ef04e4-923a-4277-99c0-6be4ce81e5ac.jpg
new file mode 100644
index 00000000..f4de7074
Binary files /dev/null and b/pics/91ef04e4-923a-4277-99c0-6be4ce81e5ac.jpg differ
diff --git a/pics/920c034c-c212-4f79-9ddb-84e4bb6cd088.png b/pics/920c034c-c212-4f79-9ddb-84e4bb6cd088.png
new file mode 100644
index 00000000..ac17fd0c
Binary files /dev/null and b/pics/920c034c-c212-4f79-9ddb-84e4bb6cd088.png differ
diff --git a/pics/923896c1-937e-4a38-b8a6-cec3040b4e2a.jpg b/pics/923896c1-937e-4a38-b8a6-cec3040b4e2a.jpg
deleted file mode 100644
index ee440e1d..00000000
Binary files a/pics/923896c1-937e-4a38-b8a6-cec3040b4e2a.jpg and /dev/null differ
diff --git a/pics/92ad9bae-7d02-43ba-8115-a9d6f530ca28.png b/pics/92ad9bae-7d02-43ba-8115-a9d6f530ca28.png
new file mode 100644
index 00000000..7b85c49a
Binary files /dev/null and b/pics/92ad9bae-7d02-43ba-8115-a9d6f530ca28.png differ
diff --git a/pics/93cbce0c-c37d-429c-815b-861976a46bd8.png b/pics/93cbce0c-c37d-429c-815b-861976a46bd8.png
deleted file mode 100644
index fca4da5d..00000000
Binary files a/pics/93cbce0c-c37d-429c-815b-861976a46bd8.png and /dev/null differ
diff --git a/pics/94589319-975f-490b-8bae-90b3a4953559.png b/pics/94589319-975f-490b-8bae-90b3a4953559.png
new file mode 100644
index 00000000..fab53e3d
Binary files /dev/null and b/pics/94589319-975f-490b-8bae-90b3a4953559.png differ
diff --git a/pics/952afa9a-458b-44ce-bba9-463e60162945.png b/pics/952afa9a-458b-44ce-bba9-463e60162945.png
new file mode 100644
index 00000000..db93c1ac
Binary files /dev/null and b/pics/952afa9a-458b-44ce-bba9-463e60162945.png differ
diff --git a/pics/955af054-8872-4569-82e7-2e10b66bc38e.png b/pics/955af054-8872-4569-82e7-2e10b66bc38e.png
new file mode 100644
index 00000000..502af347
Binary files /dev/null and b/pics/955af054-8872-4569-82e7-2e10b66bc38e.png differ
diff --git a/pics/95f4559c-3d2a-4176-b365-4fbc46c76cf1.png b/pics/95f4559c-3d2a-4176-b365-4fbc46c76cf1.png
deleted file mode 100644
index 0085f33e..00000000
Binary files a/pics/95f4559c-3d2a-4176-b365-4fbc46c76cf1.png and /dev/null differ
diff --git a/pics/96b63e13-e2d8-4ddb-9aa1-a38959ca96e5.jpg b/pics/96b63e13-e2d8-4ddb-9aa1-a38959ca96e5.jpg
new file mode 100644
index 00000000..3dbe14f1
Binary files /dev/null and b/pics/96b63e13-e2d8-4ddb-9aa1-a38959ca96e5.jpg differ
diff --git a/pics/992faced-afcf-414d-b801-9c16d6570fec.jpg b/pics/992faced-afcf-414d-b801-9c16d6570fec.jpg
new file mode 100644
index 00000000..1a363f1e
Binary files /dev/null and b/pics/992faced-afcf-414d-b801-9c16d6570fec.jpg differ
diff --git a/pics/9b5e0fa0-9274-4219-a3a9-84fbb509c735.jpg b/pics/9b5e0fa0-9274-4219-a3a9-84fbb509c735.jpg
deleted file mode 100644
index dd41f3c1..00000000
Binary files a/pics/9b5e0fa0-9274-4219-a3a9-84fbb509c735.jpg and /dev/null differ
diff --git a/pics/9b829410-86c4-40aa-ba8d-9e8e26c0eeb8.jpg b/pics/9b829410-86c4-40aa-ba8d-9e8e26c0eeb8.jpg
new file mode 100644
index 00000000..470580c3
Binary files /dev/null and b/pics/9b829410-86c4-40aa-ba8d-9e8e26c0eeb8.jpg differ
diff --git a/pics/9c997ac5-c8a7-44fe-bf45-2c10eb773e53.jpg b/pics/9c997ac5-c8a7-44fe-bf45-2c10eb773e53.jpg
deleted file mode 100644
index 0e3f2bb3..00000000
Binary files a/pics/9c997ac5-c8a7-44fe-bf45-2c10eb773e53.jpg and /dev/null differ
diff --git a/pics/9cd0ae20-4fb5-4017-a000-f7d3a0eb3529.png b/pics/9cd0ae20-4fb5-4017-a000-f7d3a0eb3529.png
new file mode 100644
index 00000000..49da824e
Binary files /dev/null and b/pics/9cd0ae20-4fb5-4017-a000-f7d3a0eb3529.png differ
diff --git a/pics/9d0a637c-6a8f-4f5a-99b9-fdcfa26793ff.png b/pics/9d0a637c-6a8f-4f5a-99b9-fdcfa26793ff.png
new file mode 100644
index 00000000..34a8f664
Binary files /dev/null and b/pics/9d0a637c-6a8f-4f5a-99b9-fdcfa26793ff.png differ
diff --git a/pics/9d2226dc-c4a3-40ec-9b3e-a46bf86af499.png b/pics/9d2226dc-c4a3-40ec-9b3e-a46bf86af499.png
deleted file mode 100644
index cbef4f66..00000000
Binary files a/pics/9d2226dc-c4a3-40ec-9b3e-a46bf86af499.png and /dev/null differ
diff --git a/pics/9dbb5fc2-936b-4c6d-b3a7-9617aae45080.jpg b/pics/9dbb5fc2-936b-4c6d-b3a7-9617aae45080.jpg
deleted file mode 100644
index 05f71904..00000000
Binary files a/pics/9dbb5fc2-936b-4c6d-b3a7-9617aae45080.jpg and /dev/null differ
diff --git a/pics/9e5e3cc6-3107-4051-b584-8ff077638fe6.png b/pics/9e5e3cc6-3107-4051-b584-8ff077638fe6.png
deleted file mode 100644
index a3e8506b..00000000
Binary files a/pics/9e5e3cc6-3107-4051-b584-8ff077638fe6.png and /dev/null differ
diff --git a/pics/9ecaebee-670e-4cb2-9cdb-3029c00f33bd.png b/pics/9ecaebee-670e-4cb2-9cdb-3029c00f33bd.png
deleted file mode 100644
index 6db44a1e..00000000
Binary files a/pics/9ecaebee-670e-4cb2-9cdb-3029c00f33bd.png and /dev/null differ
diff --git a/pics/9ee83c8c-1165-476c-85a6-e6e434e5307a.jpg b/pics/9ee83c8c-1165-476c-85a6-e6e434e5307a.jpg
deleted file mode 100644
index 644fd766..00000000
Binary files a/pics/9ee83c8c-1165-476c-85a6-e6e434e5307a.jpg and /dev/null differ
diff --git a/pics/ClienteServidorSockets1521731145260.jpg b/pics/ClienteServidorSockets1521731145260.jpg
new file mode 100644
index 00000000..18976fb3
Binary files /dev/null and b/pics/ClienteServidorSockets1521731145260.jpg differ
diff --git a/pics/CountdownLatch.png b/pics/CountdownLatch.png
new file mode 100644
index 00000000..1a581bbf
Binary files /dev/null and b/pics/CountdownLatch.png differ
diff --git a/pics/CyclicBarrier.png b/pics/CyclicBarrier.png
new file mode 100644
index 00000000..5fe5b88b
Binary files /dev/null and b/pics/CyclicBarrier.png differ
diff --git a/pics/DP-Decorator-java.io.png b/pics/DP-Decorator-java.io.png
new file mode 100644
index 00000000..897a510b
Binary files /dev/null and b/pics/DP-Decorator-java.io.png differ
diff --git a/pics/GUID_Partition_Table_Scheme.svg.png b/pics/GUID_Partition_Table_Scheme.svg.png
new file mode 100644
index 00000000..6638ae7a
Binary files /dev/null and b/pics/GUID_Partition_Table_Scheme.svg.png differ
diff --git a/pics/HTTP_RequestMessageExample.png b/pics/HTTP_RequestMessageExample.png
new file mode 100644
index 00000000..8fd213cb
Binary files /dev/null and b/pics/HTTP_RequestMessageExample.png differ
diff --git a/pics/HTTP_ResponseMessageExample.png b/pics/HTTP_ResponseMessageExample.png
new file mode 100644
index 00000000..1adf26c5
Binary files /dev/null and b/pics/HTTP_ResponseMessageExample.png differ
diff --git a/pics/How-HTTPS-Works.png b/pics/How-HTTPS-Works.png
new file mode 100644
index 00000000..c10605f7
Binary files /dev/null and b/pics/How-HTTPS-Works.png differ
diff --git a/pics/Iterator-1.jpg b/pics/Iterator-1.jpg
new file mode 100644
index 00000000..f9f818c6
Binary files /dev/null and b/pics/Iterator-1.jpg differ
diff --git a/pics/JNIFigure1.gif b/pics/JNIFigure1.gif
new file mode 100644
index 00000000..f47f289e
Binary files /dev/null and b/pics/JNIFigure1.gif differ
diff --git a/pics/JVM-Stack.png b/pics/JVM-Stack.png
new file mode 100644
index 00000000..e55ccf9b
Binary files /dev/null and b/pics/JVM-Stack.png differ
diff --git a/pics/JVM-runtime-data-area.jpg b/pics/JVM-runtime-data-area.jpg
new file mode 100644
index 00000000..88f8691f
Binary files /dev/null and b/pics/JVM-runtime-data-area.jpg differ
diff --git a/pics/MultiNode-SessionReplication.jpg b/pics/MultiNode-SessionReplication.jpg
new file mode 100644
index 00000000..0223bd80
Binary files /dev/null and b/pics/MultiNode-SessionReplication.jpg differ
diff --git a/pics/MultiNode-SpringSession.jpg b/pics/MultiNode-SpringSession.jpg
new file mode 100644
index 00000000..38d56e2c
Binary files /dev/null and b/pics/MultiNode-SpringSession.jpg differ
diff --git a/pics/MultiNode-StickySessions.jpg b/pics/MultiNode-StickySessions.jpg
new file mode 100644
index 00000000..a7e1c6aa
Binary files /dev/null and b/pics/MultiNode-StickySessions.jpg differ
diff --git a/pics/PPjwP.png b/pics/PPjwP.png
new file mode 100644
index 00000000..80631505
Binary files /dev/null and b/pics/PPjwP.png differ
diff --git a/pics/ProcessState.png b/pics/ProcessState.png
new file mode 100644
index 00000000..39269579
Binary files /dev/null and b/pics/ProcessState.png differ
diff --git a/pics/Semaphore.png b/pics/Semaphore.png
new file mode 100644
index 00000000..6b8c30b1
Binary files /dev/null and b/pics/Semaphore.png differ
diff --git a/pics/TIM截图20180227172950.png b/pics/TIM截图20180227172950.png
deleted file mode 100644
index 1cb6971e..00000000
Binary files a/pics/TIM截图20180227172950.png and /dev/null differ
diff --git a/pics/Technology-ComputerNetworking-Internet-ISPs.png b/pics/Technology-ComputerNetworking-Internet-ISPs.png
new file mode 100644
index 00000000..72a30a7b
Binary files /dev/null and b/pics/Technology-ComputerNetworking-Internet-ISPs.png differ
diff --git a/pics/ZnJvbT1jc2RuJnVybD15VkdkdVYyUXZ.jpg b/pics/ZnJvbT1jc2RuJnVybD15VkdkdVYyUXZ.jpg
new file mode 100644
index 00000000..3027986e
Binary files /dev/null and b/pics/ZnJvbT1jc2RuJnVybD15VkdkdVYyUXZ.jpg differ
diff --git a/pics/a01d1516-8168-461a-a24b-620b9cfc40f4.png b/pics/a01d1516-8168-461a-a24b-620b9cfc40f4.png
new file mode 100644
index 00000000..4b9b4b48
Binary files /dev/null and b/pics/a01d1516-8168-461a-a24b-620b9cfc40f4.png differ
diff --git a/pics/a1198642-9159-4d88-8aec-c3b04e7a2563.jpg b/pics/a1198642-9159-4d88-8aec-c3b04e7a2563.jpg
new file mode 100644
index 00000000..30e92639
Binary files /dev/null and b/pics/a1198642-9159-4d88-8aec-c3b04e7a2563.jpg differ
diff --git a/pics/a13b62da-0fa8-4224-a615-4cadacc08871.png b/pics/a13b62da-0fa8-4224-a615-4cadacc08871.png
deleted file mode 100644
index ff4ee89e..00000000
Binary files a/pics/a13b62da-0fa8-4224-a615-4cadacc08871.png and /dev/null differ
diff --git a/pics/a2670745-a7b1-497b-90a4-dbddc4e2006d.jpg b/pics/a2670745-a7b1-497b-90a4-dbddc4e2006d.jpg
deleted file mode 100644
index 66026828..00000000
Binary files a/pics/a2670745-a7b1-497b-90a4-dbddc4e2006d.jpg and /dev/null differ
diff --git a/pics/a2d13178-f1ef-4811-a240-1fe95b55b1eb.png b/pics/a2d13178-f1ef-4811-a240-1fe95b55b1eb.png
new file mode 100644
index 00000000..edc6840a
Binary files /dev/null and b/pics/a2d13178-f1ef-4811-a240-1fe95b55b1eb.png differ
diff --git a/pics/a314bb79-5b18-4e63-a976-3448bffa6f1b.png b/pics/a314bb79-5b18-4e63-a976-3448bffa6f1b.png
new file mode 100644
index 00000000..1a5a6474
Binary files /dev/null and b/pics/a314bb79-5b18-4e63-a976-3448bffa6f1b.png differ
diff --git a/pics/a3253deb-8d21-40a1-aae4-7d178e4aa319.jpg b/pics/a3253deb-8d21-40a1-aae4-7d178e4aa319.jpg
new file mode 100644
index 00000000..23258afa
Binary files /dev/null and b/pics/a3253deb-8d21-40a1-aae4-7d178e4aa319.jpg differ
diff --git a/pics/a4c17d43-fa5e-4935-b74e-147e7f7e782c.png b/pics/a4c17d43-fa5e-4935-b74e-147e7f7e782c.png
new file mode 100644
index 00000000..461fffb0
Binary files /dev/null and b/pics/a4c17d43-fa5e-4935-b74e-147e7f7e782c.png differ
diff --git a/pics/a57a6fc8-c5e9-456c-80ff-a5139dda4b6e.png b/pics/a57a6fc8-c5e9-456c-80ff-a5139dda4b6e.png
new file mode 100644
index 00000000..384e7aa6
Binary files /dev/null and b/pics/a57a6fc8-c5e9-456c-80ff-a5139dda4b6e.png differ
diff --git a/pics/a5c25452-6fa5-49e7-9322-823077442775.jpg b/pics/a5c25452-6fa5-49e7-9322-823077442775.jpg
deleted file mode 100644
index 49219388..00000000
Binary files a/pics/a5c25452-6fa5-49e7-9322-823077442775.jpg and /dev/null differ
diff --git a/pics/a5fa89e7-54b9-4e2f-8c48-a35712d7b2f5.jpg b/pics/a5fa89e7-54b9-4e2f-8c48-a35712d7b2f5.jpg
deleted file mode 100644
index 4af8eb07..00000000
Binary files a/pics/a5fa89e7-54b9-4e2f-8c48-a35712d7b2f5.jpg and /dev/null differ
diff --git a/pics/a6026bb4-3daf-439f-b1ec-a5a24e19d2fb.jpg b/pics/a6026bb4-3daf-439f-b1ec-a5a24e19d2fb.jpg
new file mode 100644
index 00000000..4ff577e2
Binary files /dev/null and b/pics/a6026bb4-3daf-439f-b1ec-a5a24e19d2fb.jpg differ
diff --git a/pics/a69af9bb-b5ad-4896-862d-697e5ee4feb1.png b/pics/a69af9bb-b5ad-4896-862d-697e5ee4feb1.png
deleted file mode 100644
index e07677bc..00000000
Binary files a/pics/a69af9bb-b5ad-4896-862d-697e5ee4feb1.png and /dev/null differ
diff --git a/pics/a6ac2b08-3861-4e85-baa8-382287bfee9f.png b/pics/a6ac2b08-3861-4e85-baa8-382287bfee9f.png
new file mode 100644
index 00000000..26b0bd03
Binary files /dev/null and b/pics/a6ac2b08-3861-4e85-baa8-382287bfee9f.png differ
diff --git a/pics/a758c8b2-0ac7-438f-90c2-3923ffad6328.png b/pics/a758c8b2-0ac7-438f-90c2-3923ffad6328.png
deleted file mode 100644
index 355d5533..00000000
Binary files a/pics/a758c8b2-0ac7-438f-90c2-3923ffad6328.png and /dev/null differ
diff --git a/pics/a797959a-0ed5-475b-8d97-df157c672019.jpg b/pics/a797959a-0ed5-475b-8d97-df157c672019.jpg
deleted file mode 100644
index 18af09d8..00000000
Binary files a/pics/a797959a-0ed5-475b-8d97-df157c672019.jpg and /dev/null differ
diff --git a/pics/a9b6c1db-0f4a-4e91-8ac8-6b19bd106b51.png b/pics/a9b6c1db-0f4a-4e91-8ac8-6b19bd106b51.png
deleted file mode 100644
index d2f3fc46..00000000
Binary files a/pics/a9b6c1db-0f4a-4e91-8ac8-6b19bd106b51.png and /dev/null differ
diff --git a/pics/a9b91b7d-65d7-4aa3-8ef6-21876b05ad16.png b/pics/a9b91b7d-65d7-4aa3-8ef6-21876b05ad16.png
deleted file mode 100644
index 8a60458f..00000000
Binary files a/pics/a9b91b7d-65d7-4aa3-8ef6-21876b05ad16.png and /dev/null differ
diff --git a/pics/aa20c123-b6b5-432a-83d3-45dc39172192.jpg b/pics/aa20c123-b6b5-432a-83d3-45dc39172192.jpg
deleted file mode 100644
index 3c2c7ffe..00000000
Binary files a/pics/aa20c123-b6b5-432a-83d3-45dc39172192.jpg and /dev/null differ
diff --git a/pics/aa29cc88-7256-4399-8c7f-3cf4a6489559.png b/pics/aa29cc88-7256-4399-8c7f-3cf4a6489559.png
new file mode 100644
index 00000000..9b93237e
Binary files /dev/null and b/pics/aa29cc88-7256-4399-8c7f-3cf4a6489559.png differ
diff --git a/pics/aa42f9c6-ad7a-48f4-8e8b-f3b6de3feaec.png b/pics/aa42f9c6-ad7a-48f4-8e8b-f3b6de3feaec.png
new file mode 100644
index 00000000..cdb78c11
Binary files /dev/null and b/pics/aa42f9c6-ad7a-48f4-8e8b-f3b6de3feaec.png differ
diff --git a/pics/ab77240d-7338-4547-9183-00215e7220ec.png b/pics/ab77240d-7338-4547-9183-00215e7220ec.png
new file mode 100644
index 00000000..d57d91ce
Binary files /dev/null and b/pics/ab77240d-7338-4547-9183-00215e7220ec.png differ
diff --git a/pics/ace830df-9919-48ca-91b5-60b193f593d2.png b/pics/ace830df-9919-48ca-91b5-60b193f593d2.png
new file mode 100644
index 00000000..79efa287
Binary files /dev/null and b/pics/ace830df-9919-48ca-91b5-60b193f593d2.png differ
diff --git a/pics/ae3fc93a-44d5-4beb-b05a-874bd9c0a657.png b/pics/ae3fc93a-44d5-4beb-b05a-874bd9c0a657.png
new file mode 100644
index 00000000..ea23c7db
Binary files /dev/null and b/pics/ae3fc93a-44d5-4beb-b05a-874bd9c0a657.png differ
diff --git a/pics/af4639f9-af54-4400-aaf5-4e261d96ace7.png b/pics/af4639f9-af54-4400-aaf5-4e261d96ace7.png
new file mode 100644
index 00000000..8de4fa4d
Binary files /dev/null and b/pics/af4639f9-af54-4400-aaf5-4e261d96ace7.png differ
diff --git a/pics/b001fa64-307c-49af-b4b2-2043fc26154e.png b/pics/b001fa64-307c-49af-b4b2-2043fc26154e.png
deleted file mode 100644
index 8575c0d2..00000000
Binary files a/pics/b001fa64-307c-49af-b4b2-2043fc26154e.png and /dev/null differ
diff --git a/pics/b0748916-1acd-4138-b24c-69326cb452fe.jpg b/pics/b0748916-1acd-4138-b24c-69326cb452fe.jpg
deleted file mode 100644
index 350f3aa3..00000000
Binary files a/pics/b0748916-1acd-4138-b24c-69326cb452fe.jpg and /dev/null differ
diff --git a/pics/b0e8ef47-2f23-4379-8c64-10d5cb44d438.jpg b/pics/b0e8ef47-2f23-4379-8c64-10d5cb44d438.jpg
new file mode 100644
index 00000000..cb17fdcb
Binary files /dev/null and b/pics/b0e8ef47-2f23-4379-8c64-10d5cb44d438.jpg differ
diff --git a/pics/b18d679b-c8e2-4564-88ee-7600090e46da.jpg b/pics/b18d679b-c8e2-4564-88ee-7600090e46da.jpg
deleted file mode 100644
index d8194ab1..00000000
Binary files a/pics/b18d679b-c8e2-4564-88ee-7600090e46da.jpg and /dev/null differ
diff --git a/pics/b1b4cf7d-c54a-4ff1-9741-cd2eea331123.jpg b/pics/b1b4cf7d-c54a-4ff1-9741-cd2eea331123.jpg
deleted file mode 100644
index fca884c9..00000000
Binary files a/pics/b1b4cf7d-c54a-4ff1-9741-cd2eea331123.jpg and /dev/null differ
diff --git a/pics/b1fa0453-a4b0-4eae-a352-48acca8fff74.png b/pics/b1fa0453-a4b0-4eae-a352-48acca8fff74.png
new file mode 100644
index 00000000..842b2f6c
Binary files /dev/null and b/pics/b1fa0453-a4b0-4eae-a352-48acca8fff74.png differ
diff --git a/pics/b29f8971-9cb8-480d-b986-0e60c2ece069.png b/pics/b29f8971-9cb8-480d-b986-0e60c2ece069.png
new file mode 100644
index 00000000..65f73b0e
Binary files /dev/null and b/pics/b29f8971-9cb8-480d-b986-0e60c2ece069.png differ
diff --git a/pics/b2b6253c-c701-4b30-aff4-bc3c713542a7.jpg b/pics/b2b6253c-c701-4b30-aff4-bc3c713542a7.jpg
deleted file mode 100644
index e4036e10..00000000
Binary files a/pics/b2b6253c-c701-4b30-aff4-bc3c713542a7.jpg and /dev/null differ
diff --git a/pics/b396d726-b75f-4a32-89a2-03a7b6e19f6f.jpg b/pics/b396d726-b75f-4a32-89a2-03a7b6e19f6f.jpg
new file mode 100644
index 00000000..a3e1a656
Binary files /dev/null and b/pics/b396d726-b75f-4a32-89a2-03a7b6e19f6f.jpg differ
diff --git a/pics/b418ca51-f005-4510-b7ad-f092eb6aeb24.png b/pics/b418ca51-f005-4510-b7ad-f092eb6aeb24.png
new file mode 100644
index 00000000..07157513
Binary files /dev/null and b/pics/b418ca51-f005-4510-b7ad-f092eb6aeb24.png differ
diff --git a/pics/b4252c85-6fb0-4995-9a68-a1a5925fbdb1.png b/pics/b4252c85-6fb0-4995-9a68-a1a5925fbdb1.png
new file mode 100644
index 00000000..59ea02cc
Binary files /dev/null and b/pics/b4252c85-6fb0-4995-9a68-a1a5925fbdb1.png differ
diff --git a/pics/b488282d-bfe0-464f-9e91-1f5b83a975bd.jpg b/pics/b488282d-bfe0-464f-9e91-1f5b83a975bd.jpg
deleted file mode 100644
index c1427eb3..00000000
Binary files a/pics/b488282d-bfe0-464f-9e91-1f5b83a975bd.jpg and /dev/null differ
diff --git a/pics/b4b29aa9-dd2c-467b-b75f-ca6541cb25b5.jpg b/pics/b4b29aa9-dd2c-467b-b75f-ca6541cb25b5.jpg
new file mode 100644
index 00000000..15f6fce8
Binary files /dev/null and b/pics/b4b29aa9-dd2c-467b-b75f-ca6541cb25b5.jpg differ
diff --git a/pics/b56ef52e-3d0f-4cdd-97dc-eaed893444a5.jpg b/pics/b56ef52e-3d0f-4cdd-97dc-eaed893444a5.jpg
deleted file mode 100644
index 1acf8feb..00000000
Binary files a/pics/b56ef52e-3d0f-4cdd-97dc-eaed893444a5.jpg and /dev/null differ
diff --git a/pics/b5c78914-066f-42be-ad1a-1c9f72aa9093.png b/pics/b5c78914-066f-42be-ad1a-1c9f72aa9093.png
deleted file mode 100644
index b8950c27..00000000
Binary files a/pics/b5c78914-066f-42be-ad1a-1c9f72aa9093.png and /dev/null differ
diff --git a/pics/b69d7184-ab62-4957-ba29-fb4fa25f9b65.jpg b/pics/b69d7184-ab62-4957-ba29-fb4fa25f9b65.jpg
deleted file mode 100644
index cae4d127..00000000
Binary files a/pics/b69d7184-ab62-4957-ba29-fb4fa25f9b65.jpg and /dev/null differ
diff --git a/pics/b6a678c0-c875-4038-afba-301846620786.jpg b/pics/b6a678c0-c875-4038-afba-301846620786.jpg
deleted file mode 100644
index c3194070..00000000
Binary files a/pics/b6a678c0-c875-4038-afba-301846620786.jpg and /dev/null differ
diff --git a/pics/b7b0eac6-e7ea-4fb6-8bfb-95fec6f235e2.png b/pics/b7b0eac6-e7ea-4fb6-8bfb-95fec6f235e2.png
deleted file mode 100644
index 746f9d84..00000000
Binary files a/pics/b7b0eac6-e7ea-4fb6-8bfb-95fec6f235e2.png and /dev/null differ
diff --git a/pics/b84ba6fb-312b-4e69-8c77-fb6eb6fb38d4.png b/pics/b84ba6fb-312b-4e69-8c77-fb6eb6fb38d4.png
new file mode 100644
index 00000000..dc44da3d
Binary files /dev/null and b/pics/b84ba6fb-312b-4e69-8c77-fb6eb6fb38d4.png differ
diff --git a/pics/b97958dd-3e43-45f7-97f5-3ec20f3f8b88.jpg b/pics/b97958dd-3e43-45f7-97f5-3ec20f3f8b88.jpg
deleted file mode 100644
index 40f51aa8..00000000
Binary files a/pics/b97958dd-3e43-45f7-97f5-3ec20f3f8b88.jpg and /dev/null differ
diff --git a/pics/b9a39d2a-618c-468b-86db-2e851f1a0057.jpg b/pics/b9a39d2a-618c-468b-86db-2e851f1a0057.jpg
new file mode 100644
index 00000000..05c7e6ce
Binary files /dev/null and b/pics/b9a39d2a-618c-468b-86db-2e851f1a0057.jpg differ
diff --git a/pics/b9d79a5a-e7af-499b-b989-f10483e71b8b.jpg b/pics/b9d79a5a-e7af-499b-b989-f10483e71b8b.jpg
new file mode 100644
index 00000000..b7502831
Binary files /dev/null and b/pics/b9d79a5a-e7af-499b-b989-f10483e71b8b.jpg differ
diff --git a/pics/b9e9ae8c-e216-4c01-b267-a50dbeb98fa4.jpg b/pics/b9e9ae8c-e216-4c01-b267-a50dbeb98fa4.jpg
deleted file mode 100644
index 29940be2..00000000
Binary files a/pics/b9e9ae8c-e216-4c01-b267-a50dbeb98fa4.jpg and /dev/null differ
diff --git a/pics/ba6ae411-82da-4d86-a434-6776d1731e8e.jpg b/pics/ba6ae411-82da-4d86-a434-6776d1731e8e.jpg
deleted file mode 100644
index c1013f01..00000000
Binary files a/pics/ba6ae411-82da-4d86-a434-6776d1731e8e.jpg and /dev/null differ
diff --git a/pics/bc775758-89ab-4805-9f9c-78b8739cf780.jpg b/pics/bc775758-89ab-4805-9f9c-78b8739cf780.jpg
new file mode 100644
index 00000000..4cd6f60f
Binary files /dev/null and b/pics/bc775758-89ab-4805-9f9c-78b8739cf780.jpg differ
diff --git a/pics/bccb799f-56e2-4356-95f0-a9ea05b0de2a.jpg b/pics/bccb799f-56e2-4356-95f0-a9ea05b0de2a.jpg
deleted file mode 100644
index 30b2ea88..00000000
Binary files a/pics/bccb799f-56e2-4356-95f0-a9ea05b0de2a.jpg and /dev/null differ
diff --git a/pics/bd6c05f3-02ee-4c8a-b374-40c87154a898.jpg b/pics/bd6c05f3-02ee-4c8a-b374-40c87154a898.jpg
deleted file mode 100644
index 94277944..00000000
Binary files a/pics/bd6c05f3-02ee-4c8a-b374-40c87154a898.jpg and /dev/null differ
diff --git a/pics/be53c00b-2534-4dc6-ad03-c55995c47db9.jpg b/pics/be53c00b-2534-4dc6-ad03-c55995c47db9.jpg
deleted file mode 100644
index 515ffd64..00000000
Binary files a/pics/be53c00b-2534-4dc6-ad03-c55995c47db9.jpg and /dev/null differ
diff --git a/pics/be7dca03-12ec-456b-8b54-b1b3161f5531.png b/pics/be7dca03-12ec-456b-8b54-b1b3161f5531.png
new file mode 100644
index 00000000..2c2214de
Binary files /dev/null and b/pics/be7dca03-12ec-456b-8b54-b1b3161f5531.png differ
diff --git a/pics/bee1ff1d-c80f-4b3c-b58c-7073a8896ab2.jpg b/pics/bee1ff1d-c80f-4b3c-b58c-7073a8896ab2.jpg
new file mode 100644
index 00000000..e4becc19
Binary files /dev/null and b/pics/bee1ff1d-c80f-4b3c-b58c-7073a8896ab2.jpg differ
diff --git a/pics/bf0ff9fc-467e-4a3f-8922-115ba2c55bde.png b/pics/bf0ff9fc-467e-4a3f-8922-115ba2c55bde.png
new file mode 100644
index 00000000..69222ad8
Binary files /dev/null and b/pics/bf0ff9fc-467e-4a3f-8922-115ba2c55bde.png differ
diff --git a/pics/bf4ed077-d481-4db7-9e7a-85d841a5a8c3.jpg b/pics/bf4ed077-d481-4db7-9e7a-85d841a5a8c3.jpg
deleted file mode 100644
index a20751ad..00000000
Binary files a/pics/bf4ed077-d481-4db7-9e7a-85d841a5a8c3.jpg and /dev/null differ
diff --git a/pics/bfbb11e2-d208-4efa-b97b-24cd40467cd8.png b/pics/bfbb11e2-d208-4efa-b97b-24cd40467cd8.png
new file mode 100644
index 00000000..b9d9dba2
Binary files /dev/null and b/pics/bfbb11e2-d208-4efa-b97b-24cd40467cd8.png differ
diff --git a/pics/c02a83b8-a6b9-4d00-a509-6f0516beaf5e.png b/pics/c02a83b8-a6b9-4d00-a509-6f0516beaf5e.png
deleted file mode 100644
index 52f44ea4..00000000
Binary files a/pics/c02a83b8-a6b9-4d00-a509-6f0516beaf5e.png and /dev/null differ
diff --git a/pics/c07035c3-a9ba-4508-8e3c-d8ae4c6ee9ee.jpg b/pics/c07035c3-a9ba-4508-8e3c-d8ae4c6ee9ee.jpg
deleted file mode 100644
index acb38080..00000000
Binary files a/pics/c07035c3-a9ba-4508-8e3c-d8ae4c6ee9ee.jpg and /dev/null differ
diff --git a/pics/c23957e9-a572-44f8-be15-f306c8b92722.jpg b/pics/c23957e9-a572-44f8-be15-f306c8b92722.jpg
new file mode 100644
index 00000000..2732c6d7
Binary files /dev/null and b/pics/c23957e9-a572-44f8-be15-f306c8b92722.jpg differ
diff --git a/pics/c24ad1af-d81b-409f-b4eb-62db9002d525.png b/pics/c24ad1af-d81b-409f-b4eb-62db9002d525.png
deleted file mode 100644
index ae164aad..00000000
Binary files a/pics/c24ad1af-d81b-409f-b4eb-62db9002d525.png and /dev/null differ
diff --git a/pics/c28fd93a-0d55-4a19-810f-72652feee00d.jpg b/pics/c28fd93a-0d55-4a19-810f-72652feee00d.jpg
deleted file mode 100644
index 7f198507..00000000
Binary files a/pics/c28fd93a-0d55-4a19-810f-72652feee00d.jpg and /dev/null differ
diff --git a/pics/c2d343f7-604c-4856-9a3c-c71d6f67fecc.png b/pics/c2d343f7-604c-4856-9a3c-c71d6f67fecc.png
new file mode 100644
index 00000000..84b2898b
Binary files /dev/null and b/pics/c2d343f7-604c-4856-9a3c-c71d6f67fecc.png differ
diff --git a/pics/c3369072-c740-43b0-b276-202bd1d3960d.jpg b/pics/c3369072-c740-43b0-b276-202bd1d3960d.jpg
new file mode 100644
index 00000000..17a2e9bf
Binary files /dev/null and b/pics/c3369072-c740-43b0-b276-202bd1d3960d.jpg differ
diff --git a/pics/c3ca36b2-8459-4cf1-98b0-cc95a0e94f20.jpg b/pics/c3ca36b2-8459-4cf1-98b0-cc95a0e94f20.jpg
deleted file mode 100644
index 94d726a1..00000000
Binary files a/pics/c3ca36b2-8459-4cf1-98b0-cc95a0e94f20.jpg and /dev/null differ
diff --git a/pics/c41d3977-e0e7-4ee4-93e1-d84f1ae3e20e.jpg b/pics/c41d3977-e0e7-4ee4-93e1-d84f1ae3e20e.jpg
deleted file mode 100644
index bf70b25a..00000000
Binary files a/pics/c41d3977-e0e7-4ee4-93e1-d84f1ae3e20e.jpg and /dev/null differ
diff --git a/pics/c470eb9b-fb05-45c5-8bb7-1057dc3c16de.jpg b/pics/c470eb9b-fb05-45c5-8bb7-1057dc3c16de.jpg
deleted file mode 100644
index ed426067..00000000
Binary files a/pics/c470eb9b-fb05-45c5-8bb7-1057dc3c16de.jpg and /dev/null differ
diff --git a/pics/c5022dd3-be22-4250-b9f6-38ae984a04d7.jpg b/pics/c5022dd3-be22-4250-b9f6-38ae984a04d7.jpg
new file mode 100644
index 00000000..3eb79406
Binary files /dev/null and b/pics/c5022dd3-be22-4250-b9f6-38ae984a04d7.jpg differ
diff --git a/pics/c50d230c-8b89-4644-8f62-8708d03aac5b.jpg b/pics/c50d230c-8b89-4644-8f62-8708d03aac5b.jpg
deleted file mode 100644
index c1afec59..00000000
Binary files a/pics/c50d230c-8b89-4644-8f62-8708d03aac5b.jpg and /dev/null differ
diff --git a/pics/c634b5ed-a14b-4302-b40e-3ee387dd3c8a.jpg b/pics/c634b5ed-a14b-4302-b40e-3ee387dd3c8a.jpg
deleted file mode 100644
index f3fffbc3..00000000
Binary files a/pics/c634b5ed-a14b-4302-b40e-3ee387dd3c8a.jpg and /dev/null differ
diff --git a/pics/c73a0b78-5f46-4d2d-a009-dab2a999b5d8.jpg b/pics/c73a0b78-5f46-4d2d-a009-dab2a999b5d8.jpg
deleted file mode 100644
index adcb251b..00000000
Binary files a/pics/c73a0b78-5f46-4d2d-a009-dab2a999b5d8.jpg and /dev/null differ
diff --git a/pics/c73aa08e-a987-43c9-92be-adea4a884c25.png b/pics/c73aa08e-a987-43c9-92be-adea4a884c25.png
deleted file mode 100644
index 33c22c9d..00000000
Binary files a/pics/c73aa08e-a987-43c9-92be-adea4a884c25.png and /dev/null differ
diff --git a/pics/c7665f73-c52f-4ce4-aed3-592bbd76265b.png b/pics/c7665f73-c52f-4ce4-aed3-592bbd76265b.png
deleted file mode 100644
index 3994ee98..00000000
Binary files a/pics/c7665f73-c52f-4ce4-aed3-592bbd76265b.png and /dev/null differ
diff --git a/pics/c77b6a18-dfac-42a2-ac89-7e99481275dc.jpg b/pics/c77b6a18-dfac-42a2-ac89-7e99481275dc.jpg
deleted file mode 100644
index 71587795..00000000
Binary files a/pics/c77b6a18-dfac-42a2-ac89-7e99481275dc.jpg and /dev/null differ
diff --git a/pics/c7b9b4c8-83d1-4eb0-8408-ea6576a9ed90.png b/pics/c7b9b4c8-83d1-4eb0-8408-ea6576a9ed90.png
deleted file mode 100644
index 283dcd28..00000000
Binary files a/pics/c7b9b4c8-83d1-4eb0-8408-ea6576a9ed90.png and /dev/null differ
diff --git a/pics/c7d4956c-9988-4a10-a704-28fdae7f3d28.png b/pics/c7d4956c-9988-4a10-a704-28fdae7f3d28.png
new file mode 100644
index 00000000..56510359
Binary files /dev/null and b/pics/c7d4956c-9988-4a10-a704-28fdae7f3d28.png differ
diff --git a/pics/c812c28a-1513-4a82-bfda-ab6a40981aa0.png b/pics/c812c28a-1513-4a82-bfda-ab6a40981aa0.png
new file mode 100644
index 00000000..b1142bff
Binary files /dev/null and b/pics/c812c28a-1513-4a82-bfda-ab6a40981aa0.png differ
diff --git a/pics/c8637fd2-3aaa-46c4-b7d9-f24d3fa04781.jpg b/pics/c8637fd2-3aaa-46c4-b7d9-f24d3fa04781.jpg
deleted file mode 100644
index 68fed2a3..00000000
Binary files a/pics/c8637fd2-3aaa-46c4-b7d9-f24d3fa04781.jpg and /dev/null differ
diff --git a/pics/c8d18ca9-0b09-441a-9a0c-fb063630d708.png b/pics/c8d18ca9-0b09-441a-9a0c-fb063630d708.png
new file mode 100644
index 00000000..514ab054
Binary files /dev/null and b/pics/c8d18ca9-0b09-441a-9a0c-fb063630d708.png differ
diff --git a/pics/c9a1de44-b1c0-4d13-a654-827d4ef8a723.png b/pics/c9a1de44-b1c0-4d13-a654-827d4ef8a723.png
new file mode 100644
index 00000000..f9ad721b
Binary files /dev/null and b/pics/c9a1de44-b1c0-4d13-a654-827d4ef8a723.png differ
diff --git a/pics/c9cfcd20-c901-435f-9a07-3e46830c359f.jpg b/pics/c9cfcd20-c901-435f-9a07-3e46830c359f.jpg
new file mode 100644
index 00000000..813b5142
Binary files /dev/null and b/pics/c9cfcd20-c901-435f-9a07-3e46830c359f.jpg differ
diff --git a/pics/ca3a793e-06e5-4ff3-b28e-a9c20540d164.png b/pics/ca3a793e-06e5-4ff3-b28e-a9c20540d164.png
deleted file mode 100644
index d74230b1..00000000
Binary files a/pics/ca3a793e-06e5-4ff3-b28e-a9c20540d164.png and /dev/null differ
diff --git a/pics/ca711108-e937-4d7d-99aa-61b325c61f1a.jpg b/pics/ca711108-e937-4d7d-99aa-61b325c61f1a.jpg
deleted file mode 100644
index 52f0dec3..00000000
Binary files a/pics/ca711108-e937-4d7d-99aa-61b325c61f1a.jpg and /dev/null differ
diff --git a/pics/cb0ed469-27ab-471b-a830-648b279103c8.png b/pics/cb0ed469-27ab-471b-a830-648b279103c8.png
new file mode 100644
index 00000000..88c2ce86
Binary files /dev/null and b/pics/cb0ed469-27ab-471b-a830-648b279103c8.png differ
diff --git a/pics/cbf50eb8-22b4-4528-a2e7-d187143d57f7.png b/pics/cbf50eb8-22b4-4528-a2e7-d187143d57f7.png
new file mode 100644
index 00000000..27f2b742
Binary files /dev/null and b/pics/cbf50eb8-22b4-4528-a2e7-d187143d57f7.png differ
diff --git a/pics/cd5fbcff-3f35-43a6-8ffa-082a93ce0f0e.png b/pics/cd5fbcff-3f35-43a6-8ffa-082a93ce0f0e.png
new file mode 100644
index 00000000..f8550a11
Binary files /dev/null and b/pics/cd5fbcff-3f35-43a6-8ffa-082a93ce0f0e.png differ
diff --git a/pics/cdbe1d12-5ad9-4acb-a717-bbc822c2acf3.png b/pics/cdbe1d12-5ad9-4acb-a717-bbc822c2acf3.png
new file mode 100644
index 00000000..63792c74
Binary files /dev/null and b/pics/cdbe1d12-5ad9-4acb-a717-bbc822c2acf3.png differ
diff --git a/pics/ce039f03-6588-4f0c-b35b-a494de0eac47.png b/pics/ce039f03-6588-4f0c-b35b-a494de0eac47.png
new file mode 100644
index 00000000..d5025bb3
Binary files /dev/null and b/pics/ce039f03-6588-4f0c-b35b-a494de0eac47.png differ
diff --git a/pics/ceee91c2-da26-4169-94c3-e4608b46b9ac.png b/pics/ceee91c2-da26-4169-94c3-e4608b46b9ac.png
deleted file mode 100644
index 7f3890ad..00000000
Binary files a/pics/ceee91c2-da26-4169-94c3-e4608b46b9ac.png and /dev/null differ
diff --git a/pics/cf4386a1-58c9-4eca-a17f-e12b1e9770eb.png b/pics/cf4386a1-58c9-4eca-a17f-e12b1e9770eb.png
new file mode 100644
index 00000000..9b9f3838
Binary files /dev/null and b/pics/cf4386a1-58c9-4eca-a17f-e12b1e9770eb.png differ
diff --git a/pics/class_loader_hierarchy.png b/pics/class_loader_hierarchy.png
new file mode 100644
index 00000000..6f4baf6f
Binary files /dev/null and b/pics/class_loader_hierarchy.png differ
diff --git a/pics/cookiedata.png b/pics/cookiedata.png
new file mode 100644
index 00000000..a425fca6
Binary files /dev/null and b/pics/cookiedata.png differ
diff --git a/pics/d1f81ac3-9fdb-4371-a49d-ca84917aa89f.jpg b/pics/d1f81ac3-9fdb-4371-a49d-ca84917aa89f.jpg
new file mode 100644
index 00000000..90953621
Binary files /dev/null and b/pics/d1f81ac3-9fdb-4371-a49d-ca84917aa89f.jpg differ
diff --git a/pics/d206d090-d911-4263-a1fe-d6f63f5d1776.png b/pics/d206d090-d911-4263-a1fe-d6f63f5d1776.png
deleted file mode 100644
index 1db3d0a5..00000000
Binary files a/pics/d206d090-d911-4263-a1fe-d6f63f5d1776.png and /dev/null differ
diff --git a/pics/d2c55c84-aa1f-43c1-bd97-457bcb7816b3.png b/pics/d2c55c84-aa1f-43c1-bd97-457bcb7816b3.png
deleted file mode 100644
index 6972e560..00000000
Binary files a/pics/d2c55c84-aa1f-43c1-bd97-457bcb7816b3.png and /dev/null differ
diff --git a/pics/d2d34239-e7c1-482b-b33e-3170c5943556.jpg b/pics/d2d34239-e7c1-482b-b33e-3170c5943556.jpg
new file mode 100644
index 00000000..22f5062a
Binary files /dev/null and b/pics/d2d34239-e7c1-482b-b33e-3170c5943556.jpg differ
diff --git a/pics/d301774f-e0d2-41f3-95f4-bfe39859b52e.jpg b/pics/d301774f-e0d2-41f3-95f4-bfe39859b52e.jpg
deleted file mode 100644
index b2a49201..00000000
Binary files a/pics/d301774f-e0d2-41f3-95f4-bfe39859b52e.jpg and /dev/null differ
diff --git a/pics/d49466db-fdd3-4d36-8a86-47dc45c07a1e.jpg b/pics/d49466db-fdd3-4d36-8a86-47dc45c07a1e.jpg
new file mode 100644
index 00000000..c4c7a915
Binary files /dev/null and b/pics/d49466db-fdd3-4d36-8a86-47dc45c07a1e.jpg differ
diff --git a/pics/d4c3a4a1-0846-46ec-9cc3-eaddfca71254.jpg b/pics/d4c3a4a1-0846-46ec-9cc3-eaddfca71254.jpg
new file mode 100644
index 00000000..bbc7f102
Binary files /dev/null and b/pics/d4c3a4a1-0846-46ec-9cc3-eaddfca71254.jpg differ
diff --git a/pics/d4eef1e2-5703-4ca4-82ab-8dda93d6b81f.png b/pics/d4eef1e2-5703-4ca4-82ab-8dda93d6b81f.png
new file mode 100644
index 00000000..4f62e6f5
Binary files /dev/null and b/pics/d4eef1e2-5703-4ca4-82ab-8dda93d6b81f.png differ
diff --git a/pics/d7f6dec1-02b6-4969-b3ab-e01ee78659b9.png b/pics/d7f6dec1-02b6-4969-b3ab-e01ee78659b9.png
new file mode 100644
index 00000000..2fa7a4e4
Binary files /dev/null and b/pics/d7f6dec1-02b6-4969-b3ab-e01ee78659b9.png differ
diff --git a/pics/d8355d56-aa2b-4452-8001-8475cc095af1.jpg b/pics/d8355d56-aa2b-4452-8001-8475cc095af1.jpg
deleted file mode 100644
index 96051090..00000000
Binary files a/pics/d8355d56-aa2b-4452-8001-8475cc095af1.jpg and /dev/null differ
diff --git a/pics/d8f873fc-00bc-41ee-a87c-c1b4c0172844.png b/pics/d8f873fc-00bc-41ee-a87c-c1b4c0172844.png
deleted file mode 100644
index 19cd1543..00000000
Binary files a/pics/d8f873fc-00bc-41ee-a87c-c1b4c0172844.png and /dev/null differ
diff --git a/pics/d990c0e7-64d1-4ba3-8356-111bc91e53c5.png b/pics/d990c0e7-64d1-4ba3-8356-111bc91e53c5.png
deleted file mode 100644
index cf31e608..00000000
Binary files a/pics/d990c0e7-64d1-4ba3-8356-111bc91e53c5.png and /dev/null differ
diff --git a/pics/d99dc9e2-197c-4085-813d-7195da1c6762.png b/pics/d99dc9e2-197c-4085-813d-7195da1c6762.png
new file mode 100644
index 00000000..a2c04eba
Binary files /dev/null and b/pics/d99dc9e2-197c-4085-813d-7195da1c6762.png differ
diff --git a/pics/db4921d4-184b-48ba-a3cf-1d1141e3ba2d.png b/pics/db4921d4-184b-48ba-a3cf-1d1141e3ba2d.png
new file mode 100644
index 00000000..f88f3b54
Binary files /dev/null and b/pics/db4921d4-184b-48ba-a3cf-1d1141e3ba2d.png differ
diff --git a/pics/db54db2f-82b2-4222-8d63-e49a8a7fc966.png b/pics/db54db2f-82b2-4222-8d63-e49a8a7fc966.png
new file mode 100644
index 00000000..23fb9449
Binary files /dev/null and b/pics/db54db2f-82b2-4222-8d63-e49a8a7fc966.png differ
diff --git a/pics/dbb8516d-37ba-4e2c-b26b-eefd7de21b45.png b/pics/dbb8516d-37ba-4e2c-b26b-eefd7de21b45.png
new file mode 100644
index 00000000..1ee66db8
Binary files /dev/null and b/pics/dbb8516d-37ba-4e2c-b26b-eefd7de21b45.png differ
diff --git a/pics/dbc5c9f1-c13c-4d06-86ba-7cc949eb4c8f.jpg b/pics/dbc5c9f1-c13c-4d06-86ba-7cc949eb4c8f.jpg
new file mode 100644
index 00000000..6ff8388d
Binary files /dev/null and b/pics/dbc5c9f1-c13c-4d06-86ba-7cc949eb4c8f.jpg differ
diff --git a/pics/dbd60b1f-b700-4da6-a993-62578e892333.jpg b/pics/dbd60b1f-b700-4da6-a993-62578e892333.jpg
new file mode 100644
index 00000000..aa076b2a
Binary files /dev/null and b/pics/dbd60b1f-b700-4da6-a993-62578e892333.jpg differ
diff --git a/pics/dc00f70e-c5c8-4d20-baf1-2d70014a97e3.jpg b/pics/dc00f70e-c5c8-4d20-baf1-2d70014a97e3.jpg
new file mode 100644
index 00000000..8090706b
Binary files /dev/null and b/pics/dc00f70e-c5c8-4d20-baf1-2d70014a97e3.jpg differ
diff --git a/pics/dc3e704c-7c57-42b8-93ea-ddd068665964.jpg b/pics/dc3e704c-7c57-42b8-93ea-ddd068665964.jpg
deleted file mode 100644
index 558b8e11..00000000
Binary files a/pics/dc3e704c-7c57-42b8-93ea-ddd068665964.jpg and /dev/null differ
diff --git a/pics/dc695f48-4189-4fc7-b950-ed25f6c80f82.jpg b/pics/dc695f48-4189-4fc7-b950-ed25f6c80f82.jpg
deleted file mode 100644
index d612005a..00000000
Binary files a/pics/dc695f48-4189-4fc7-b950-ed25f6c80f82.jpg and /dev/null differ
diff --git a/pics/dcf265ad-fe35-424d-b4b7-d149cdf239f4.png b/pics/dcf265ad-fe35-424d-b4b7-d149cdf239f4.png
deleted file mode 100644
index 8e598077..00000000
Binary files a/pics/dcf265ad-fe35-424d-b4b7-d149cdf239f4.png and /dev/null differ
diff --git a/pics/dd15a984-e977-4644-b127-708cddb8ca99.png b/pics/dd15a984-e977-4644-b127-708cddb8ca99.png
new file mode 100644
index 00000000..4397e88e
Binary files /dev/null and b/pics/dd15a984-e977-4644-b127-708cddb8ca99.png differ
diff --git a/pics/dd782132-d830-4c55-9884-cfac0a541b8e.png b/pics/dd782132-d830-4c55-9884-cfac0a541b8e.png
new file mode 100644
index 00000000..e338c1bb
Binary files /dev/null and b/pics/dd782132-d830-4c55-9884-cfac0a541b8e.png differ
diff --git a/pics/dd78a1fe-1ff3-4bcf-a56f-8c003995beb6.jpg b/pics/dd78a1fe-1ff3-4bcf-a56f-8c003995beb6.jpg
new file mode 100644
index 00000000..69f4dd05
Binary files /dev/null and b/pics/dd78a1fe-1ff3-4bcf-a56f-8c003995beb6.jpg differ
diff --git a/pics/dda1608d-26e0-4f10-8327-a459969b150a.png b/pics/dda1608d-26e0-4f10-8327-a459969b150a.png
new file mode 100644
index 00000000..7bab8f15
Binary files /dev/null and b/pics/dda1608d-26e0-4f10-8327-a459969b150a.png differ
diff --git a/pics/ddcf2327-8d84-425d-8535-121a94bcb88d.jpg b/pics/ddcf2327-8d84-425d-8535-121a94bcb88d.jpg
new file mode 100644
index 00000000..2a95d92d
Binary files /dev/null and b/pics/ddcf2327-8d84-425d-8535-121a94bcb88d.jpg differ
diff --git a/pics/ddf72ca9-c0be-49d7-ab81-57a99a974c8e.jpg b/pics/ddf72ca9-c0be-49d7-ab81-57a99a974c8e.jpg
deleted file mode 100644
index a31c3adf..00000000
Binary files a/pics/ddf72ca9-c0be-49d7-ab81-57a99a974c8e.jpg and /dev/null differ
diff --git a/pics/de1e46d2-748f-4da3-a29e-7de7bc840366.jpg b/pics/de1e46d2-748f-4da3-a29e-7de7bc840366.jpg
deleted file mode 100644
index d5b8c80b..00000000
Binary files a/pics/de1e46d2-748f-4da3-a29e-7de7bc840366.jpg and /dev/null differ
diff --git a/pics/de284292-b275-4454-8a98-f7e0de370a78.jpg b/pics/de284292-b275-4454-8a98-f7e0de370a78.jpg
deleted file mode 100644
index b74e10bd..00000000
Binary files a/pics/de284292-b275-4454-8a98-f7e0de370a78.jpg and /dev/null differ
diff --git a/pics/de7c5a31-55f5-4e9d-92ec-4ed5b2ec3828.jpg b/pics/de7c5a31-55f5-4e9d-92ec-4ed5b2ec3828.jpg
deleted file mode 100644
index bb2f80cf..00000000
Binary files a/pics/de7c5a31-55f5-4e9d-92ec-4ed5b2ec3828.jpg and /dev/null differ
diff --git a/pics/dec6c6cc-1b5f-44ed-b8fd-464fcf849dac.png b/pics/dec6c6cc-1b5f-44ed-b8fd-464fcf849dac.png
deleted file mode 100644
index e642c84a..00000000
Binary files a/pics/dec6c6cc-1b5f-44ed-b8fd-464fcf849dac.png and /dev/null differ
diff --git a/pics/decb0936-e83c-4a55-840a-fe8aa101ac61.png b/pics/decb0936-e83c-4a55-840a-fe8aa101ac61.png
deleted file mode 100644
index 1e7a8bca..00000000
Binary files a/pics/decb0936-e83c-4a55-840a-fe8aa101ac61.png and /dev/null differ
diff --git a/pics/df48ea1b-3069-4fb7-93c0-4c8a26c7ed7c.png b/pics/df48ea1b-3069-4fb7-93c0-4c8a26c7ed7c.png
deleted file mode 100644
index 4f0ffa3d..00000000
Binary files a/pics/df48ea1b-3069-4fb7-93c0-4c8a26c7ed7c.png and /dev/null differ
diff --git a/pics/dfd078b2-aa4f-4c50-8319-232922d822b8.jpg b/pics/dfd078b2-aa4f-4c50-8319-232922d822b8.jpg
deleted file mode 100644
index d708d036..00000000
Binary files a/pics/dfd078b2-aa4f-4c50-8319-232922d822b8.jpg and /dev/null differ
diff --git a/pics/e024bd7e-fb4e-4239-9451-9a6227f50b00.jpg b/pics/e024bd7e-fb4e-4239-9451-9a6227f50b00.jpg
deleted file mode 100644
index 99352484..00000000
Binary files a/pics/e024bd7e-fb4e-4239-9451-9a6227f50b00.jpg and /dev/null differ
diff --git a/pics/e026c24d-00fa-4e7c-97a8-95a98cdc383a.png b/pics/e026c24d-00fa-4e7c-97a8-95a98cdc383a.png
new file mode 100644
index 00000000..427cfbf8
Binary files /dev/null and b/pics/e026c24d-00fa-4e7c-97a8-95a98cdc383a.png differ
diff --git a/pics/e0be6970-5b0e-44a2-bc71-df4d61c42b8f.jpg b/pics/e0be6970-5b0e-44a2-bc71-df4d61c42b8f.jpg
deleted file mode 100644
index 0067761b..00000000
Binary files a/pics/e0be6970-5b0e-44a2-bc71-df4d61c42b8f.jpg and /dev/null differ
diff --git a/pics/e13833c8-e215-462e-855c-1d362bb8d4a0.jpg b/pics/e13833c8-e215-462e-855c-1d362bb8d4a0.jpg
deleted file mode 100644
index 3f535bf1..00000000
Binary files a/pics/e13833c8-e215-462e-855c-1d362bb8d4a0.jpg and /dev/null differ
diff --git a/pics/e143f6da-d114-4ba4-8712-f65299047fa2.png b/pics/e143f6da-d114-4ba4-8712-f65299047fa2.png
new file mode 100644
index 00000000..38cbf667
Binary files /dev/null and b/pics/e143f6da-d114-4ba4-8712-f65299047fa2.png differ
diff --git a/pics/e198c201-f386-4491-8ad6-f7e433bf992d.png b/pics/e198c201-f386-4491-8ad6-f7e433bf992d.png
deleted file mode 100644
index 5c0e1f15..00000000
Binary files a/pics/e198c201-f386-4491-8ad6-f7e433bf992d.png and /dev/null differ
diff --git a/pics/e2f0d889-2330-424c-8193-198edebecff7.png b/pics/e2f0d889-2330-424c-8193-198edebecff7.png
new file mode 100644
index 00000000..d747d6f3
Binary files /dev/null and b/pics/e2f0d889-2330-424c-8193-198edebecff7.png differ
diff --git a/pics/e3124763-f75e-46c3-ba82-341e6c98d862.jpg b/pics/e3124763-f75e-46c3-ba82-341e6c98d862.jpg
new file mode 100644
index 00000000..80643657
Binary files /dev/null and b/pics/e3124763-f75e-46c3-ba82-341e6c98d862.jpg differ
diff --git a/pics/e31abb94-9201-4e06-9902-61101b92f475.png b/pics/e31abb94-9201-4e06-9902-61101b92f475.png
new file mode 100644
index 00000000..90833a5c
Binary files /dev/null and b/pics/e31abb94-9201-4e06-9902-61101b92f475.png differ
diff --git a/pics/e41405a8-7c05-4f70-8092-e961e28d3112.jpg b/pics/e41405a8-7c05-4f70-8092-e961e28d3112.jpg
new file mode 100644
index 00000000..7fef7ba5
Binary files /dev/null and b/pics/e41405a8-7c05-4f70-8092-e961e28d3112.jpg differ
diff --git a/pics/e5ad625e-729d-4a8d-923a-7c3df5773e1c.jpg b/pics/e5ad625e-729d-4a8d-923a-7c3df5773e1c.jpg
deleted file mode 100644
index c232264b..00000000
Binary files a/pics/e5ad625e-729d-4a8d-923a-7c3df5773e1c.jpg and /dev/null differ
diff --git a/pics/e5baeb38-0ec9-4ad7-8374-1cdb0dba74a6.jpg b/pics/e5baeb38-0ec9-4ad7-8374-1cdb0dba74a6.jpg
deleted file mode 100644
index 917ae55a..00000000
Binary files a/pics/e5baeb38-0ec9-4ad7-8374-1cdb0dba74a6.jpg and /dev/null differ
diff --git a/pics/e6723b94-1a33-4605-b775-f6813352d383.png b/pics/e6723b94-1a33-4605-b775-f6813352d383.png
new file mode 100644
index 00000000..fdb9280e
Binary files /dev/null and b/pics/e6723b94-1a33-4605-b775-f6813352d383.png differ
diff --git a/pics/e800b001-7779-495b-8459-d33a7440d7b8.jpg b/pics/e800b001-7779-495b-8459-d33a7440d7b8.jpg
new file mode 100644
index 00000000..07d6b7a9
Binary files /dev/null and b/pics/e800b001-7779-495b-8459-d33a7440d7b8.jpg differ
diff --git a/pics/e92d0ebc-7d46-413b-aec1-34a39602f787.png b/pics/e92d0ebc-7d46-413b-aec1-34a39602f787.png
new file mode 100644
index 00000000..1090a779
Binary files /dev/null and b/pics/e92d0ebc-7d46-413b-aec1-34a39602f787.png differ
diff --git a/pics/ea5ed9b2-6d9f-48fb-b890-0288caf9088a.jpg b/pics/ea5ed9b2-6d9f-48fb-b890-0288caf9088a.jpg
deleted file mode 100644
index c69b6b27..00000000
Binary files a/pics/ea5ed9b2-6d9f-48fb-b890-0288caf9088a.jpg and /dev/null differ
diff --git a/pics/ea5f3efe-d5e6-499b-b278-9e898af61257.jpg b/pics/ea5f3efe-d5e6-499b-b278-9e898af61257.jpg
new file mode 100644
index 00000000..a07e736d
Binary files /dev/null and b/pics/ea5f3efe-d5e6-499b-b278-9e898af61257.jpg differ
diff --git a/pics/eb4a7007-d437-4740-865d-672973effe25.png b/pics/eb4a7007-d437-4740-865d-672973effe25.png
new file mode 100644
index 00000000..e2a0faab
Binary files /dev/null and b/pics/eb4a7007-d437-4740-865d-672973effe25.png differ
diff --git a/pics/eb6271de-22c9-4f4b-8b31-eab1f560efac.png b/pics/eb6271de-22c9-4f4b-8b31-eab1f560efac.png
deleted file mode 100644
index 05d5d201..00000000
Binary files a/pics/eb6271de-22c9-4f4b-8b31-eab1f560efac.png and /dev/null differ
diff --git a/pics/ebf03f56-f957-4435-9f8f-0f605661484d.jpg b/pics/ebf03f56-f957-4435-9f8f-0f605661484d.jpg
deleted file mode 100644
index 90c92eac..00000000
Binary files a/pics/ebf03f56-f957-4435-9f8f-0f605661484d.jpg and /dev/null differ
diff --git a/pics/ed62f400-192c-4185-899b-187958201f0c.jpg b/pics/ed62f400-192c-4185-899b-187958201f0c.jpg
deleted file mode 100644
index 3be32766..00000000
Binary files a/pics/ed62f400-192c-4185-899b-187958201f0c.jpg and /dev/null differ
diff --git a/pics/ed7b96ac-6428-4bd5-9986-674c54c2a959.png b/pics/ed7b96ac-6428-4bd5-9986-674c54c2a959.png
new file mode 100644
index 00000000..78b3cace
Binary files /dev/null and b/pics/ed7b96ac-6428-4bd5-9986-674c54c2a959.png differ
diff --git a/pics/edc23f99-c46c-4200-b64e-07516828720d.jpg b/pics/edc23f99-c46c-4200-b64e-07516828720d.jpg
deleted file mode 100644
index 71abc4d3..00000000
Binary files a/pics/edc23f99-c46c-4200-b64e-07516828720d.jpg and /dev/null differ
diff --git a/pics/eebdeb57-8efb-4848-9bb6-97512990897c.jpg b/pics/eebdeb57-8efb-4848-9bb6-97512990897c.jpg
new file mode 100644
index 00000000..5632fdca
Binary files /dev/null and b/pics/eebdeb57-8efb-4848-9bb6-97512990897c.jpg differ
diff --git a/pics/ef280699-da36-4b38-9735-9b048a3c7fe0.png b/pics/ef280699-da36-4b38-9735-9b048a3c7fe0.png
new file mode 100644
index 00000000..a54c85e1
Binary files /dev/null and b/pics/ef280699-da36-4b38-9735-9b048a3c7fe0.png differ
diff --git a/pics/ef8eab00-1d5e-4d99-a7c2-d6d68ea7fe92.png b/pics/ef8eab00-1d5e-4d99-a7c2-d6d68ea7fe92.png
new file mode 100644
index 00000000..f6256867
Binary files /dev/null and b/pics/ef8eab00-1d5e-4d99-a7c2-d6d68ea7fe92.png differ
diff --git a/pics/f0321ed1-fa93-460e-951b-4239fef819f3.jpg b/pics/f0321ed1-fa93-460e-951b-4239fef819f3.jpg
deleted file mode 100644
index f9b99f27..00000000
Binary files a/pics/f0321ed1-fa93-460e-951b-4239fef819f3.jpg and /dev/null differ
diff --git a/pics/f0e35b7a-2948-488a-a5a9-97d3f6b5e2d7.png b/pics/f0e35b7a-2948-488a-a5a9-97d3f6b5e2d7.png
new file mode 100644
index 00000000..8f797bdf
Binary files /dev/null and b/pics/f0e35b7a-2948-488a-a5a9-97d3f6b5e2d7.png differ
diff --git a/pics/f1fb826b-ecf4-4ddb-91f0-2bafecf08869.jpg b/pics/f1fb826b-ecf4-4ddb-91f0-2bafecf08869.jpg
deleted file mode 100644
index 736e72f4..00000000
Binary files a/pics/f1fb826b-ecf4-4ddb-91f0-2bafecf08869.jpg and /dev/null differ
diff --git a/pics/f2e0cee9-ecdc-4a96-853f-d9f6a1ad6ad1.jpg b/pics/f2e0cee9-ecdc-4a96-853f-d9f6a1ad6ad1.jpg
deleted file mode 100644
index 7e71f69c..00000000
Binary files a/pics/f2e0cee9-ecdc-4a96-853f-d9f6a1ad6ad1.jpg and /dev/null differ
diff --git a/pics/f3080f83-6239-459b-8e9c-03b6641f7815.png b/pics/f3080f83-6239-459b-8e9c-03b6641f7815.png
new file mode 100644
index 00000000..77868686
Binary files /dev/null and b/pics/f3080f83-6239-459b-8e9c-03b6641f7815.png differ
diff --git a/pics/f3bfe11f-9cba-4ff2-8cc6-629068408a80.jpg b/pics/f3bfe11f-9cba-4ff2-8cc6-629068408a80.jpg
new file mode 100644
index 00000000..9dc7a1d5
Binary files /dev/null and b/pics/f3bfe11f-9cba-4ff2-8cc6-629068408a80.jpg differ
diff --git a/pics/f3d3e072-e947-43e9-b999-22385fd569a0.jpg b/pics/f3d3e072-e947-43e9-b999-22385fd569a0.jpg
new file mode 100644
index 00000000..7b08be6d
Binary files /dev/null and b/pics/f3d3e072-e947-43e9-b999-22385fd569a0.jpg differ
diff --git a/pics/f42443e0-208d-41ea-be44-c7fd97d2e3bf.png b/pics/f42443e0-208d-41ea-be44-c7fd97d2e3bf.png
new file mode 100644
index 00000000..4a5dbd1d
Binary files /dev/null and b/pics/f42443e0-208d-41ea-be44-c7fd97d2e3bf.png differ
diff --git a/pics/f48e2b92-2c2a-48cb-a443-bd313e187a25.jpg b/pics/f48e2b92-2c2a-48cb-a443-bd313e187a25.jpg
new file mode 100644
index 00000000..66c29824
Binary files /dev/null and b/pics/f48e2b92-2c2a-48cb-a443-bd313e187a25.jpg differ
diff --git a/pics/f4cdda3e-324c-49b5-8c14-08a3db634b29.png b/pics/f4cdda3e-324c-49b5-8c14-08a3db634b29.png
new file mode 100644
index 00000000..60fc81bb
Binary files /dev/null and b/pics/f4cdda3e-324c-49b5-8c14-08a3db634b29.png differ
diff --git a/pics/f50a8e52-a683-444c-8e32-63c1890fe84a.jpg b/pics/f50a8e52-a683-444c-8e32-63c1890fe84a.jpg
new file mode 100644
index 00000000..46a01b3e
Binary files /dev/null and b/pics/f50a8e52-a683-444c-8e32-63c1890fe84a.jpg differ
diff --git a/pics/f5477abd-c246-4851-89ab-6b1cde2549b1.png b/pics/f5477abd-c246-4851-89ab-6b1cde2549b1.png
new file mode 100644
index 00000000..f358a009
Binary files /dev/null and b/pics/f5477abd-c246-4851-89ab-6b1cde2549b1.png differ
diff --git a/pics/f5757d09-88e7-4bbd-8cfb-cecf55604854.png b/pics/f5757d09-88e7-4bbd-8cfb-cecf55604854.png
new file mode 100644
index 00000000..1664247b
Binary files /dev/null and b/pics/f5757d09-88e7-4bbd-8cfb-cecf55604854.png differ
diff --git a/pics/f61b5419-c94a-4df1-8d4d-aed9ae8cc6d5.png b/pics/f61b5419-c94a-4df1-8d4d-aed9ae8cc6d5.png
new file mode 100644
index 00000000..dc0d4e34
Binary files /dev/null and b/pics/f61b5419-c94a-4df1-8d4d-aed9ae8cc6d5.png differ
diff --git a/pics/f716427a-94f2-4875-9c86-98793cf5dcc3.jpg b/pics/f716427a-94f2-4875-9c86-98793cf5dcc3.jpg
new file mode 100644
index 00000000..15da1c6b
Binary files /dev/null and b/pics/f716427a-94f2-4875-9c86-98793cf5dcc3.jpg differ
diff --git a/pics/f7d170a3-e446-4a64-ac2d-cb95028f81a8.png b/pics/f7d170a3-e446-4a64-ac2d-cb95028f81a8.png
new file mode 100644
index 00000000..26799762
Binary files /dev/null and b/pics/f7d170a3-e446-4a64-ac2d-cb95028f81a8.png differ
diff --git a/pics/f7d5da89-2d75-4d8f-85e7-6b608865dc00.jpg b/pics/f7d5da89-2d75-4d8f-85e7-6b608865dc00.jpg
deleted file mode 100644
index a0abac53..00000000
Binary files a/pics/f7d5da89-2d75-4d8f-85e7-6b608865dc00.jpg and /dev/null differ
diff --git a/pics/f7d880c9-740a-4a16-ac6d-be502281b4b2.jpg b/pics/f7d880c9-740a-4a16-ac6d-be502281b4b2.jpg
deleted file mode 100644
index d1232fcf..00000000
Binary files a/pics/f7d880c9-740a-4a16-ac6d-be502281b4b2.jpg and /dev/null differ
diff --git a/pics/f7ecbb8d-bb8b-4d45-a3b7-f49425d6d83d.jpg b/pics/f7ecbb8d-bb8b-4d45-a3b7-f49425d6d83d.jpg
new file mode 100644
index 00000000..ab51d486
Binary files /dev/null and b/pics/f7ecbb8d-bb8b-4d45-a3b7-f49425d6d83d.jpg differ
diff --git a/pics/f87afe72-c2df-4c12-ac03-9b8d581a8af8.jpg b/pics/f87afe72-c2df-4c12-ac03-9b8d581a8af8.jpg
new file mode 100644
index 00000000..6a090993
Binary files /dev/null and b/pics/f87afe72-c2df-4c12-ac03-9b8d581a8af8.jpg differ
diff --git a/pics/f8b12555-967b-423d-a84e-bc9eff104b8b.jpg b/pics/f8b12555-967b-423d-a84e-bc9eff104b8b.jpg
deleted file mode 100644
index 9e26464a..00000000
Binary files a/pics/f8b12555-967b-423d-a84e-bc9eff104b8b.jpg and /dev/null differ
diff --git a/pics/f8b16d1e-7363-4544-94d6-4939fdf849dc.png b/pics/f8b16d1e-7363-4544-94d6-4939fdf849dc.png
new file mode 100644
index 00000000..300b6e06
Binary files /dev/null and b/pics/f8b16d1e-7363-4544-94d6-4939fdf849dc.png differ
diff --git a/pics/f99c019e-7e91-4c2e-b94d-b031c402dcb5.jpg b/pics/f99c019e-7e91-4c2e-b94d-b031c402dcb5.jpg
deleted file mode 100644
index af9fe3e8..00000000
Binary files a/pics/f99c019e-7e91-4c2e-b94d-b031c402dcb5.jpg and /dev/null differ
diff --git a/pics/f9ed4da5-0032-41e6-991a-36d995ec28fd.png b/pics/f9ed4da5-0032-41e6-991a-36d995ec28fd.png
deleted file mode 100644
index cf85f6be..00000000
Binary files a/pics/f9ed4da5-0032-41e6-991a-36d995ec28fd.png and /dev/null differ
diff --git a/pics/f9f9f993-8ece-4da7-b848-af9b438fad76.png b/pics/f9f9f993-8ece-4da7-b848-af9b438fad76.png
new file mode 100644
index 00000000..824904db
Binary files /dev/null and b/pics/f9f9f993-8ece-4da7-b848-af9b438fad76.png differ
diff --git a/pics/fa4101d7-19ce-4a69-a84f-20bbe64320e5.jpg b/pics/fa4101d7-19ce-4a69-a84f-20bbe64320e5.jpg
deleted file mode 100644
index 0d9e2948..00000000
Binary files a/pics/fa4101d7-19ce-4a69-a84f-20bbe64320e5.jpg and /dev/null differ
diff --git a/pics/fa568fac-ac58-48dd-a9bb-23b3065bf2dc.png b/pics/fa568fac-ac58-48dd-a9bb-23b3065bf2dc.png
new file mode 100644
index 00000000..736a2861
Binary files /dev/null and b/pics/fa568fac-ac58-48dd-a9bb-23b3065bf2dc.png differ
diff --git a/pics/fabd5fa0-b75e-48d0-9e2c-31471945ceb9.jpg b/pics/fabd5fa0-b75e-48d0-9e2c-31471945ceb9.jpg
deleted file mode 100644
index 650394df..00000000
Binary files a/pics/fabd5fa0-b75e-48d0-9e2c-31471945ceb9.jpg and /dev/null differ
diff --git a/pics/fb327611-7e2b-4f2f-9f5b-38592d408f07.png b/pics/fb327611-7e2b-4f2f-9f5b-38592d408f07.png
new file mode 100644
index 00000000..774ecf10
Binary files /dev/null and b/pics/fb327611-7e2b-4f2f-9f5b-38592d408f07.png differ
diff --git a/pics/fb546e12-e1fb-4b72-a1fb-8a7f5000dce6.jpg b/pics/fb546e12-e1fb-4b72-a1fb-8a7f5000dce6.jpg
new file mode 100644
index 00000000..e15f11db
Binary files /dev/null and b/pics/fb546e12-e1fb-4b72-a1fb-8a7f5000dce6.jpg differ
diff --git a/pics/fbe54203-c005-48f0-8883-b05e564a3173.png b/pics/fbe54203-c005-48f0-8883-b05e564a3173.png
new file mode 100644
index 00000000..dee6a88d
Binary files /dev/null and b/pics/fbe54203-c005-48f0-8883-b05e564a3173.png differ
diff --git a/pics/fe3d224c-8ffd-40f9-85b1-86ffe1393f6c.jpg b/pics/fe3d224c-8ffd-40f9-85b1-86ffe1393f6c.jpg
deleted file mode 100644
index 6cd4d187..00000000
Binary files a/pics/fe3d224c-8ffd-40f9-85b1-86ffe1393f6c.jpg and /dev/null differ
diff --git a/pics/ff0c019c-6461-467d-a266-0455341fd4f4.png b/pics/ff0c019c-6461-467d-a266-0455341fd4f4.png
new file mode 100644
index 00000000..658867d2
Binary files /dev/null and b/pics/ff0c019c-6461-467d-a266-0455341fd4f4.png differ
diff --git a/pics/ff17c103-750a-4bb8-9afa-576327023af9.png b/pics/ff17c103-750a-4bb8-9afa-576327023af9.png
deleted file mode 100644
index 5f9f99b3..00000000
Binary files a/pics/ff17c103-750a-4bb8-9afa-576327023af9.png and /dev/null differ
diff --git a/pics/ff396233-1bb1-4e74-8bc2-d7c90146f5dd.png b/pics/ff396233-1bb1-4e74-8bc2-d7c90146f5dd.png
new file mode 100644
index 00000000..4eaa11ba
Binary files /dev/null and b/pics/ff396233-1bb1-4e74-8bc2-d7c90146f5dd.png differ
diff --git a/pics/flow.png b/pics/flow.png
new file mode 100644
index 00000000..aa0492a4
Binary files /dev/null and b/pics/flow.png differ
diff --git a/pics/hashMap_u54C8_u5E0C_u7B97_u6CD5_u4F8B_u56FE.png b/pics/hashMap_u54C8_u5E0C_u7B97_u6CD5_u4F8B_u56FE.png
new file mode 100644
index 00000000..f19ceb66
Binary files /dev/null and b/pics/hashMap_u54C8_u5E0C_u7B97_u6CD5_u4F8B_u56FE.png differ
diff --git a/pics/image005.jpg b/pics/image005.jpg
new file mode 100644
index 00000000..a20df7d8
Binary files /dev/null and b/pics/image005.jpg differ
diff --git a/pics/image007.jpg b/pics/image007.jpg
new file mode 100644
index 00000000..822aa76e
Binary files /dev/null and b/pics/image007.jpg differ
diff --git a/pics/image008.jpg b/pics/image008.jpg
new file mode 100644
index 00000000..952f4aa8
Binary files /dev/null and b/pics/image008.jpg differ
diff --git a/pics/image009.jpg b/pics/image009.jpg
new file mode 100644
index 00000000..6e956fec
Binary files /dev/null and b/pics/image009.jpg differ
diff --git a/pics/java-collections.png b/pics/java-collections.png
new file mode 100644
index 00000000..6a68af89
Binary files /dev/null and b/pics/java-collections.png differ
diff --git a/pics/java-collections1.png b/pics/java-collections1.png
new file mode 100644
index 00000000..852282ef
Binary files /dev/null and b/pics/java-collections1.png differ
diff --git a/pics/linux-filesystem.png b/pics/linux-filesystem.png
new file mode 100644
index 00000000..ae965295
Binary files /dev/null and b/pics/linux-filesystem.png differ
diff --git a/pics/monitor-lock-rule.png b/pics/monitor-lock-rule.png
new file mode 100644
index 00000000..6590d94b
Binary files /dev/null and b/pics/monitor-lock-rule.png differ
diff --git a/pics/mutualssl_small.png b/pics/mutualssl_small.png
new file mode 100644
index 00000000..1541eba8
Binary files /dev/null and b/pics/mutualssl_small.png differ
diff --git a/pics/n2U3N.png b/pics/n2U3N.png
new file mode 100644
index 00000000..cd58b5b9
Binary files /dev/null and b/pics/n2U3N.png differ
diff --git a/pics/network-of-networks.gif b/pics/network-of-networks.gif
new file mode 100644
index 00000000..7473f913
Binary files /dev/null and b/pics/network-of-networks.gif differ
diff --git a/pics/pcrypt.gif b/pics/pcrypt.gif
new file mode 100644
index 00000000..c17b9853
Binary files /dev/null and b/pics/pcrypt.gif differ
diff --git a/pics/ppt_img.gif b/pics/ppt_img.gif
new file mode 100644
index 00000000..8e5ad145
Binary files /dev/null and b/pics/ppt_img.gif differ
diff --git a/pics/scrypt.gif b/pics/scrypt.gif
new file mode 100644
index 00000000..150fea7f
Binary files /dev/null and b/pics/scrypt.gif differ
diff --git a/pics/single-thread-rule.png b/pics/single-thread-rule.png
new file mode 100644
index 00000000..d6583e9e
Binary files /dev/null and b/pics/single-thread-rule.png differ
diff --git a/pics/ssl-offloading.jpg b/pics/ssl-offloading.jpg
new file mode 100644
index 00000000..8f01a418
Binary files /dev/null and b/pics/ssl-offloading.jpg differ
diff --git a/pics/tGPV0.png b/pics/tGPV0.png
new file mode 100644
index 00000000..89fb7bfe
Binary files /dev/null and b/pics/tGPV0.png differ
diff --git a/pics/thread-join-rule.png b/pics/thread-join-rule.png
new file mode 100644
index 00000000..c17d0456
Binary files /dev/null and b/pics/thread-join-rule.png differ
diff --git a/pics/thread-start-rule.png b/pics/thread-start-rule.png
new file mode 100644
index 00000000..60ee7862
Binary files /dev/null and b/pics/thread-start-rule.png differ
diff --git a/pics/url_diagram.png b/pics/url_diagram.png
new file mode 100644
index 00000000..18bbd49b
Binary files /dev/null and b/pics/url_diagram.png differ
diff --git a/pics/volatile-variable-rule.png b/pics/volatile-variable-rule.png
new file mode 100644
index 00000000..1747664b
Binary files /dev/null and b/pics/volatile-variable-rule.png differ