为了账号安全,请及时绑定邮箱和手机立即绑定

Bash正则表达式-似乎无法匹配\ s,\ S等

Bash正则表达式-似乎无法匹配\ s,\ S等

慕雪6442864 2019-10-12 10:49:57
我有一个脚本试图从gparted获取信息块。我的数据如下所示:Disk /dev/sda: 42.9GBSector size (logical/physical): 512B/512BPartition Table: msdosNumber  Start   End     Size    Type     File system     Flags 1      1049kB  316MB   315MB   primary  ext4            boot 2      316MB   38.7GB  38.4GB  primary  ext4 3      38.7GB  42.9GB  4228MB  primary  linux-swap(v1)log4net.xmlModel: VMware Virtual disk (scsi)Disk /dev/sdb: 42.9GBSector size (logical/physical): 512B/512BPartition Table: msdosNumber  Start   End     Size    Type     File system     Flags 1      1049kB  316MB   315MB   primary  ext4            boot 5      316MB   38.7GB  38.4GB  primary  ext4 6      38.7GB  42.9GB  4228MB  primary  linux-swap(v1)我使用正则表达式将其分为两个磁盘块^磁盘(/ dev [\ S] +):((?! Disk)[\ s \ S])*这适用于多行。当我在bash脚本中对此进行测试时,我似乎无法匹配\ s或\ S-我在做什么错?我正在通过如下脚本进行测试:data=`cat disks.txt`morematches=1x=0regex="^Disk (/dev[\S]+):((?!Disk)[\s\S])*"if [[ $data =~ $regex ]]; thenecho "Matched"while [ $morematches == 1 ]do        x=$[x+1]        if [[ ${BASH_REMATCH[x]} != "" ]]; then                echo $x "matched" ${BASH_REMATCH[x]}        else                echo $x "Did not match"                morematches=0;        fidonefi但是,当我遍历正则表达式的各个部分时,无论何时匹配\ s或\ S,它都不起作用-我在做什么错?
查看完整描述

3 回答

?
喵喔喔

TA贡献1735条经验 获得超5个赞

因为这是一个常见的常见问题解答,所以让我列出一些Bash不支持的结构,以及如何解决它们,这里有一个简单的解决方法。


正则表达式有多种常用方言。Bash支持的是扩展正则表达式的变体。例如,这不同于许多在线正则表达式测试器支持的功能,后者通常是更现代的Perl 5 / PCRE变体。


击不支持\d \D \s \S \w \W-这些可以与POSIX字符类等同替换[[:digit:]],[^[:digit:]],[[:space:]],[^[:space:]],[_[:alnum:]],和[^_[:alnum:]],分别。(请注意最后一种情况,其中[:alnum:]POSIX字符类使用下划线增强,以完全等同于Perl的\w速记形式。)

Bash不支持非贪婪匹配。有时您可以a.*?b用类似的东西代替,a[^ab]*b以在实践中获得类似的效果,尽管两者并不完全相同。

Bash不支持非括号(?:...)。在平凡的情况下,只需使用捕获括号即可(...);当然,如果您使用捕获组和/或反向引用,则会重新编号捕获组。

击不支持lookarounds像(?<=before)或者(?!after),事实上任何与(?是一个Perl扩展。尽管您经常可以将问题改写为可以避免环顾的问题,但没有简单的常规解决方法。


查看完整回答
反对 回复 2019-10-12
?
茅侃侃

TA贡献1842条经验 获得超21个赞

我知道您已经“解决”了这个问题,但是您最初的问题很可能就像$regex在测试中不引用一样简单。即:


if [[ $data =~ "$regex" ]]; then

Bash变量扩展将简单地放入字符串中,并且原始正则表达式中的空格将破坏测试,因为:


regex="^Disk (/dev[\S]+):((?!Disk)[\s\S])*"

if [[ $data =~ $regex ]]; then

等价于:


if [[ $data =~ ^Disk (/dev[\S]+):((?!Disk)[\s\S])* ]]; then

bash / test会很有趣地解释奖金参数和所有未引用的元字符。


请记住,bash不会传递变量,而是会扩展变量。


查看完整回答
反对 回复 2019-10-12
  • 3 回答
  • 0 关注
  • 1015 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信