awk从放弃到入门(11):拾遗之”三元运算”与”打印奇偶行”

  • A+
所属分类:Linux基础

在本博客中,AWK是一个系列文章,本人会尽量以通俗易懂的方式递进的总结awk命令的相关知识点。

awk系列博文直达链接:AWK命令总结之从放弃到入门(通俗易懂,快进来看)

 

 

这篇文章中的知识点是建立在前文的基础上的,如果你还没有掌握前文中的知识,请先参考之前的文章。

 

这篇文章其实是对之前知识点的拾遗,有些遗漏的常用知识点会在这篇文章中介绍。

三元运算

还记的我们在学习awk的 "if..else" 结构时,举的例子吗?我们来回顾一下。

在centos6中,我们可以判断用户的UID是否小于500,如果用户的UID大于500,则用户为普通用户,如果用户的UID小于500,则用户为系统用户。

所以,我们可以通过awk的 "if...else结构",判断用户的UID范围,从而判断出用户属于哪种用户类型,示例如下

awk从放弃到入门(11):拾遗之

正如上图所示,我们使用"if...else"结构,对usertype变量进行了赋值,如果用户的UID小于500,则对usertype变量赋值为"系统用户",否则则赋值usertype变量为"普通用户",最后打印出用户名所在的列与usertype变量的值。

其实,我们可以使用三元运算,替换上例中的"if...else"结构语句,示例如下

awk从放弃到入门(11):拾遗之

正如上图所示,红线标注部分则使用了三元运算的语法,代替了之前"if...else"的语法,而三元运算的语法如下:

条件 ? 结果1 : 结果2

上述语法表示,如果条件成立,则返回结果1,如果条件不成立,则返回结果2。

而上例中,"$3<500"就是上述语法中的"条件","系统用户"就是上述语法中"?"后面的"结果1","普通用户"就是上述语法中":"后面的"结果2"  ,同时,在上例中我们使用usertype变量接收了三元运算后的返回值,所以,当条件成立时,usertype变量被赋值为"系统用户",当条件不成立时,usertype变量被赋值为"普通用户"。

 

是不是很方便?其实,三元运算还有另外一种使用方式,示例如下

awk从放弃到入门(11):拾遗之

我们通过上述命令,统计出了,系统用户有42个,普通用户有7个,上图中红线标注的用法可以理解为三元运算的另一种语法。如下

表达式1 ? 表达式2 : 表达式3

上述语法表示,如果表达式1为真,则执行表达式2,如果表达式1为假,则执行表达式3

而上例中,"$3<500"即为表达式1,"a++"即为表达式2,"b++"即为表达式3

也就是说,当每遇到一个UID小于500的用户,我就对变量a加1,否则我就对变量b加1,从而算出了系统用户与普通用户的数量,最后再END模式中输出了变量a与变量b的值。

是不是很容易理解?你一定已经明白了。

 

打印奇偶行

如果我们想要使用awk打印文本中的奇数行或者偶数行,则是非常简单的。

 

我们先来看看怎样使用awk打印奇数行或偶数行,然后再结合示例解释原理,所以看不懂没关系,后面会有解释。

awk从放弃到入门(11):拾遗之

正如上图所示,test12文件中有11行文本,我们可以使用非常简洁的awk命令,打印出了奇数行或者偶数行。

 

但是如果我们想要彻底搞明白原理,则需要搞明白如下两个知识点(后面会有更详细的解释)

1、在awk中,如果省略了模式对应的动作,当前行满足模式时,默认动作为打印整行,即{print $0}。

2、在awk中,0或者空字符串表示"假",非0值或者非空字符串表示"真"

 

上述两个知识点是什么意思呢?我们慢慢聊。

在之前介绍awk模式的文章中提及过,模式可以理解为条件,如果当前行能与模式匹配,则会执行对应的动作。示例如下

awk从放弃到入门(11):拾遗之

上图中的两个命令均使用到了模式

