《Linux命令行与shell脚本编程大全》第二十章 正则表达式

20.1 什么是正则表达式

20.1.1 定义

正则表达式是你所定义的模式模板。linux工具可以用它来过滤文本。

正则表达式利用通配符来描述数据流中第一个或多个字符。

正则表达式模式含有文本或特殊字符,为sed编辑器和gawk程序定义了一个匹配数据时采用的模板。

20.1.2 正则表达式的类型

使用正则表达式最大的问题在于有不止一种类型的正则表达式。

正则表达式是通过正则表达式引擎实现的,正则表达式引擎是一套底层软件,负责解释正则表达式模式并使用这些模式进行文本匹配。

在linux中有两种流行的正则表达式引擎:

1)POSIX基础正则表达式(BRE)引擎

2)POSIX扩展正则表达式(ERE)引擎

大部分linux工具都至少符合POSIX BRE引擎规范,能够识别该规范定义的所有模式符号。

但是,有些工具只支持BRE引擎规范下的子集。比如sed,这是出于速度方面的考虑。

gawk程序用ERE引擎来处理它的正则表达式模式。

20.2 定义BRE模式

20.2.1 纯文本

例子:

$echo “This is test line” | sed -n ‘/test/p’

$echo “This is test line” | gawk ‘/test/{print $0}’

正则表达式并不关心模式在数据流中的位置,也不关心出现了多少次,只要匹配了就会将该字符串传会linux工具。

正则表达式模式区分大小写。

20.2.2 特殊字符

正则表达式识别的特殊字符包括:

.*[]^${}\+?|()

如果要用某个特殊字符作为文本字符,就必须转义。在前面加上反斜线\。

比如:

$echo “This cost is \$200” | sed –n ‘/\$/p’  // 注意这里应该是\$200,而不是$200

$echo “\ hahah, this is fanxiexian” | sed –n ‘/\\/p’

要使用正斜线也需要用转义字符

$ehco “6 / 3 = 2” | sed -n ‘/\//p’

20.2.3 锚字符 ^ $

默认情况下,模式出现再数据流中的任何地方,它就能匹配。

有两个特殊字符可以用来将模式锁定在数据流中的行首或行尾。

1.锁定在行首(脱字符 ^)

^ 定义从数据流中文本行的行首开始的模式。如果模式出现在行首之外的位置,正则表达式模式则无法匹配。

需要用^,就必须将它放在正则表达式中指定的模式前面。

比如:

$echo “The book store” | sed –n ‘/^book/p’  // 这样不会有输出

$echo “The book store” | sed –n ‘/^The/p’   // 这个才有

还可以输入文件:

$sed -n ‘/^this/p’ data.txt

data.txt 中以this开头的行就能找出来。

注意如果将^放到模式开头之外的其他位置,那么久跟普通字符一样了。

$echo “This is ^ test” | sed –n ‘/is ^/p’  // 匹配 is ^ 。

注意:

如果指定正则表达式模式时只用了脱字符,就不需要用反斜线来转义。

如果你在模式中先指定了脱字符,随后还有一些其他文本,那么你必须在脱字符前用转义字符。

2. 锁定在行尾

用美元符$

$echo “This is test line” | sed –n ‘/line$/p’  //这样可以匹配到

$echo “This is test lines” | sed –n ‘/line$/p’  //这样不可以匹配到

要想匹配,文本模式必须是行的最后一部分。

3. 组合锚点

比如想到匹配指定内容

$sed ‘/^this is test line$/p’ data.txt   // 匹配行 this is test line

将两个锚点直接组合在一起,之间不加任何东西,这样就过滤出数据流中的空白行。

$sed -n ‘/^$/p’ data.txt   // 这样可以把data.txt 中的空白行过滤出来。

20.2.4 点号字符 .

用来匹配除换行符之外的任意单个字符。它必须匹配一个字符,如果点字符的位置没有字符那么模式就不成立。

例子:

xcy@xcy-virtual-machine:~/shell/20zhang$ cat data.txt

this is a test line

the cat is sleeping

that is a very nice hat

this test is at line four

at ten o'clock we'll go home

xcy@xcy-virtual-machine:~/shell/20zhang$ sed -n '/.at/p'  data.txt

the cat is sleeping

that is a very nice hat

this test is at line four

xcy@xcy-virtual-machine:~/shell/20zhang$

空格也算一个字符。注意第5行没有匹配到。at前面没有字符了。

20.2.5 字符组 []

可以限定待匹配的具体字符,在正则表达式中,这称为字符组。用[]括起来

比如:

$sed –n ‘/[ch]at/p’ data.txt  // 相当于只匹配cat 或者 hat。其他的at就不匹配了。

在不确定大小写的时候,字符组会非常有用:

$echo “Yes” | sed –n ‘/[yY]es/p’

还可以用多个字符组:

$echo ‘YeS’ | sed –n ‘/[Yy][Ee][Ss]/p’

这样就相当于可以限制行的字符个数和区间了

20.2.6 排除型字符组

相当于字符组取反,可以寻找字符组中没有的字符。在前面加个脱字符就好了

