【Linux 基础入门】(九)命令执行顺序控制与管道

1 介绍

内容

  • 顺序执行、选择执行、管道、cut 命令、grep 命令、wc 命令、sort 命令等,高效率使用 Linux 的技巧。

知识点

  • cut,grep,wc,sort 命令的使用
  • 管道的理解

2 命令执行顺序的控制

通常情况下,每次只能在终端输入一条命令,按下回车执行,执行完成后,再输入第二条命令,然后再按回车执行……,当有时候会一次输入多条命令,这个时候的执行过程又是如何的呢?下面我们将为大家详细讲解下命令的执行顺序的控制问题。

2.1 顺序执行多条命令

当需要使用 apt-get 安装一个软件,然后安装完成后立即运行安装的软件(或命令工具),又恰巧主机才更换的软件源还没有更新软件列表,那么可能会有如下一系列操作:

$ sudo apt-get update
# 等待——————————然后输入下面的命令
$ sudo apt-get install some-tool # 这里some-tool是指具体的软件包
# 等待——————————然后输入下面的命令
$ some-tool

如果可以一次性输入完,让它自己去依次执行各命令就好了,简单的顺序执行你可以使用 ; 来完成,比如上述操作你可以

$ sudo apt-get update;sudo apt-get install some-tool;some-tool
# 让它自己运行

2.2 有选择的执行命令

关于上面的操作,如果自动顺序执行命令时,前面的命令执行不成功,而后面的命令又依赖于上一条命令的结果,那么就会造成花了时间,最终却得到一个错误的结果,而且有时候直观的看无法判断结果是否正确。那么我们需要能够有选择性的来执行命令,比如上一条命令执行成功才继续下一条,或者不成功又该做出其它什么处理,比如我们使用 which 来查找是否安装某个命令,如果找到就执行该命令,否则什么也不做(虽然这个操作没有什么实际意义,但可以更好的理解一些概念):

$ which cowsay>/dev/null && cowsay -f head-in ohch~

你如果没有安装 cowsay,先执行上述命令会发现什么也没发生,安装好之后你再执行一次上述命令,你也会发现一些惊喜。

上面的 && 就是用来实现选择性执行的,它表示如果前面的命令执行结果(不是表示终端输出的内容,而是表示命令执行状态的结果)返回 0 则执行后面的,否则不执行,你可以从 $? 环境变量获取上一次命令的返回结果
在这里插入图片描述

shell 中 ||&& 控制效果相反,表示上一条命令执行结果为 ≠0($?≠0)时则执行它后面的命令。还可以结合着 &&|| 来实现一些操作,比如:

$ which cowsay>/dev/null && echo "exist" || echo "not exist"

在这里插入图片描述
用流程图来解释一下上面的流程

在这里插入图片描述

将上述顺序反过来效果是一样的

$ which cowsay>/dev/null || echo "not exist" && echo "exist"

在这里插入图片描述

3 管道

管道是一种通信机制,通常用于进程间的通信(也可通过 socket 进行网络通信),它表现出来的形式就是将前面每一个进程的输出(stdout)直接作为下一个进程的输入(stdin)。

管道又分为匿名管道和具名管道。我们在使用一些过滤程序时经常会用到的就是匿名管道,在命令行中由 | 分隔符表示。具名管道简单的说就是有名字的管道,通常只会在源程序中用到具名管道。下面通过一些使用管道的"过滤程序"熟练管道的使用。

3.1 试用

查看 /etc 目录下有哪些文件和目录,使用 ls 命令来查看:

$ ls -al /etc

有太多内容,屏幕不能完全显示,可以使用滚动条或快捷键滚动窗口来查看。也可以使用管道:

$ ls -al /etc | less

3.2 cut 命令,打印每一行的某一字段

cut 是一个选取命令,选取文件中的指定字符,主要有如下用途

  • 文件内容查看
  • 显示行中的指定部分,删除文件中指定字段
  • 显示文件的指定内容

