本希望把Linux高效操作这一个系列文章命名为unix-like命令高效操作,但是后来发现本人并没有在多数的unix-like平台实践过。本着实践第一的精神,就不把话题扯那么大了,决定更名为“linux命令高效操作”,哈哈。废话不多说,我的实验系统:Linux之centos7。


今天我想聊的是每个用过Linux命令操作的同学或许都知道的一个存在:历史命令。什么是历史命令呢?通俗简单的说,就是你执行过的命令的历史。大家最常见的用处就是使用上下方向键进行命令的切换喽!刚接触命令行操作的同学感觉这就是福祉啊,但是用的久了,会觉得每次上下键翻阅查看太烦人。如果历史命令支持搜索该有多棒。还有些命令我之前明明执行过了为什么找不到呢?甚至还有很多其他更高的要求。


可以告诉你,这些要求都可以做到,甚至可以做的更牛逼。


开始之前,首先按如下顺序输入一些命令

$ cd /home/poloxue/Public/

$ mkdir Work

$ cd Work

$ mkdir web

$ mkdir app

$ mkdir logs

$ cd logs

$ cp /var/log/open***as.log .


执行结束之后。


首先学习体验一下历命令史的查看,增加和删除等基本操作。

一 如何查看历史命令

1. 简单查阅

首先明确一点,历史命令是按照执行时间先后顺序进行排序的。

简单查看:上下键查阅历史命令;

除了上下键操作查看历史命令之外,在上一篇文章也介绍了两个快捷键:

ctrl+p 上查历史命令,同向上键,prev;

ctrl+n 下查历史命令,同向下键,next;

这种方式查阅最近几条执行命令相对效率较高,使用频率最到。但是对于较早时间执行的命令查阅相当耗时;


2. 使用history命令查询

history命令是和历史命令息息相关的一个命令,作用对象就是历史命令。看一下history命令怎么查询最近使用的命令记录?

history [n] 展示最近执行的n条记录,例如:

$ history 3

按时间倒序展示最近历史命令,结果如下:

mkdir logs

cd logs

cp /var/log/open***as.log .

如果没有追加任何的参数,则展示全部历史记录;


3. 历史命令搜索

history命令本身好像不支持搜索,两种方式可以实现这个功能:ctrl+r快捷键和history+grep组合命令。

a. ctrl+r搜索查看功能:

假设你搜索最近关于cp的操作,如下:

按下ctrl+r快捷键之后输入cp,效果如上图,此时按下enter键选中执行。

ctrl+r搜索结果为最近执行的符合搜索要求的命令。

b. history+grep搜索查看功能:

为什么需要?ctrl+r只能搜索最近执行的命令,如果需要所有满足匹配的历史记录,或者不能完整记住有关单词的情况下,history+grep的组合可以满足你。

例如,搜索所有关于open***的操作。

$ history | grep open***

history列出有所历史记录,grep支持强大的搜索功能,支持通配符,正则等高级特性,可以基本满足你的任何需求;