$sed –n ‘/[^ch]at/p’ data.txt

匹配出c和h以外的字符。

20.2.7 区间

0 – 9,可以直接这么写[0-9] 而不需要[0123456789]

$sed –n ‘/^[0-9][0-9][0-9][0-9]&/p’ data.txt

字母也可以

$sed –n ‘/[c-h]at/p’ data.txt   // 匹配c到h这个区间的字符。

还可以指定多个不连续的区间:

$sed –n ‘/[a-ch-m]at/p’ data.txt  // 指定 a-c  和 h-m区间的字母。

$echo “This is foot” | sed –n ‘/[a-ch-n]oot/p’

20.2.8 特殊的字符组

除了自定义的区间(比如[0-9] [a-f])之外,BRE还包含了一些特殊的字符组。见下表

描述

[[:alpha:]]

匹配任意字母字符,不管大小写

[[:alnum:]]

匹配任意字母数字字符 0-9 a-z A-Z

[[:blank:]]

匹配空格或制表符

[[:digit:]]

匹配数字 0-9

[[:lower:]]

匹配小写字母 a-z

[[:print:]]

匹配任意可打印字符

[[:punct:]]

匹配标点符号

[[:space:]]

匹配任意空白字符:空格、制表符、NL、FF、VT和CR

[[:upper:]]

匹配大写字母A-Z

使用:

$echo “achsdsd” | sed –n ‘/[[:digit:]]/p’

$echo “2344” | sed –n ‘/[[:digit:]]/p’

$echo “achsdsd” | sed –n ‘/[[:lower:]]/p’

$echo “this is , a test” | sed –n ‘/[[:punct:]]/p’

20.2.9 星号 *

字符后面放置星号用来表明该字符必须在匹配模式的文本中出现0次或多次。

例子:

$echo “I am hahahaaaaa” | sed -n ‘/ha*h/p’ // 表明a可以出现0次或多次。

*还能用到字符组上,它允许指定可能在文本中出现多次的字符组或区间:

$echo ‘bt’ | sed –n ‘/b[ae]*t/p’   // a出现0次或次,e出现0次或多次

20.3 扩展正则表达式(POSIX ERE)

提供了一些可以供linux应用和工具使用的额外符号。

gawk程序(会慢一点)能够识别,sed编辑器(查找比较快)不能识别。

20.3.1 问号?

类似于星号,但是有点不同。

问号表明前面的字符可以出现0次或1次,不会匹配出现多次的字符。

$echo “bt” | gawk ‘/be?t/{print $0}’

$echo “bet” | gawk ‘/be?t/{print $0}’

$echo “beet” | gawk ‘/be?t/{print $0}’   // e 出现了2次,这里就不输出了

还可以跟字符组一起使用:

$echo “bet” | gawk ‘/b[ae]?t/{print $0}’

$echo “bat” | gawk ‘/b[ae]?t/{print $0}’

$echo “baet” | gawk ‘/b[ae]?t/{print $0}’  // 这里相当于出现了2次,也不会输出

$echo “baet” | gawk ‘/b[a-f]?t/{print $0}’

20.3.2 加号+

有点像*号。但是必须出现1次以上。可以有多次

$echo “baet” | gawk ‘/b[ae]+t/{print $0}’

20.3.3 使用花括号{}

花括号允许你为可重复的正则表达式指定一个上限,这通常称为间隔。可以用两种格式来指定区间。

1)m:正则表达式准确出现m次

2)m,n:正则表达式至少出现m次,至多n次。

注意:默认情况下gawk程序不识别正则表达式间隔。必须指定gawk程序的 –re-interval命令行选项才能识别正则表达式间隔。

例子:

$echo “bt” | gawk –re-interval ‘/be{1}t/{print $0}’  // 指定出现1次e

$echo “bet” | gawk –re-interval ‘/be{1}t/{print $0}’

$echo “beeet” | gawk –re-interval ‘/be{3,6}t/{print $0}’  // e至少出现3次,至多6

$echo “baaeeet” | gawk –re-interval ‘/b[ae]{1,3}t/{print $0}’  // a e 出现1 -3

20.3.4 管道符号|

管道符号允许你在检查数据流时,用逻辑or方式指定正则表达式引擎要用的两个或多个模式。

格式如下:

expr1|expr2|……

例子:

echo "the dog is sleeping" | gawk '/cat|dog/{print $0}'  //  去匹配cat或dog

echo "the tat is sleeping" | gawk '/[ct]at|dog/{print $0}' // 去匹配 [ct]at或dog

20.3.5 表达式分组()

正则表达式模式也可以用圆括号进行分组。当你将正则表达式模式分组时,该组会被视为一个标准字符。可以像对普通字符一样给该组使用特殊字符。

例子:

$echo “Sat” | gawk ‘/Sat(urday)?/ {print $0}’

相当于把urday当做一个整体了, /SatF?/  跟这个类似,F出现0次或1次。

()需要转义的用法:

echo "Sat(urday)" | gawk '/Sat\(urday\)/ {print $0}'

echo "Saturday" | gawk '/Sat\(urday\)/ {print $0}'

还可以将分组和管道符号一起使用来创建可能的模式匹配组:

