顺序执行、选择执行、管道、cut 命令、grep 命令、wc 命令、sort 命令等,高效率使用 Linux 的技巧。
命令执行顺序控制
通常情况下,我们每次只能在终端输入一条命令,按下回车执行,执行完成后,我们再输入第二条命令,然后再按回车执行……,当有时候我们会一次输入多条命令,这个时候的执行过程又是如何的呢?下面我们将为大家详细讲解下命令的执行顺序的控制问题。
顺序执行多条命令
当我们需要使用apt-get安装一软件时,然后安装完成后立即运行安装软件,又凑巧你的主机才更换的软件源还没有更新软件表,那么可能有如下一系列操作:
sudo apt-get update
#等待更新
sudo apt-get install somen-tool #some-tool是指具体的软件包
#等待安装
some-tool
如上操作,可以使用一条命令执行完成,不需要多次输入,简单的顺序执行可以使用;
来完成,以上操作可以如下输入进行:
sudo apt-get update;sudo apt-get install some-tool;some-tool
有选择的执行命令
顺序执行多条命令时,可能前面的命令执行不成功,而后面的命令依赖上一条命令的结果,那么就会造成,花了很长的时间,却得到了一个错误的结果,而且有时候只管还无法判断结果是否正确,那么需要有选择性的执行命令。比如上一条执行成功才继续执行下一条,或者不成功又该做出什么样的选择。
使用which
来查看是否安装某个命令,如果找到命令,则执行该命令,否则什么也不做:
which cowsy>/dev/null && cowsay -f head-in ihch~
上面的&&
就是用来实现选择性执行的,它表示如果前面的命令执行结果(不是表示终端输出的内容,而是表示命令执行状态的结果)返回0则执行后面的,否则不执行,你可以从 ?≠0)时则执行后面的命令。
which cowsay>/dev/null || echo "cowsay has not been install, please run 'sudo apt-get install cowsay' to install'"
||
和&&
结合使用:
which cowsay>/dev/null && echo "exist" || echo "not exist"
管道
管道是一种通信机制,通常用户进程间的通信(也可通过socket进行网络通信),它表现出来的形式就是将前面每一个进程的输出(stdout)直接作为下一个进程的输入(stdin)。
管道又分为匿名管道和具名管道;在实际中使用一些过滤程序是经常会用到的就是匿名管道,在命令行中由|
分割符表示。具名管道简单的说就是有名字的管道,通常只会在源程序中用到具名管道。
管道例子
ls -al /etc
将/etc
下的文件全部打印出来,内容太多,屏幕不能完全显示,可以使用滚动条或快捷键来查看,不太友好,不过可以使用管道,一行一行查看:
ls -al /etc | less
通过管道将前一个命令ls
的输出作为下一个命令less
的输入,然后一行一行查看
cut命令:打印每一行的某一个字段
打印/etc/passwd
文件中以’:’为分隔符的第1个字段和第6个字段分布表示用户名和其目录:
cut /etc/passwd -d ':' -f 1,6
打印/etc/passwd
文件中指定的内容:
# 前五个(包含第五个)
$ cut /etc/passwd -c -5
# 前五个之后的(包含第五个)
$ cut /etc/passwd -c 5-
# 第五个
$ cut /etc/passwd -c 5
# 2到5之间的(包含第五个)
$ cut /etc/passwd -c 2-5
grep: 在文本中或stdin中查找匹配字符串
grep
命令非常强大,相当常用的一个命令,结合正则表达式可以实现非常复杂却很高效的匹配和查找。
grep
命令的一般格式:
grep 【命令选项】 ... 用户匹配的表达式 [文件]...
例子:查找/home/shiyanlou
目录下的包行“shiyanlou”文本的文件
grep -rnI "shiyanlou"
-r
参数表示递归搜索子目录中的文件,-n
表示打印匹配先行号,’-I’表示忽略二进制文件。
使用正则表达式的例子:
export | grep ".*yanlou$"
其中$
就表示一行的末尾,*
匹配前面的子表达式任意次,.
匹配除“\n”和”\r”之外的任何单个字符
wc命令:计数工具
wc
命令用户统计并输出一个文件中行、单词和字节的数目,比如输出/etcpasswd
文件中的统计信息:
wc /etc/passwd
输出行数、单词数、字节数:
分别只输出行数、单词数、字节数、字符数和输入文本中最长一行的字节数:
# 行数
$ wc -l /etc/passwd
# 单词数
$ wc -w /etc/passwd
# 字节数
$ wc -c /etc/passwd
# 字符数
$ wc -m /etc/passwd
# 最长行字节数
$ wc -L /etc/passwd
注意:对于西文字符来说,一个字符就是一个字节,但对于中文字符一个汉字是大于2个字节的,具体数目是由字符编码决定的
结合管道来操作,统计/etc
下面所有目录数:
ls -dl /etc/*/ | wc -l
sort排序命令
该命令功能很简单就是将输入按照一定方式排序,然后再输出,它支持的排序有按字典排序,数字排序,按月份排序,随机排序,反转排序,指定特定字段进行排序等等。
默认为字典排序:
cat /etc/passwd | sort
反转排序:
cat /etc/passwd | sort -r
按特定字段排序:
cat /etc/passwd | sort -t':' -k 3
上面的-t参数用于指定字段的分隔符,这里是以”:”作为分隔符;-k 字段号用于指定对哪一个字段进行排序。这里/etc/passwd文件的第三个字段为数字,默认情况下是以字典序排序的,如果要按照数字排序就要加上-n参数:
cat /etc/passwd | sort -t':' -k 3 -n
uniq去重复命令
uniq
命令可以用于过滤或者输出重复行。
过滤行
使用history命令查看最近执行过的命令(实际为读取${SHELL}_history文件,如我们环境中的~/.zsh_history文件),不过你可能只想查看使用了哪个命令而不需要知道具体干了什么,那么你可能就会要想去掉命令后面的参数然后去掉重复的命令:
history | cut -c 8- | cut -d ' ' -f 1 | uniq
然后经过层层过滤,你会发现确是只输出了执行的命令那一列,不过去重效果好像不明显,仔细看你会发现它确实去重了,只是不那么明显,之所以不明显是因为uniq命令只能去连续重复的行,不是全文去重,所以要达到预期效果,我们先排序:
history | cut -c 8- | cut -d ' ' -f 1 | sort | uniq
# 或者 history | cut -c 8- | cut -d ' ' -f 1 | sort -u
这就是 Linux/UNIX 哲学吸引人的地方,大繁至简,一个命令只干一件事却能干到最好。
输出重复行
# 输出重复过的行(重复的只输出一个)及重复次数
$ history | cut -c 8- | cut -d ' ' -f 1 | sort | uniq -dc
# 输出所有重复的行
$ history | cut -c 8- | cut -d ' ' -f 1 | sort | uniq -D