Shell

输入相似文件名太麻烦

用花括号括起来的字符串用逗号连接,可以自动扩展,非常有用,直接看例子:

  1. $ echo {one,two,three}file
  2. onefile twofile threefile
  3. $ echo {one,two,three}{1,2,3}
  4. one1 one2 one3 two1 two2 two3 three1 three2 three3

花括号中的每个字符都可以和之后(或之前)的字符串进行组合拼接,注意花括号和其中的逗号不可以用空格分隔,否则会被认为是普通的字符串对待
这个技巧的实际用处最简单实用的就是给cp,mv,rm等命令扩展参数:

  1. $ cp /very/long/path/file{,.bak}
  2. # 给 file 复制一个叫做 file.bak 的副本
  3. $ rm file{1,3,5}.txt
  4. # 删除 file1.txt file3.txt file5.txt
  5. $ mv *.{c,cpp} src/
  6. # 将所有 .c 和 .cpp 为后缀的文件移入 src 文件夹

输入路径名称太麻烦

cd -返回刚才的目录

  1. $ pwd
  2. /very/long/path
  3. $ cd # 回到家目录
  4. $ pwd
  5. /home/labuladong
  6. $ cd - # 再返回刚才的目录
  7. $ pwd
  8. /very/long/path

特殊命令!$会替换成上一次命令最后的路径,直接看例子:

  1. # 没有加可执行权限
  2. $ /usr/bin/script.sh
  3. zsh: permission denied: /usr/bin/script.sh
  4. $ chmod +x !$
  5. chmod +x /usr/bin/script.sh

特殊命令!*会替换成上一次命令输入的所有文件路径,直接看例子:

  1. # 创建了三个脚本文件
  2. $ file script1.sh script2.sh script3.sh
  3. # 给它们全部加上可执行权限
  4. $ chmod +x !*
  5. chmod +x script1.sh script2.sh script3.sh

可以在环境变量CDPATH中加入常用的工作目录,当cd命令在当前目录中找不到指定的文件/目录时,会自动到CDPATH中的目录中寻找。
比如说常去家目录,也常去/var/log目录找日志,可以执行如下命令:

  1. $ export CDPATH='~:/var/log'
  2. # cd 命令将会在 ~ 目录和 /var/log 目录扩展搜索
  3. $ pwd
  4. /home/labuladong/musics
  5. $ cd mysql
  6. cd /var/log/mysql
  7. $ pwd
  8. /var/log/mysql
  9. $ cd my_pictures
  10. cd /home/labuladong/my_pictures

这个技巧是十分好用的,这样就免了经常写完整的路径名称,节约不少时间。
需要注意的是,以上操作是 bash 支持的,其他主流 shell 解释器当然都支持扩展cd命令的搜索目录,但可能不是修改CDPATH这个变量,具体的设置方法可以自行搜索。

输入重复命令太麻烦

使用特殊命令!!,可以自动替换成上一次使用的命令

  1. $ apt install net-tools
  2. E: Could not open lock file - open (13: Permission denied)
  3. $ sudo !!
  4. sudo apt install net-tools
  5. [sudo] password for labuladong:

有的命令很长,一时间想不起来具体参数了怎么办?
对于 bash 终端,可以使用Ctrl+R快捷键反向搜索历史命令,之所以说是反向搜索,就是搜索最近一次输入的命令。
比如按下Ctrl+R之后,输入sudo,bash 就会搜索出最近一次包含sudo的命令,回车之后就可以运行该命令了:

(reverse-i-search)`sudo': sudo apt install git

但是这个方法有缺点:首先,该功能似乎只有 bash 支持,用 zsh 作为 shell 终端,不起作用;第二,只能查找出一个(最近的)命令,如果想找以前的某个命令,就没办法了。
对于这种情况,最常用的方法是使用history命令配合管道符和grep命令来寻找某个历史命令

# 过滤出所有包含 config 字段的历史命令
$ history | grep 'config'
 7352  ./configure
 7434  git config --global --unset https.proxy
 9609  ifconfig
 9985  clip -o | sed -z 's/\n/,\n/g' | clip
10433  cd ~/.config

使用的所有 shell 命令都会被记录,前面的数字就表示这是第几个命令,找到想重复使用的命令后,也不需要复制粘贴该命令,只要使用!+ 想重用的命令编号即可运行该命令
拿上面的例子,想重新运行git config那条命令,就可以这样:

$ !7434
git config --global --unset https.proxy
# 运行完成

觉得history加管道加grep这样打的字还是太多,可以在 shell 配置文件中(.bashrc.zshrc等) 中写这样一个函数:

his()
{
    history | grep "$@"
}

这样就不需要写那么多,只需要his 'some_keyword'即可搜索历史命令。
一般不使用 bash 作为终端,给大家推荐一款很好用的 shell 终端叫做 zsh,。这款终端还可以扩展各种插件,非常好用,具体配置方法可自行搜索。

其他小技巧

1、yes命令自动输入字符y进行确认:

安装某些软件的时候,可能有交互式的提问:

$ sudo apt install XXX
...
XXX will use 996 MB disk space, continue? [y/n]

一般情况下都是一路 y 到底,但如果想自动化一些软件的安装就很烦,遇到这种交互式提问就卡住了,还得手动处理。
yes命令可以帮助我们:

$ yes | your_cmd

这样就会一路自动y下去,不会停下让我们输入了。
单独运行一下yes命令,发现它就是打印出一大堆字符 y,通过管道把输出和your_cmd的标准输入相连接,如果your_cmd又提出无聊的问题,就会从标准输入读取数据,也就会读取到一个 y 和换行符,和手动输入 y 确认是一个效果。

2、特殊变量$?记录上一次命令的返回值。

在 Linux shell 中,遵循 C 语言的习惯,返回值为 0 的话就是程序正常退出,非 0 值就是异常退出。读取上一次命令的返回值在平时使用命令行时感觉没什么用,但是如果想编写一些 shell 脚本,知道返回值非常有用。
举个实际的例子,比如 Github 仓库 fcant ,需要给其中所有 markdown 文件最下方添加上一篇、下一篇、目录三个页脚链接,有的文章已经有了页脚,大部分都没有。
为了防止重复添加,需要知道一个 md 文件末尾是否已添加,这时候就可以使用$?变量配合grep命令做到:

#!/bin/bash
filename=$1
# 查看文件尾部是否包含关键词
tail | grep '下一篇' $filename
# grep 查找到匹配会返回 0,找不到则返回非 0 值
[ $? -ne 0 ] && { 添加页脚; }

3、特殊变量$$记录当前进程的 PID。

这个功能可能在平时使用时也不怎么用,但是在写 shell 脚本时也非常有用,比如说要在/tmp创建临时文件,给文件起名字一直都是非常让人费脑子的,这时候可以使用$$变量扩展出当前进程的 PID 作为临时文件名,PID 在计算机中都是唯一的,所以绝不会重复,也不需要记住临时文件的名字。