echo "cat" | gawk '/(b|c)a(b|t)/ {print $0}'

这样相当于匹配 bab bat cab cat 四种。

20.4 正则表达式实战

20.4.1 目录文件计数

这个例子用于对PATH环境变量中各个目录里的可执行文件进行计数:

#!/bin/bash

mypath=$(echo $PATH | sed 's/:/ /g')   # 这里把用冒号分割的字符串换成用空格分割

#echo "mypath = $mypath"

count=0

for dir in $mypath

do

         check=$(ls $dir)  // 查询单个目录

         for item in $check

         do

                   count=$[ count + 1 ]

         done

         echo "$dir - $count"

         count=0

done

20.4.2 验证电话号码

美国的号码格式如下:

(223)456-7890

(223) 456-7890

223-456-7890

223.456.7890

最开始是( : ^\(?

三位区号(第一位必须大于2):[2-9][0-9]{2}

然后又是括号:\)?

然后是分隔符(空格,点,减号):( |-|\.)

然后是3位数字:[0-9]{3}

又是分隔符:( |-|\.)

最后四位数字:[0-9]{4}

连起来就是:

^\(?[2-9][0-9]{2}\)?( |-|\.)[0-9]{3}( |-|\.)[0-9]{4}$

例子:

xcy@xcy-virtual-machine:~/shell/20zhang$ cat isphone

#!/bin/bash

gawk --re-interval '/^\(?[2-9][0-9]{2}\)?( |-|\.)[0-9]{3}( |-|\.)[0-9]{4}$/ {print $0}'

xcy@xcy-virtual-machine:~/shell/20zhang$ echo "222-098-2231" | ./isphone

222-098-2231

xcy@xcy-virtual-machine:~/shell/20zhang$ echo "122-098-2231" | ./isphone

xcy@xcy-virtual-machine:~/shell/20zhang$ echo "222.098.2231" | ./isphone

222.098.2231

xcy@xcy-virtual-machine:~/shell/20zhang$

还可以将整个文件重定向到脚本:

phonelist里面存放着一行一行的数据:

$cat phonelist | ./isphone

这个例子其实也不是特别合理,比如下面几种情况也算是合法号码:

(235.561-4430

343) 451.4651

总之还是有待优化。

20.4.3 解析邮件地址

邮件地址的形式如下:

username@hostname

username值可用字母数字字符以及以下特殊字符:

1)点号

2)单破折线

3)加号

4)下划线

hostname部分由一个或多个域名和一个服务器名组成。只允许字母数字字符以及下面的特殊字符:比如(xiaochongyong@amwell-haha.com)

1)点号

2)下划线

username@相当于:^([a-zA-Z0-9_\-\.\+]+)@

注意: [] 里面是字符组,相当于之前的[xcs]。() 里面是表达式分组

hostname相当于:([a-zA-Z0-9_\-\.\]+)

后面还要接*域名。.com .cn .org  等

只能是字母字符,必须不少于两个字符,长度不超多5个字符:\.([a-zA-Z]{2,5})$

连起来就是:

^([a-zA-Z0-9_\.\-\+]+)@([a-zA-Z0-9_\.\-]+)\.([a-zA-Z]{2,5})$

例子:

xcy@xcy-virtual-machine:~/shell/20zhang$ cat isemail

#!/bin/bash

gawk --re-interval '/^([a-zA-Z0-9_\.\-\+]+)\@([a-zA-Z0-9_\-\.]+)\.([a-zA-Z]{2,5})$/ {print $0}'

xcy@xcy-virtual-machine:~/shell/20zhang$ echo "222x2iaochongyong@s1amwell-elec.com" | ./isemail

222x2iaochongyong@s1amwell-elec.com

xcy@xcy-virtual-machine:~/shell/20zhang$ echo "222x2iaocho%ngyong@s1amwell-elec.com" | ./isemail

xcy@xcy-virtual-machine:~/shell/20zhang$ echo "222x2iaocho%ngyong+@s1amwell-elec.com" | ./isemail

xcy@xcy-virtual-machine:~/shell/20zhang$ echo "222x2iaochongyong+@s1amwell-elec.com" | ./isemail

222x2iaochongyong+@s1amwell-elec.com

xcy@xcy-virtual-machine:~/shell/20zhang$ echo "222x2iaochongyong+@s1amwell-elec..com" | ./isemail

222x2iaochongyong+@s1amwell-elec..com

xcy@xcy-virtual-machine:~/shell/20zhang$ echo "222x2iaochongyong+@s1amwell-elec._.com" | ./isemail

222x2iaochongyong+@s1amwell-elec._.com

xcy@xcy-virtual-machine:~/shell/20zhang$

20.5 小结

正则表达式定义了用来过滤数据流中文本的模式模板。

模式由标准文本字符和特殊字符的组成。

正则表达式引擎用特殊字符来匹配一系列单个或多个字符,这类似于其他应用程序中通配符的工作方式。

上一篇:云边端EasyGBS视频智能分析算法接入方案


下一篇:EasyGBS视频平台如何通过限制用户播放时间来减小带宽压力?