第一个命令表示如果当前行中包含字符"1",则执行对应的动作,而对应的动作就是打印整行。

第二个命令表示如果test12文本中文本行的第二列的值如果大于10,则执行对应的动作,而对应的动作就是打印整行。

 

那么,如果我们将上例中awk命令中的动作都省略,会出现什么情况呢?我们来试试。

awk从放弃到入门(11):拾遗之

我们发现,当使用了模式时,如果省略了对应的动作,会默认的输出整行。

也就是说,当使用了模式时,如果省略了模式对应的动作,默认动作为"{print $0}"

当然,"空模式"与"BEGIN/END模式"除外。

 

这就是第1个知识点的含义,我想你应该明白了,那么我们来聊聊第2个知识点。

在awk中,0或者空字符串表示"假",非0值或者非空字符串表示"真",什么意思呢?我们还是可以从模式说起,"模式"可以理解为"条件",当条件成立,则为真,当条件不成立,则为假,所以,当模式为真时,则会执行对应的动作,当模式为假时,则不会执行对应的动作。

那么,我们能不能直接把模式替换为"真"或者"假"呢?我们来试试。

awk从放弃到入门(11):拾遗之

上例中,命令1使用了"空模式",也就是说,每一行都满足模式,每一行经过"空模式"匹配以后结果都是"真",所以每一行都会执行对应的动作。

命令2中,原来"模式的位置"被替换为了数字"1",我们可以把数字"1"理解成一种模式匹配后的结果,而1是非零值,刚才说过,在awk中非零值表示真,所以,"1"表示"真",  换句话说就是模式的匹配结果为真,模式成立则会执行对应的动作,而命令2中,对应的动作为打印整行。

命令3 与 命令2 同理,在命令3中,  数字"2"为非零值,表示真,可以理解为:模式的匹配结果为真,则会执行对应的动作,聪明如你一定想到了,数值"2"可以换做任何非0值或者非空字符串。

命令4中,数字"2"为非零值,表示模式为真,而之前说过,当使用模式时,可以省略动作,当使用模式并省略动作时,默认动作为打印整行,所以,命令4表示打印所有行,因为每一行的模式都为真。

命令5与命令6同理,在awk中,数字"0"与空字符串表示假,当模式为假时,不会执行对应的动作,而当存在模式并省略动作时,默认动作为打印整行,但是由于模式为假,所以对应的动作并未执行。

 

其实,我们还能对真与假进行取反,非真即为假,非假即为真,示例如下。

awk从放弃到入门(11):拾遗之

正如上图所示,我们使用"!"对真假进行取反。

 

如果你已经看懂了上面的例子,那么,我们再来延伸一下。

你猜猜,如下示例会输出什么?

awk从放弃到入门(11):拾遗之

没错,聪明如你一定想到了,上例中,其实是使用了awk的变量,将变量 i  赋值为1,当 i=1 以后,i为非零值,表示为真,我们可以认为这是一种模式匹配后的结果,当模式为真时,同时省略了对应动作时,默认动作为打印整行,所以上例会输出test3中的所有行。

awk从放弃到入门(11):拾遗之

 

理解完上述示例以后,我们再回过头来,看看之前打印奇数行的示例,你可能就会明白了。

awk从放弃到入门(11):拾遗之

当awk开始处理第一行时,变量 i 被初始化,变量 i 在被初始化时,值为"空",而awk中,数字0或者"空字符串"表示假,所以可以认为模式为假,但是 i 直接取反了,对假取反后的值为真,将取反后的值又赋值给了变量i,此刻,变量i的值为真,所以当awk处理第一行文本时,变量i的值被赋值为真,模式成立则需要执行对应的动作,而上例中又省略了动作,所以默认动作为"{print $0}",所以,第一行被整行打印了。

当第一行文本处理完毕后,awk开始处理第二行文本,此时,i 为真,但是取反后,i 为假,所以第二行没有被输出,依次类推,最终只打印了奇数行。

