Linux中有好多很有名的搜索类,比如find、loc令、sed命令,但还有三位兄弟是不得不提的,它们在搜索界的地位举足轻重,那就是grep、egrep、fgrep三兄弟。

为何称它们为搜索三兄弟呢?你们瞧瞧下边的家谱就一目了然了。

grep

/

(选项)(-E)(-F)

/

egrepfgrep

从前面的关系图可以看出,egrep和fgrep都可以通过grep加上不同选项来实现,真是“打断双腿连着筋”,血脉相亲的一家人啊。它们三兄弟,各有特征,如表1所示。

表1各自特征命令是否支持正则支持的正则类型

grep

基本正则表达式

egrep

扩充正则表达式

fgrep

×

grep入门剑法grep是能力最全面的,下边我们就先来学习grep,围绕/etc/passwd文件来讲解grep的作用。

[root@roclinux ~]# cat /etc/passwd
root:x:0:0:root:/root:/bin/bash
bin:x:1:1:bin:/bin:/sbin/nologin
daemon:x:2:2:daemon:/sbin:/sbin/nologin
adm:x:3:4:adm:/var/adm:/sbin/nologin
lp:x:4:7:lp:/var/spool/lpd:/sbin/nologin
(中间省略数十行)
apache:x:48:48:Apache:/var/www:/sbin/nologin
test:x:502:502::/home/test:/bin/bash
leo:x:503:503::/home/leo:/bin/bash
roc:x:504:504::/home/roc:/bin/bash
尝试搜索包含leo字符串的行:
[root@roclinux ~]# grep --color "leo" /etc/passwd
leo:x:503:503::/home/leo:/bin/bash

注意里面的--color选项,它的作用是高亮我们查找的字符串,这儿,leo字符串弄成了黑色字体。grep的反查技能下边来搜索不包含leo字符串的行。

# 使用-v选项
[root@roclinux ~]# grep -v "leo" /etc/passwd
root:x:0:0:root:/root:/bin/bash
bin:x:1:1:bin:/bin:/sbin/nologin
daemon:x:2:2:daemon:/sbin:/sbin/nologin
adm:x:3:4:adm:/var/adm:/sbin/nologin
lp:x:4:7:lp:/var/spool/lpd:/sbin/nologin
(中间省略数十行)
apache:x:48:48:Apache:/var/www:/sbin/nologin
test:x:502:502::/home/test:/bin/bash
roc:x:504:504::/home/roc:/bin/bash

我们使用-v选项实现了反查疗效,可以看见,富含leo的行都没有展示下来。grep展示行号和统计行数有时,我们希望grep除了能搜索到字符串,能够展示出它们坐落文件的第几行,这时我们可以使用-n选项来实现这个疗效。

来瞧瞧包含leo的行坐落文件的第几行。

# 使用选项 n
[root@roclinux ~]# grep -n leo /etc/passwd
29:leo:x:503:503::/home/leo:/bin/bash

grep 正则表达式_linux grep 正则表达式 或_linux grep 正则表达式 或

你们注意观察输出内容的开头部份,leo前的内容表示的就是行位置信息,原先是在第29行呀!

另外一些时侯,我们希望grep不要输出搜索到的行的内容,而是简单地告诉我们究竟搜索到了多少行就好了,试试-c选项吧。

# 使用-c选项
[root@roclinux ~]# grep -c leo /etc/passwd
1

输出很简单,这表示富含leo的行只有1行。grep能扫视四周我想搜索包含leo的行,但grep在输出时,最好能把leo所在行的前面或下边相邻的行也都展示下来。

我们来瞧瞧grep是怎么做到的。

[root@roclinux ~]# grep -A 1 leo passwd
leo:x:503:503::/home/leo:/bin/bash
roc:x:504:504::/home/roc:/bin/bash

里面示例中的-A选项,是After的简写,表示不仅展示匹配行之外,还要展示出匹配行下边的若干行。而示例中的-A1则表示还展示匹配行下边一行的内容。

[root@roclinux ~]# grep -B 1 leo passwd
test:x:502:502::/home/test:/bin/bash
leo:x:503:503::/home/leo:/bin/bash

里面示例中的-B选项,是Before的简写,表示不仅展示匹配行之外,还要展示出匹配行里面的若干行。而示例中的-B1则表示还展示匹配行里面一行的内容。

[root@roclinux ~]# grep -C 1 leo passwd
test:x:502:502::/home/test:/bin/bash
leo:x:503:503::/home/leo:/bin/bash
roc:x:504:504::/home/roc:/bin/bash