-b:仅显示行中指定直接范围的内容;
-c:仅显示行中指定范围的字符;
-d:指定字段的分隔符,默认的字段分隔符为“TAB”;
-f:显示指定字段的内容;
-n:与“-b”选项连用,不分割多字节字符;
–complement:补足被选择的字节、字符或字段;

example:

  1. 有一个学生报表信息,包含No、Name、Mark、Percent
$ cat test.txt
No Name Mark Percent
01 tom 69 91
02 jack 71 87
03 alex 68 98

使用 -f 选项提取指定字段

$ cut -d ' ' -f 1 test.txt
No
01
02
03
$ cut -d ' ' -f 1,2 test.txt
No Name
01 tom
02 jack
03 alex

–complement 选项提取指定字段之外的列(打印除了第二列之外的列)

$ cut -d ' ' -f 2 --complement test.txt
No Mark Percent
01 69 91
02 71 87
03 68 98
  1. 使用 -d 选项指定字段分隔符
$ cat test2.txt 
No;Name;Mark;Percent
01;tom;69;91
02;jack;71;87
03;alex;68;98
$ cut -f 2 -d ';' test2.txt 
Name
tom
jack
alex
  1. 指定字段的字符或者字节范围

cut命令可以将一串字符作为列来显示,字符字段的记法
N-:从第N个字节、字符、字段到结尾
N-M:从第N个字节、字符、字段到第M个(包括M在内)字节、字符、字段
-M:从第1个字节、字符、字段到第M个(包括M在内)字节、字符、字段

上面是记法,结合下面选项将某个范围的字节、字符指定为字段:
-b 表示字节
-c 表示字符
-f 表示定义字段

打印第 1 个到第 3 个字符:

$ cat test3.txt 
abcdefghijklmnopqrstuvwxyz
abcdefghijklmnopqrstuvwxyz
abcdefghijklmnopqrstuvwxyz
abcdefghijklmnopqrstuvwxyz

# 打印第 1 个到第 3 个字符
$ cut -c 1-3 test3.txt
abc
abc
abc
abc

# 打印前 2 个字符
$ cut -c -2 test3.txt

# 打印从第 5 个字符开始到结尾
$ cut -c 5- test3.txt

3.3 grep 命令,在文本中或 stdin 中查找匹配字符串

grep 命令很强大,结合正则表达式可以实现很复杂却很高效的匹配和查找,不过在学习正则表达式之前,这里介绍它简单的使用,而关于正则表达式后面将会有单独一小节介绍到时会再继续学习 grep 命令和其他一些命令。

grep 命令的一般形式为:

grep [命令选项]... 用于匹配的表达式 [文件]...

先体验一下,搜索/home/skx 目录下所有包含"skx"的文本文件,并显示出现在文本中的行号:

$ grep -rnI "skx" ~

-r 参数表示递归搜索子目录中的文件,-n 表示打印匹配项行号,-I 表示忽略二进制文件。这个操作实际没有多大意义,但可以感受到 grep 命令的强大与实用。

当然也可以在匹配字段中使用正则表达式,下面简单的演示:

# 查看环境变量中包含"skx"的字符串
$ export | grep ".*skx"

在这里插入图片描述

3.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

在这里插入图片描述
注意:对于西文字符来说,一个字符就是一个字节,但对于中文字符一个汉字是大于 2 个字节的,具体数目是由字符编码决定的

再来结合管道来操作一下,下面统计 /etc 下面所有目录数

