1. 命令执行顺序的控制
1.1 顺序执行多条命令
# sudo apt-get update;sudo apt-get install some-tool;some-tool
1.2 有选择的执行命令
如果我们按照上面的命令执行时,前面的命令执行不成功,而后面的命令又依赖于上一条命令的结果,那么就会造成花了时间,最终却得到一个错误的结果,而且有时候直观的看你还无法判断结果是否正确。那么我们需要能够有选择性的来执行命令,比如上一条命令执行成功才继续下一条,或者不成功又该做出其它什么处理,比如我们使用which来查找是否安装某个命令,如果找到就执行该命令,否则什么也不做(虽然这个操作没有什么实际意义,但可帮你更好的理解一些概念)
which cowsay>/dev/null && cowsay -f head-in ohch~
你如果没有安装cowsay,你可以先执行一次上述命令,你会发现什么也没发生,你再安装好之后你再执行一次上述命令,你也会发现一些惊喜。
&&就是用来实现选择性执行的,它表示如果前面的命令执行结果(不是表示终端输出的内容,而是表示命令执行状态的结果)返回0(执行成功)则执行后面的,否则不执行,你可以从$?环境变量获取上一次命令的返回结果
|| 在这里就是与&&相反的控制效果,当上一条命令执行结果为≠0($?≠0)时则执行它后面的命令。
2. 管道
管道是一种通信机制,通常用于进程间的通信(也可以用socket进行网络通信),它的表现形式就是将前面每一个进程的输出stdout直接作为下一个进程的输入stdin。
管道分为匿名管道和具名管道,我们在使用一些过滤程序时经常会用到的就是匿名管道,在命令行中由 | 分隔符表示。具名管道简单的说就是有名字的管道,通常只会在源程序中用到具名管道。
2.1 简单使用
ls -al /etc | less
#通过管道将前一个命令(ls)的输出作为下一个命令(less)的输入,然后就可以一行一行地看。
2.2 cut命令,打印每一行的某一字段
打印/etc/passwd文件中以:为分隔符的第1个字段和第6个字段分别表示用户名和其家目录:
cut /etc/passwd -d ':' -f 1,6
打印/etc/passwd文件的每一行的前n个字符:
#前五个(包含第五个)
cut /etc/passwd -c -5
# 前五个之后的(包含第五个)
cut /etc/passwd -c 5-
# 第五个
cut /etc/passwd -c 5
# 2到5之间的(包含第五个)
cut /etc/passwd -c 2-5
2.3 grep命令,在文本中或stdin中查找匹配字符串
grep命令是很强大的,也是相当常用的一个命令,它结合正则表达式可以实现很复杂却很高效的匹配和查找.
grep命令的一般形式为:
grep [命令选项]... 用于匹配的表达式 [文件]...
例如在/home/name目录下寻找所有包含“file”的文本文件,并显示出现在文本中的行号
grep -rnI "file" ~
# -r 参数表示递归搜索子目录中的文件,-n表示打印匹配项行号,-I表示忽略二进制文件。这个操作实际没有多大意义,但可以感受到grep命令的强大与实用。
2.4 wc命令,简单的计数工具
wc命令用于统计并输出一个文件中行、单词和字节的数目。
# 输出/etc/passwd文件的统计信息
wc /etc/passwd
# 行数
wc -l /etc/passwd
# 单词数
wc -w /etc/passwd
# 字节数
wc -c /etc/passwd
# 字符数
wc -m /etc/passwd
# 最长行字节数
wc -L /etc/passwd
结合管道操作一下
#统计 /etc 下面所有目录数
ls -dl /etc/*/ | wc -l
2.5 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
2.6 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 -dc
# 输出所有重复的行
$ history | cut -c 8- | cut -d ' ' -f 1 | sort | uniq -D
2.7 tr命令
tr命令可以用来删除一段文本信息中的某些文字,或者将其转换。
使用方式:
tr [option]...SET1 [SET2]
常用的选项有:
举例:
# 删除 "hello world" 中所有的'o','l'
echo 'hello world' | tr -d 'ol'
# 将"hello" 中的ll,去重为一个l
echo 'hello' | tr -s 'l'
# 将输入文本,全部转换为大写或小写输出
echo 'input some text here' | tr '[:lower:]' '[:upper:]'
# 上面的'[:lower:]' '[:upper:]'你也可以简单的写作'[a-z]' '[A-Z]',当然反过来将大写变小写也是可以的
2.8 col命令
col 命令可以将Tab换成对等数量的空格键,或反转这个操作。
使用方式:
col [option]
常用的选项有:
举例:
# 查看 /etc/protocols 中的不可见字符,可以看到很多 ^I ,这其实就是 Tab 转义成可见字符的符号
cat -A /etc/protocols
# 使用 col -x 将 /etc/protocols 中的 Tab 转换为空格,然后再使用 cat 查看,你发现 ^I 不见了
cat /etc/protocols | col -x | cat -A
2.9 join命令
用于将两个文件中包含相同内容的那一行合并在一起
使用方式:
join [option]... file1 file2
常用选项:
操作举例:
# 创建两个文件
$ echo '1 hello' > file1
$ echo '1 world' > file2
$ join file1 file2
# 将/etc/passwd与/etc/shadow两个文件合并,指定以':'作为分隔符
$ sudo join -t':' /etc/passwd /etc/shadow
# 将/etc/passwd与/etc/group两个文件合并,指定以':'作为分隔符, 分别比对第4和第3个字段
$ sudo join -t':' -1 4 /etc/passwd -2 3 /etc/group
3.0 paste命令
与join命令类似,它是在不对比数据的情况下,简单的将多个文件合在一起,以Tab隔开
使用方式:
paste [option] file...
常用选项有:
操作举例:
echo hello > file1
echo world > file2
echo www.helloworld.com > file3
paste -d ':' file1 file2 file3
paste -s file1 file2 file3