里面示例中使用了-C选项,它是-A和-B选项的合体,表示不仅展示匹配行之外,还要展示出匹配行前面和下边各若干行。而示例中的-C1则表示还展示匹配行里面一行和下边一行的内容。让grep不要分辨大小写在搜索过程中,我们有时侯希望不要分辨字母的大小写,这样做可以增强搜索命中的机率linux grep 正则表达式 或,而-i选项则可以帮我们这个忙。

[root@roclinux ~]# grep -i "LEO" passwd
leo:x:503:503::/home/leo:/bin/bash

见到了吧,毕竟我们把要搜索的字符串指定为全部小写的LEO,依然可以顺利地搜索到全部大写的leo。grep处理多文件grep命令可以一次搜索好多个文件,最常使用的一个场景就是:从大量的文件中找出富含特定字符的文件。下边我们来试验一下。

# 当前目录下有三个文件
[roc@roclinux ~]$ ll
total 12
-rw-rw-r-- 1 roc roc 58 Mar 15 17:47 1.txt
-rw-rw-r-- 1 roc roc 59 Mar 15 17:51 2.txt
-rw-rw-r-- 1 roc roc 58 Mar 15 17:52 3.txt
 
# 1.txt文件的内容如下
[roc@roclinux ~]$ cat 1.txt
this first file
this file contain some import infomation.
 
# 2.txt文件的内容如下
[roc@roclinux ~]$ cat 2.txt
this second file
this file contain some import infomation.
 
# 3.txt文件的内容如下
[roc@roclinux ~]$ cat 3.txt
this third file
this file contain some import infomation.

我们的搜索需求是,找出内容中富含first词组的文件都有什么。我们希望得到的是一个文件列表。

# 使用-l选项
[roc@roclinux ~]$ grep -l "first" *.txt
1.txt

原先只有1.txt文件中包含有first词组。假如我想找出不含first词组的文件都有什么,该怎么操作呢?

[roc@roclinux ~]$ grep -L "first" *.txt
2.txt
3.txt

反向操作就要用反向选项,只需把-l弄成-L即可。grep也正则里面讲解的内容,更多的是以普通字符串作为搜索对象的,接出来的内容里,我们会更多地设置一些复杂的搜索需求,让你们感受到正则的益处与威力。

我们希望搜索/etc/passwd文件中开头是leo的行:

# ^表示行首
[root@roclinux ~]# grep '^leo' /etc/passwd
leo:x:503:503::/home/leo:/bin/bash

见到了吧,我们使用“^leo”表达了“开头是leo的行”,而grep也心领神会,帮我们找到了所求。

那假如我们希望搜索/etc/passwd文件建行尾是bash的行呢?

# $ 表示行尾
[root@roclinux ~]# grep 'bash$' /etc/passwd
root:x:0:0:root:/root:/bin/bash
cloud-user:x:500:500::/home/cloud-user:/bin/bash
test:x:502:502::/home/test:/bin/bash
leo:x:503:503::/home/leo:/bin/bash
roc:x:504:504::/home/roc:/bin/bash

使用“bash$”即可linux,你会发觉正则表达式显然并不难。有关“词”的知识我们希望搜索富含bin这个词的行,根据原先我们学到的知识,一定会这样来搜索:

[root@roclinux ~]# grep bin /etc/passwd
root:x:0:0:root:/root:/bin/bash
bin:x:1:1:bin:/bin:/sbin/nologin
daemon:x:2:2:daemon:/sbin:/sbin/nologin
adm:x:3:4:adm:/var/adm:/sbin/nologin
lp:x:4:7:lp:/var/spool/lpd:/sbin/nologin
(此处省略数十行)
apache:x:48:48:Apache:/var/www:/sbin/nologin
test:x:502:502::/home/test:/bin/bash
leo:x:503:503::/home/leo:/bin/bash
roc:x:504:504::/home/roc:/bin/bash

然而,你会发觉,这样的搜索结果有一个问题,那就是连sbin这样的词也会被搜索下来,而我们并不想听到这样的结果。这个问题,算是一个比较难的问题了,须要我们了解“词”的定义,以及晓得正则表达式中怎样表示一个“词”。

首先,正则表达式中的“词(word)”,通常是由字母、数字和顿号所组成的,且词与词之间一般使用空格、制表符或换行符分隔。举个反例,“Iloveyou.”中的“love”就是一个“词”,但“Myglovesarered.”中的“love”就不能叫做一个“词”。

在正则表达式中,我们一般用尖括弧表示一个“词”,例如就能匹配“Iloveyou.”,但不能匹配“Myglovesarered.”。

下边我们通过示例学习这个知识点:

#我们尝试匹配bin这个“词”
[root@roclinux ~]# grep '' /etc/passwd --color
root:x:0:0:root:/root:/bin/bash
bin:x:1:1:bin:/bin:/sbin/nologin
sync:x:5:0:sync:/sbin:/bin/sync
cloud-user:x:500:500::/home/cloud-user:/bin/bash
test:x:502:502::/home/test:/bin/bash
leo:x:503:503::/home/leo:/bin/bash
roc:x:504:504::/home/roc:/bin/bash

我们通过使用来确切匹配到了bin这个词,而sbin却被挡在了门外。

虽然,里面的方式还是有些复杂了,grep中还有一个愈发简单的方式来实现对“词”的搜索:

# 使用-w选项来搜索一个词
[root@roclinux ~]# grep -w 'bin' /etc/passwd
root:x:0:0:root:/root:/bin/bash
bin:x:1:1:bin:/bin:/sbin/nologin
sync:x:5:0:sync:/sbin:/bin/sync
cloud-user:x:500:500::/home/cloud-user:/bin/bash
test:x:502:502::/home/test:/bin/bash
leo:x:503:503::/home/leo:/bin/bash
roc:x:504:504::/home/roc:/bin/bash

grep的多条件查找假如我想查找以root为行首的或以bash为行尾的行,那应当如何查找呢?这时就要有请我们的二师弟egrep来帮忙了。

[root@roclinux ~]# egrep '^root|bash$' passwd
root:x:0:0:root:/root:/bin/bash
cloud-user:x:500:500::/home/cloud-user:/bin/bash
test:x:502:502::/home/test:/bin/bash
leo:x:503:503::/home/leo:/bin/bash
roc:x:504:504::/home/roc:/bin/bash

我们通过egrep命令实现了两个条件的搜索,其中“|”符号表示“或”,联接了两个搜索条件。

同样的搜索需求,grep就没有办法满足(除非使用-E选项):

[root@roclinux ~]# grep '^root|bash$' passwd

这就是grep和egrep的区别了,grep的正则表达式是基本正则表达式,而egrep的正则表达式是扩充正则表达式linux grep 正则表达式 或,这两种表达式是有区别的,下边我们就一上去瞧瞧它们的区别。基本正则表达式和扩充正则表达式世界上的正则表达式种类繁杂且复杂,面对这样的状况,POSIX将正则表达式进行了标准化,并把实现方式分为了两大类:

二者的区别,更多的是元字符的区别。

在基本正则表达式(BRE)中,只承认“^”、“$”、“.”、“[”、“]”、“*”这些是元字符,所有其他的字符都被辨识为普通字符。

而在扩充正则表达式(ERE)中,则在BRE的基础上降低了“(”、“)”、“{”、“}”、“?”和“+”、“|”等元字符。

最后要非常说明的一点,只有在用反斜杠进行通配符的情况下,字符“(”、“)”、“{”和“}”才会在扩充正则表达式(ERE)中被当成元字符处理,而在基本正则表达式(ERE)中,任何元字符后面加上反斜杠反倒会使其被当做普通字符来处理。这样的设计,有些雷人,朋友们一定要记清楚哦。三师兄fgrep最朴实grep和egrep都支持正则表达式,但fgrep却完全不支持正则表达式。

#匹配到了好多行
[root@roclinux ~]# grep -c '^root' /etc/passwd
6688
 
#一行都没有匹配到
[root@roclinux ~]# fgrep -c '^root' /etc/passwd
0

这个示例证明了fgrep的确不支持正则表达式。那fgrep还有用武之地么?答案是肯定的。

当我们搜索时,如果搜索字符串中包含了不少特殊字符,而这种特殊字符正好又是正则表达式预留的字符,例如说“^”、“$”等,这时,我们就可以使用fgrep来防止繁杂的通配符了,由于我们心中晓得,在fgrep的眼中,没有特殊字符,都是普通字符。我们来看下边的示例。

#我们的roc.txt文件中有几个^和$
[roc@roclinux ~]$ cat roc.txt
^this third file
^$this file contain some import infomation.
 
# grep会尝试去找开头为this的行, 但并未找到
[roc@roclinux ~]$ grep '^this' roc.txt
 
# fgrep会老老实实地去找^this字符串, 它找到了
[roc@roclinux ~]$ fgrep '^this' roc.txt
^this third file

好了,洋洋洒洒的一篇grep三兄弟就介绍到这儿了linux格式化命令,要和三兄弟道别了,之后遇见搜索字符串的场景,可不要忘了它们三位哦!

本文原创地址://q13zd.cn/lzzbdshlxtzy.html编辑:刘遄,审核员:暂无