二 记录历史命令(写入与删除

个人认为,历史命令存在于两个区域,一是每开启一个新终端都会加载的HISTFILE文件,二是当前会话终端开启后执行的命令,这些命令可以理解为存储在内存中,但尚未写入HOSTFILE文件中的部分。


1. 默认效果

当当前会话终端关闭后,会自动将此会话终端执行的历史命令追加到$HISTFILE文件中。


2. 手动写入

除了默认效果,需要关闭终端才能将当前终端的历史命令写入到$HISTFILE中,执行如下命令:

$ history -w

效果与关闭终端效果一样,会将当前终端执行的没有写入到$HISTFILE文件中的历史命令追加到$HISTFILE中。


3. 删除指定历史命令

$ history -d 1

删除指定历史命令位置的命令,history -d n。

如果历史记录如上所示,执行history -d 1会将mkdir public这条记录删除。


4. 清空历史命令

history -c 会清空当前会话终端中的所有历史命令,重新记录。

注意:此时只是改变了当前会话终端中的记录,并没有记录到$HISTFILE文件中。


三 哪些配置项

1. 关于HISTSIZE与HISTFILESIZE

HISTSIZE 定义了控制台输出命令记录数量;

HISTFILESIZE 定义了HISTFILE文件中存放命令记录的数量;

$ echo $HISTSIZE

$ echo $HISTFILESIZE

可以看到打印的数值,默认配置HISTSIZE与HISTFILESIZE大小一样,当历史命令超过规定数量会自动覆盖,为了更好的利用和防止历史命令,可以将HISTFILESIZE与HISTSIZE调整的大一些;

$ vim .bashrc

在/$HOME/.bashrc添加如下内容:

export HISTFILESIZE=3000

export HISTSIZE=3000


2. 关于HISTTIMEFORMAT显示命令执行时间戳

默认情况下,执行history会输出记录编号和历史命令。如果为了审查的目的,想知道某条命令具体的执行时间,配置HISTTIMEFORMAT。可以配置HISTTIMEFORMAT为'%F %T ' 

$ export HISTTIMEFORMAT='%F %T ' 

此时执行history命令,如下效果:

Linux命令高效操作(2)-妙用历史命令_第1张图片

从上图可以看出,除了显示记录编号与历史命令之外,还多了命令执行时间一列。如果此功能经常用到,可以将export HISTTIMEFORMAT='%F %T ' 添加到/$HOME/.bashrc文件中。


3. HISTCONTROL与HISTIGNORE

HISTCONTROL与HISTIGNORE可提高文件有限的利用率。通过上面的了解,我们知道历史命令的记录数量是有上限的。为了提高历史命令的利用,可以使用两种方式:一是增加历史命令的HISTFILESIZE和HISTSIZE,二是减少无用历史命令记录,增加固定大小HISTFILESIZE的空间利用率;

第一种方式就不做介绍,第二种方式就使用两个配置项:HISTCONTROL与 HISTIGNORE。

a. 去除连续重复命令

设置HISTCONTROL为ignoredups,如下:

$ export HISTCONTROL=ignoredups

效果如下:

Linux命令高效操作(2)-妙用历史命令_第2张图片

从上图可以看出,当HISTCONTROL为ignoredups,重复两次的pwd会合并为一条。但是从上图中可以看出,对于不连续的history仍然保留了两条,下面解决一下这个问题。

b. 去除全局重复历史命令

上面的示例提出了一个问题,HISTCONTROL为ignoredups并没有去除非连续的重复命令。下面将HISCONTROL设置为erasedups,如下:

$ export HISTCONTROL=erasedups

效果如下:

Linux命令高效操作(2)-妙用历史命令_第3张图片

从上图可以看出,虽然我执行了两次pwd,而且是非连续的,但是只保存了一条。这样就可以很大限度高效的使用历史命令有限的空间了,但是如果你有需求要保留每次的命令执行情况方便后面审查,那就不能干了。不过本人目前还没有遇到这样的情况,可能经验太少吧。

c. 空格隐去无用历史命令

有没有这样一种需求?有些命令我执行了,但是我不想让别人知道。实现这种需求有两种方式:一种是通过配置空格隐去无用历史命令,二是通过配置指定某些命令需要隐去。好,第二种放在下面一点说明,先说第一种。

首先,进行设置HISTCONTROL为ignorespace,如下所示:

$ export HISTCONTROL=ignorespace

作用是所执行命令以空格开头,不做历史记录。效果如下:

Linux命令高效操作(2)-妙用历史命令_第4张图片

上图可以看出,pwd和cd以空格开头,而history和ls都是正常输入,此时历史命令中只有history与ls的记录,没有记录pwd与cd。不过,个人感觉这种方式不是很好且具有不可控性,有可能产生误操作导致想要的历史命令丢失历史等情况。那有没有办法指定固定的某些命令被忽略,比如ls, history这些不想保留在历史中。继续往下看...

d. 指定忽略的命令

为了实现指定命令忽略,需要另外一个配置项:HISTIGNORE。如需要隐去ls, ls -l,pwd, history的执行记录,可以通过如下设置:

$ export HISTIGNORE="pwd:ls:ls -l:history"

效果如下:

Linux命令高效操作(2)-妙用历史命令_第5张图片


上图执行了history, pwd, ls -l, ls Public,通过历史命令查看发现只有ls Public/ 被记录了下来。突然觉得历史命令真的好强大,有效利用可以减少无用历史命令的存储。不过,具体省略哪些命令,要因人具体情况而定,也不要省略太多的命令。


4. 配置HISTFILE

通过配置HISTFILE可以改变记录历史命令的文件文件,默认/$HOME/.bashrc。这个就不多说了,我个人基本不会改变此文件位置。


四 利用历史命令快速编辑

上面说了那么多关于历史命令的内容,目的是为了其更高效的使用。最简单的使用场景就是通过上下手动查询的方式,这就不多说了。

下面看些更牛的用法,来真正提高你的命令操作效率:


1. 快速指定历史命令

快速选中指定命令,一种是使用 ! ,通过命令执行顺序编号或者通过命令开头选中,二可以通过ctrl+r搜索快速选中,前面已作过介绍,不多说。下面看看第一种情况,假设此刻历史命令如下:


直接示例说明:

$ !1                   // 执行命令为ls public

$ !-1                  // 执行命令为history

$ !!                    // 等同于!-1

$ !export          // 执行命令为export HISTCONTROL=erasedups

从上面可以看出,!的几种常见用法。

!n    指定编号命令

!-n   倒序n处命令

!!      !-1的简写,也就是最近的一条命令

!str   以str开头的最近一条命令,不过这种方式好像str中不能存在空格,可以使用ctrl+r快捷键代替。


2. 快速选中指定参数

通常在执行命令时,参数的复杂度都大于命令自身。如果能够实现快速选中命令的某个参数那就非常cool了。非常幸运,此要求的确可以做到。假设历史命令如下:

Linux命令高效操作(2)-妙用历史命令_第6张图片

示例说明:

a. 引用某命令的第一个参数

例如,引用第三条命令的第一个参数

$ ls -l !3:^

上述命令的效果相当于ls -l example01

例如,引用最新一条以export开头的命令的第一个参数

$ ls -l !export:^ 

效果相当于ls -l HISTCONTROL=erasedups

记住一个简写,最近一条命令的第一个参数!!:^,简写!^

b. 引用某条命令的最后一个参数

例如,引用第三条命令的最后一个参数

$ ls -l !3:$

效果相等于ls -l Public/

对于引用最新一条命令的最后一个参数同样也有!!:$和!$两种方式。

c. 引用命令任意位置的参数

除了引用像开头结尾这种特殊位置参数外,同样也支持任意位置参数的引用。

还是以上图中的第三条命令为例,现在我希望引用此命令的第二个参数:

$ ls -l !3:2

如上面所示命令,此时的!3:2为example02。以此类推,简单测试一下我们可以知道,!3:0为ls, !3:1为example01, !3:3为Public/。由此可以得出,分号之后的数字指定的就是命令参数的位置,从0计数。

d. 关于引用命令参数任意区间列表

除了指定某一个参数之外,选择参数时,我们还可以指定参数区间,如选择1-2位置的参数,开头至结尾处的参数等。格式:cmd:offset1-offset2。

以第三条历史命令为例,如下:

$ echo !3:^-$    // 打印全部参数列表,即example01 example02 Public/

$ echo !3:1-2    // 打印1至2位置参数,即example01 exmple02


从上面的示例,可以看出引用命令具体参数的格式通过分号分为两个部分,一是从哪个命令查找,二是查找具体哪个位置的参数。


3. 历史命令替换执行

历史命令替换就是用处不多,也只在有比较长的命令比较高效,格式如下:

cmd:offset1-offset2:s/old/new/:s/old/new/

例如,替换历史命令第3条命令参数example01位example1,example02为example2,Public为Work,如下:

$ echo !3:1-3:s/01/1/:s/02/2/:s/Public/Work/



虽然历史命令理解起来比较简单,但是真正用好且确实提高工作效率是一个不断实践总结的过程。部分功能是我们需要常用的功能,当然这里面也有些功能在平时工作中很少用到,但在有些特殊场景,确实可以提高我们工作效率。