From dea82f66b3382d92d26e84a1590f00dc7e84786c Mon Sep 17 00:00:00 2001 From: ruanyf Date: Thu, 14 Dec 2023 14:34:05 +0800 Subject: [PATCH] docs(history): add history chapter --- chapters.yml | 1 + docs/history.md | 325 +++++++++++++++++++++++++++++++++++++++++++++++ docs/readline.md | 193 ---------------------------- 3 files changed, 326 insertions(+), 193 deletions(-) create mode 100644 docs/history.md diff --git a/chapters.yml b/chapters.yml index 1c7d663..fab0b7d 100644 --- a/chapters.yml +++ b/chapters.yml @@ -5,6 +5,7 @@ - variable.md: 变量 - string.md: 字符串操作 - arithmetic.md: 算术运算 +- history.md: 操作历史 - readline.md: 行操作 - stack.md: 目录堆栈 - script.md: 脚本入门 diff --git a/docs/history.md b/docs/history.md new file mode 100644 index 0000000..c3a90bf --- /dev/null +++ b/docs/history.md @@ -0,0 +1,325 @@ +# 操作历史 + +## 简介 + +Bash 会保留用户的操作历史,即用户输入的每一条命令都会记录,默认是保存最近的500条命令。有了操作历史以后,就可以使用方向键的`↑`和`↓`,快速浏览上一条和下一条命令。 + +退出当前 Shell 的时候,Bash 会将用户在当前 Shell 的操作历史写入`~/.bash_history`文件,该文件默认储存500个操作。 + +环境变量`HISTFILE`总是指向这个文件。 + +```bash +$ echo $HISTFILE +/home/me/.bash_history +``` + +## history 命令 + +`history`命令会输出`.bash_history`文件的全部内容,即输出操作历史。 + +```bash +$ history +... +498 echo Goodbye +499 ls ~ +500 cd +``` + +用户可以使用这个命令,查看最近的操作。相比直接读取`.bash_history`文件,它的优势在于所有命令之前加上了行号。最近的操作在最后面,行号最大。 + +如果想搜索某个以前执行的命令,可以配合`grep`命令搜索操作历史。 + +```bash +$ history | grep /usr/bin +``` + +上面命令返回`.bash_history`文件里面,那些包含`/usr/bin`的命令。 + +`history`命令的`-c`参数可以清除操作历史,即清空`.bash_history`文件。 + +```bash +$ history -c +``` + +## 环境变量 + +### HISTTIMEFORMAT + +通过定制环境变量`HISTTIMEFORMAT`,`history`的输出结果还可以显示每个操作的时间。 + +```bash +$ export HISTTIMEFORMAT='%F %T ' +$ history +1 2013-06-09 10:40:12 cat /etc/issue +2 2013-06-09 10:40:12 clear +``` + +上面代码中,`%F`相当于`%Y - %m - %d`(年-月-日),`%T`相当于` %H : %M : %S`(时:分:秒)。 + +只要设置`HISTTIMEFORMAT`这个环境变量,就会在`.bash_history`文件保存命令的执行时间戳。如果不设置,就不会保存时间戳。 + +### HISTSIZE + +环境变量`HISTSIZE`设置保存历史操作的数量。 + +```bash +$ export HISTSIZE=10000 +``` + +上面命令设置保存过去10000条操作历史。 + +如果不希望保存本次操作的历史,可以设置`HISTSIZE`等于0。 + +```bash +export HISTSIZE=0 +``` + +如果`HISTSIZE=0`写入用户主目录的`~/.bashrc`文件,那么就不会保留该用户的操作历史。如果写入`/etc/profile`,整个系统都不会保留操作历史。 + +### HISTIGNORE + +环境变量`HISTIGNORE`可以设置哪些命令不写入操作历史。 + +```bash +export HISTIGNORE='pwd:ls:exit' +``` + +上面示例设置,`pwd`、`ls`、`exit`这三个命令不写入操作历史。 + +## Ctrl + r + +输入命令时,按下`Ctrl + r`快捷键,就可以搜索操作历史,选择以前执行过的命令。 + +`Ctrl + r`相当于打开一个`.bash_history`文件的搜索接口,直接键入命令的开头部分,Shell 就会自动在该文件中反向查询(即先查询最近的命令),显示最近一条匹配的结果,这时按下回车键,就会执行那条命令。 + +## ! 命令 + +### ! + 行号 + +操作历史的每一条记录都有行号。知道了命令的行号以后,可以用`感叹号 + 行号`执行该命令。如果想要执行`.bash_history`里面的第8条命令,可以像下面这样操作。 + +```bash +$ !8 +``` + +### !- 数字 + +如果想执行本次 Shell 对话中倒数的命令,比如执行倒数第3条命令,就可以输入`!-3`。 + +```bash +$ touch a.txt +$ touch b.txt +$ touch c.txt + +$ !-3 +touch a.txt +``` + +上面示例中,`!-3`返回倒数第3条命令,即`touch a.txt`。 + +它跟`! + 行号`的主要区别是,后者是在`.bash_history`文件中从头开始计算行数,而`!- 数字`是从底部开始向上计算行数。 + +### !! + +`!!`命令返回上一条命令。如果需要重复执行某一条命令,就可以不断键入`!!`,这样非常方便。它等同于`!-1`。 + +```bash +$ echo hello +hello + +$ !! +echo hello +hello +``` + +上面示例中,`!!`会返回并执行上一条命令`echo hello`。 + +有时候,我们使用某条命令,系统报错没有权限,这时就可以使用`sudo !!`。 + +```bash +# 报错,没有执行权限 +$ yum update + +$ sudo !! +sudo yum update +``` + +上面示例中,`sudo !!`返回`sudo yum update`,从而就可以正确执行了。 + +### ! + 搜索词 + +`感叹号 + 搜索词`可以快速执行匹配的命令。 + +```bash +$ echo Hello World +Hello World + +$ echo Goodbye +Goodbye + +$ !e +echo Goodbye +Goodbye +``` + +上面例子中,`!e`表示找出操作历史之中,最近的那一条以`e`开头的命令并执行。Bash 会先输出那一条命令`echo Goodbye`,然后直接执行。 + +同理,`!echo`也会执行最近一条以`echo`开头的命令。 + +```bash +$ !echo +echo Goodbye +Goodbye + +$ !echo H +echo Goodbye H +Goodbye H + +$ !echo H G +echo Goodbye H G +Goodbye H G +``` + +注意,`感叹号 + 搜索词`语法只会匹配命令,不会匹配参数。所以`!echo H`不会执行`echo Hello World`,而是会执行`echo Goodbye`,并把参数`H`附加在这条命令之后。同理,`!echo H G`也是等同于`echo Goodbye`命令之后附加`H G`。 + +由于`感叹号 + 搜索词`会扩展成以前执行过的命令,所以含有`!`的字符串放在双引号里面,必须非常小心,如果它后面有非空格的字符,就很有可能报错。 + +```bash +$ echo "I say:\"hello!\"" +bash: !\: event not found +``` + +上面的命令会报错,原因是感叹号后面是一个反斜杠,Bash 会尝试寻找,以前是否执行过反斜杠开头的命令,一旦找不到就会报错。解决方法就是在感叹号前面,也加上反斜杠。 + +```bash +$ echo "I say:\"hello\!\"" +I say:"hello\!" +``` + +### !? + 搜索词 + +`!? + 搜索词`可以搜索命令的任意部分,包括参数部分。它跟`! + 搜索词`的主要区别是,后者是从行首开始匹配。 + +```bash +$ cat hello.txt +Hello world ..! + +$ !?hello.txt +cat hello.txt +Hello world ..! +``` + +上面示例中,`!?hello.txt`会返回最近一条包括`hello.txt`的命令。 + +### !$,!* + +`!$`代表上一个命令的最后一个参数,它的另一种写法是`$_`。 + +`!*`代表上一个命令的所有参数,即除了命令以外的所有部分。 + +```bash +$ cp a.txt b.txt +$ echo !$ +b.txt + +$ cp a.txt b.txt +$ echo !* +a.txt b.txt +``` + +上面示例中,`!$`代表上一个命令的最后一个参数(`b.txt`),`!*`代表上一个命令的所有参数(`a.txt b.txt`)。 + +如果想匹配上一个命令的某个指定位置的参数,使用`!:n`。 + +```bash +$ ls a.txt b.txt c.txt + +$ echo !:2 +b.txt +``` + +上面示例中,`!:2`返回上一条命令的第二个参数(`b.txt`)。 + +这种写法的`!:$`,代表上一个命令的最后一个参数。事实上,`!$`就是`!:$`的简写形式。 + +```bash +$ ls a.txt b.txt c.txt + +$ echo !:$ +echo c.txt +c.txt +``` + +上面示例中,`!:$`代表上一条命令的最后一个参数(`c.txt`)。 + +如果想匹配更久以前的命令的参数,可以使用`!<命令>:n`(指定位置的参数)和`!<命令>:$`(最后一个参数)。 + +```bash +$ ls !mkdir:$ +``` + +上面示例中,`!mkdir:$`会返回前面最后一条`mkdir`命令的最后一个参数。 + +```bash +$ ls !mk:2 +``` + +上面示例中,`!mk:2`会返回前面最后一条以`mk`开头的命令的第二个参数。 + +### !:p + +如果只是想输出上一条命令,而不是执行它,可以使用`!:p`。 + +```bash +$ echo hello + +$ !:p +echo hello +``` + +上面示例中,`!:p`只会输出`echo hello`,而不会执行这条命令。 + +如果想输出最近一条匹配的命令,而不执行它,可以使用`!<命令>:p`。 + +```bash +$ !su:p +``` + +上面示例中,`!su:p`会输出前面最近一条以`su`开头的命令,而不执行它。 + +## `^string1^string2` + +`^string1^string2`用来执行最近一条包含`string1`的命令,将其替换成`string2`。 + +```bash +$ rm /var/log/httpd/error.log +$ ^error^access +rm /var/log/httpd/access.log +``` + +上面示例中,`^error^access`将最近一条含有`error`的命令里面的`error`,替换成`access`。 + +## histverify 参数 + +上面的那些快捷命令(比如`!!`命令),都是找到匹配的命令后,直接执行。如果希望增加一个确认步骤,先输出是什么命令,让用户确认后再执行,可以打开 Shell 的`histverify`选项。 + +```bash +$ shopt -s histverify +``` + +打开`histverify`这个选项后,使用`!`快捷键所返回的命令,就会先输出,等到用户按下回车键后再执行。 + +## 快捷键 + +下面是其他一些与操作历史相关的快捷键。 + +- `Ctrl + p`:显示上一个命令,与向上箭头效果相同(previous)。 +- `Ctrl + n`:显示下一个命令,与向下箭头效果相同(next)。 +- `Alt + <`:显示第一个命令。 +- `Alt + >`:显示最后一个命令,即当前的命令。 +- `Ctrl + o`:执行历史文件里面的当前条目,并自动显示下一条命令。这对重复执行某个序列的命令很有帮助。 + +## 参考链接 + +- [Bash bang commands: A must-know trick for the Linux command line](https://www.redhat.com/sysadmin/bash-bang-commands) + diff --git a/docs/readline.md b/docs/readline.md index c7dbf33..9e981c4 100644 --- a/docs/readline.md +++ b/docs/readline.md @@ -99,199 +99,6 @@ Readline 提供快速移动光标的快捷键。 上面的`Alt`键也可以用 ESC 键代替。 -## 操作历史 - -### 基本用法 - -Bash 会保留用户的操作历史,即用户输入的每一条命令都会记录。有了操作历史以后,就可以使用方向键的`↑`和`↓`,快速浏览上一条和下一条命令。 - -退出当前 Shell 的时候,Bash 会将用户在当前 Shell 的操作历史写入`~/.bash_history`文件,该文件默认储存500个操作。 - -环境变量`HISTFILE`总是指向这个文件。 - -```bash -$ echo $HISTFILE -/home/me/.bash_history -``` - -`history`命令会输出这个文件的全部内容。用户可以看到最近执行过的所有命令,每条命令之前都有行号。越近的命令,排在越后面。 - -```bash -$ history -... -498 echo Goodbye -499 ls ~ -500 cd -``` - -输入命令时,按下`Ctrl + r`快捷键,就可以搜索操作历史,选择以前执行过的命令。这时键入命令的开头部分,Shell 就会自动在历史文件中,查询并显示最近一条匹配的结果,这时按下回车键,就会执行那条命令。 - -下面的方法可以快速执行以前执行过的命令。 - -```bash -$ echo Hello World -Hello World - -$ echo Goodbye -Goodbye - -$ !e -echo Goodbye -Goodbye -``` - -上面例子中,`!e`表示找出操作历史之中,最近的那一条以`e`开头的命令并执行。Bash 会先输出那一条命令`echo Goodbye`,然后直接执行。 - -同理,`!echo`也会执行最近一条以`echo`开头的命令。 - -```bash -$ !echo -echo Goodbye -Goodbye - -$ !echo H -echo Goodbye H -Goodbye H - -$ !echo H G -echo Goodbye H G -Goodbye H G -``` - -注意,`!string`语法只会匹配命令,不会匹配参数。所以`!echo H`不会执行`echo Hello World`,而是会执行`echo Goodbye`,并把参数`H`附加在这条命令之后。同理,`!echo H G`也是等同于`echo Goodbye`命令之后附加`H G`。 - -由于`!string`语法会扩展成以前执行过的命令,所以含有`!`的字符串放在双引号里面,必须非常小心,如果它后面有非空格的字符,就很有可能报错。 - -```bash -$ echo "I say:\"hello!\"" -bash: !\: event not found -``` - -上面的命令会报错,原因是感叹号后面是一个反斜杠,Bash 会尝试寻找,以前是否执行过反斜杠开头的命令,一旦找不到就会报错。解决方法就是在感叹号前面,也加上反斜杠。 - -```bash -$ echo "I say:\"hello\!\"" -I say:"hello\!" -``` - -### history 命令 - -前面说过,`history`命令能显示操作历史,即`.bash_history`文件的内容。 - -```bash -$ history -``` - -使用该命令,而不是直接读取`.bash_history`文件的好处是,它会在所有的操作前加上行号,最近的操作在最后面,行号最大。 - -通过定制环境变量`HISTTIMEFORMAT`,可以显示每个操作的时间。 - -```bash -$ export HISTTIMEFORMAT='%F %T ' -$ history -1 2013-06-09 10:40:12 cat /etc/issue -2 2013-06-09 10:40:12 clear -``` - -上面代码中,`%F`相当于`%Y - %m - %d`,`%T`相当于` %H : %M : %S`。 - -只要设置`HISTTIMEFORMAT`这个环境变量,就会在`.bash_history`文件保存命令的执行时间戳。如果不设置,就不会保存时间戳。 - -环境变量`HISTSIZE`设置保存历史操作的数量。 - -```bash -$ export HISTSIZE=10000 -``` - -上面命令设置保存过去10000条操作历史。 - -如果不希望保存本次操作的历史,可以设置`HISTSIZE`等于0。 - -```bash -export HISTSIZE=0 -``` - -如果`HISTSIZE=0`写入用户主目录的`~/.bashrc`文件,那么就不会保留该用户的操作历史。如果写入`/etc/profile`,整个系统都不会保留操作历史。 - -环境变量`HISTIGNORE`可以设置哪些命令不写入操作历史。 - -```bash -export HISTIGNORE='pwd:ls:exit' -``` - -上面示例设置,`pwd`、`ls`、`exit`这三个命令不写入操作历史。 - -如果想搜索某个以前执行的命令,可以配合`grep`命令搜索操作历史。 - -```bash -$ history | grep /usr/bin -``` - -上面命令返回`.bash_history`文件里面,那些包含`/usr/bin`的命令。 - -操作历史的每一条记录都有编号。知道了命令的编号以后,可以用`感叹号 + 编号`执行该命令。如果想要执行`.bash_history`里面的第8条命令,可以像下面这样操作。 - -```bash -$ !8 -``` - -`history`命令的`-c`参数可以清除操作历史。 - -```bash -$ history -c -``` - -### 相关快捷键 - -下面是一些与操作历史相关的快捷键。 - -- `Ctrl + p`:显示上一个命令,与向上箭头效果相同(previous)。 -- `Ctrl + n`:显示下一个命令,与向下箭头效果相同(next)。 -- `Alt + <`:显示第一个命令。 -- `Alt + >`:显示最后一个命令,即当前的命令。 -- `Ctrl + o`:执行历史文件里面的当前条目,并自动显示下一条命令。这对重复执行某个序列的命令很有帮助。 - -感叹号`!`的快捷键如下。 - -- `!!`:执行上一个命令。 -- `!n`:`n`为数字,执行历史文件里面行号为`n`的命令。 -- `!-n`:执行当前命令之前`n`条的命令。 -- `!string`:执行最近一个以指定字符串`string`开头的命令。 -- `!?string`:执行最近一条包含字符串`string`的命令。 -- `!$`:代表上一个命令的最后一个参数。 -- `!*`:代表上一个命令的所有参数,即除了命令以外的所有部分。 -- `^string1^string2`:执行最近一条包含`string1`的命令,将其替换成`string2`。 - -下面是`!$`和`!*`的例子。 - -```bash -$ cp a.txt b.txt -$ echo !$ -b.txt - -$ cp a.txt b.txt -$ echo !* -a.txt b.txt -``` - -上面示例中,`!$`代表上一个命令的最后一个参数(`b.txt`),`!*`代表上一个命令的所有参数(`a.txt b.txt`)。 - -下面是`^string1^string2`的例子。 - -```bash -$ rm /var/log/httpd/error.log -$ ^error^access -rm /var/log/httpd/access.log -``` - -上面示例中,`^error^access`将最近一条含有`error`的命令里面的`error`,替换成`access`。 - -如果希望确定是什么命令,然后再执行,可以打开`histverify`选项。这样的话,使用`!`快捷键所产生的命令,会先打印出来,等到用户按下回车键后再执行。 - -```bash -$ shopt -s histverify -``` - ## 其他快捷键 - `Ctrl + j`:等同于回车键(LINEFEED)。