$ ls -dl /etc/*/ | wc -l

统计 /etc 下面所有文件数(目录加文件)

$ ls -dl /etc/* | wc -l

3.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

3.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
# 或者
$ history | cut -c 8- | cut -d ' ' -f 1 | sort -u
  • 输出重复行
# 输出重复过的行(重复的只输出一个)及重复次数
$ history | cut -c 8- | cut -d ' ' -f 1 | sort | uniq -dc
# 输出所有重复的行
$ history | cut -c 8- | cut -d ' ' -f 1 | sort | uniq -D
  • 3
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
3.3 简述先行控制的基本思想。 答:先行控制是缓冲技术和预处理技术相结合的产物。缓冲技术是在工作速度不固定的两个功能部件之间设置缓冲器,用以平滑它们的工作。预处理技术是指预取指令、对指令进行加工以及预取操作数等。先行控制实现了多条指令的重叠并行执行,提高了处理及部件的利用率,大大加快了处理机的执行速度。 3.4 设一条指令的执行过程分为取指令、分析指令和执行指令3个阶段,每个阶段所需的时间分别为t、t和2t。分别求出下列情况下,连续执行N条指令所需的时间。 (1)顺序执行方式。 (2)只有“取指令”和“执行指令”重叠。 (3)“取指令”、“分析指令”与“执行指令”重叠。 解:(1)顺序执行方式 每条指令需要t + t + 2t = 4t,N条指令需要4Nt。 取指令i 分析指令i 执行指令i 取指令j 分析指令j 执行指令j (2)只有“取指令”和“执行指令”重叠。 N条指令需要4t + (N-1)3t = (3N+1)t (3)“取指令”、“分析指令”与“执行指令”重叠。 N条指令需要4t + (N-1)2t = (2N+2)t 3.5 简述流水线技术的特点。 答:(1)流水过程由多个相联系的子过程组成,每个过程称为流水线的“级”或“段”。 (2)每个子过程由专用的功能段实现。 (3)各个功能段所需时间应尽量相等。 (4)流水线需要有“通过时间”,在此之后流水过程才进入稳定工作状态,每一个时钟周期(拍)流出一个结果。 (5)流水技术适合于大量重复的时序过程,只有在输入端能连续地提供任务,流水线的效率才能充分发挥。 3.6 解决流水线瓶颈问题有哪两种常用方法? 答:(1) 细分瓶颈段; (2) 重复设置瓶颈段。 3.7 减少分支延迟的静态方法有哪些? 答:(1) 预测分支失败的方法。 (2) 预测分支成功的方法。 (3) 延迟分支方法。 3.8 简述延迟分支方法中的3种调度策略的优缺点。 答: 调度策略 优点 缺点 从前调度 总是可以有效提高流水线性能 分支必须不依赖于被调度的指令 从目标处调度 分支转移成功时,可以提高流水线性能。 由于要复制指令,可能加大程序空间;如果分支转移失败,必须保证被调度的指令对程序的执行没有影响,可能需要复制被调度指令 从失败处调度 分支转移失败时,可以提高流水线性能 如果分支转移成功,必须保证被调度的指令对程序的执行没有影响 3.9 列举出下面循环中的所有相关,包括输出相关、反相关、真相关和循环相关。 for (i=2; i=100; i=i+1) a[i]=b[i]+a[i]; c[i+1]=a[i]+d[i]; a[i-1]=2b[i]; b[i+1]=2b[i]; 解:展开循环两次: a[i]=b[i]+a[i]; /*s1*/ c[i+1]=a[i]+d[i]; /*s2*/ a[i-1]=2*b[i]; /*s3*/ b[i+1]=2*b[i]; /*s4*/ a[i+1]=b[i+1]+a[i+1]; /*s1’*/ c[i+2]=a[i+1]+d[i+1]; /*s2’*/ a[i]=2*b[i+1]; /*s3’*/ b[i+2]=2*b[i+1]; /*s4’*/ 输出相关:无 反相关:无 真相关:s1与s2;(写后读) 由于循环引入的相关:s1与s3’(输出相关,反相关)、s4与s4’(真相关)、s4与s3’(真相关)、s4与s1’(真相关)、s2与s3’(反相关) 比如当i=2和3时,循环展开如下所示: a[2]=b[2]+a[2]; /*s1*/ c[3]=a[2]+d[2]; /*s2*/ a[1]=2*b[2]; /*s3*/ b[3]=2*b[2]; /*s4*/ a[3]=b[3]+a[3]; c[4]=a[3]+d[3]; a[2]=2*b[3]; b[4]=2*b[3];

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值