为了能够更加直观的看到上述过程,我们将i的值打印出来,通过如下动作,能够打印出处理每一行时,i 对应的值。

awk从放弃到入门(11):拾遗之

当然,聪明如你,我就不用再解释打印偶数行的原理了,我想你应该已经能够举一反三了。

 

好了,这篇文章就写到这里,希望能够对你有所帮助~客官常来呦~~~

 

weinxin
我的微信公众号
关注"实用运维笔记"微信公众号,当博客中有新文章时,可第一时间得知哦~
    • avatar 运维小白 2

      一天看完了博主的awk,受益匪浅,算是入门了吧。后续还要回来复习巩固

                      • avatar 老年程序猿 0

                        花了几个小时把awk都看了一遍,感受颇深,看的晚饭都忘记吃,看完感觉好饱~好后悔这么晚才发现竟然有写的如此好的文章,希望朱老师多多分享干货

                                    • avatar Ning 1

                                      临别前村里的长者告诉我:去了大城市遇到知识渊博的人都要尊称”老师”,这个称号能让他们shuang
                                      刚看完IPtables和awk的我有话要说:朱老师,你爽到我了,我想榨干你的精华,感谢朱老师的万千精华让我成长,正在入运维的门,希望在朱老师的指导下能过上左搂右抱的生活,也希望朱老师的博客越做越好,精华越产越多,福泽天下,雨露均沾,造福全人类……我的功力还是差老师太多太多,还得再接再励,努力学习,勇争第一,争做新时代”四有优秀青年”,争取早日扛起大旗,纵马驰骋,大杀四方,耀武扬威,扬名立外,足迹遍布天下,旌旗×满全图。
                                      到时候,我就可以骄傲的告诉大家,拜师就得朱老师,走遍天下都不怕!

                                              • avatar 怒风 3

                                                打印奇偶行完全可以使用自然一点的思路来实现。 打印偶数行:awk ‘NR%2==0’ file,打印奇数行:awk ‘NR%2!=0’ file

                                                                  • avatar t 2

                                                                    i!=i 好骚

                                                                      • avatar coool 1

                                                                        @t i!=1没结果吧
                                                                        应该是
                                                                        i=!i
                                                                        !(i=!i)

                                                                        朱哥写到很详细,看了之后学到了不少东西

                                                                                          • avatar feyjobs 1

                                                                                            为什么一个赋值运算能在模式里做判断啊, ‘a = 1’ 可以理解为赋值运算返回值就是被赋的值吗

                                                                                              • avatar 易知难 4

                                                                                                博主能帮忙看下这是什么原因么? :sad:
                                                                                                [root@oldboy /]# su – test -c a.sh
                                                                                                -bash: a.sh: command not found
                                                                                                [root@oldboy /]# su – test -c /bin/sh a.sh
                                                                                                sh: cannot set terminal process group (-1): Inappropriate ioctl for device
                                                                                                sh: no job control in this shell
                                                                                                sh-4.1$
                                                                                                ——————————————–分隔符———————————————–
                                                                                                a.sh内容 :echo 1234>>/a.txt 权限以及给了777 , sudo也配置了 test ALL=(ALL) NOPASSWD:ALL

                                                                                                  • avatar 妞妞 4

                                                                                                    @易知难 su – test -c “/bin/sh a.sh”
                                                                                                    评论中必须含有中文。。。。

                                                                                                  • avatar 易知难 4

                                                                                                    打卡!小二 器大活好!!!5星好评!

                                                                                                    • avatar zero 1

                                                                                                      2天awk系列看完,扩展正则那边,–posix 和–re-interval ubuntu系统无法使用,可能是ubuntu的awk不是gun awk,换到centos上正常,谢谢博主好文。

                                                                                                          • avatar 诸葛 0

                                                                                                            @zero 厉害 能否加下我qq845893109