docs(history): add history chapter

This commit is contained in:
ruanyf
2023-12-14 14:34:05 +08:00
parent 8eccc162d3
commit dea82f66b3
3 changed files with 326 additions and 193 deletions

View File

@@ -5,6 +5,7 @@
- variable.md: 变量
- string.md: 字符串操作
- arithmetic.md: 算术运算
- history.md: 操作历史
- readline.md: 行操作
- stack.md: 目录堆栈
- script.md: 脚本入门

325
docs/history.md Normal file
View File

@@ -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)

View File

@@ -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