如何在 bash 中进行文本匹配

如何在 bash 中进行文本匹配

我正在尝试找到一种检查文本以设置变量的好方法。例如执行逻辑:“如果版本字符串是 1.x,则设置一个标志”。一种方法是:

versionFlag="0"
if grep -q "^1.*" versionfile; then
  versionFlag="1"
fi

这有点冗长。如果我想切换多种可能性,那么这更容易:

case `grep "^1.*" versionfile` in
  1.*)
    echo "version 1"
    ;;
...
esac

但是,当某些文本与正则表达式或通配符匹配时,是否有一种好用的单行方法可以仅将标志设置为真或假?

答案1

versionFlag=`grep "^1.*" /tmp/test | wc -l`

versionFlag然后设置为 0。测试匹配中的任何数据都versionFlag设置为匹配的行数,因此

if [ $versionFlag == 0 ] ; then
  echo false
else
  echo true 
fi

编辑:这可以做得更简单

versionFlag=$(grep "^1.*" /tmp/test -c)

这将返回文件中实例的数量

答案2

您可以在 bash 3.x 中执行类似这样的操作(旧版本不支持正则表达式匹配)

version=`cat versionfile`
[[ $version =~ "^1.*" ]] && versionFlag=1

答案3

Bash 中没有三元运算符(例如 ?:),因此您只能使用多行 if, then 构造。您可以使用其他具有三元运算符的语言(例如 PERL)。

请注意,很多人不鼓励使用三元运算符,因为它们在代码中不太容易阅读 - 换句话说,你的注释最终会比多行代码还要长。

我怀疑这两种方法是否具有性能优势。

答案4

实际上,这个结构:

[[ $version =~ ^1\.+ ]] && versionFlag=1

很容易扩展以支持“不匹配”的情况;只需这样做:

[[ $version =~ ^1\.+ ]] && versionFlag=1 || versionFlag=0

顺便说一句,请注意正则表达式中从“*”到“+”的变化。使用星号,正则表达式仍然会错误地匹配版本值“10”。(“*”修饰符表示“匹配零个或多个出现”,即,您仍然不需要小数点;“+”修饰符表示“匹配一个或多个出现”。)但是,进行此更改将意味着“1”(即没有小数的值)将不是匹配;这对于您的应用程序来说可能是也可能不是一个问题。

如果您没有正则表达式匹配运算符,还有一种不使用它的方法;试试这个:

[[ ${version#1.} != ${version} ]] && versionFlag=1 || versionFlag=0

这只是一个模式匹配,而不是正则表达式,所以如果你真的需要正则表达式功能,它就不起作用了,但对于简单的东西,它就很棒了。(如果你不熟悉这种形式的参数扩展,那么“${version#1.}”构造将扩展为“version”的值,但如果值开头有后置“#”模式,则会将其删除。示例:如果“version”的值为“1.03”,则“${version#1.}”的结果为“03”。由于这与原始值不同,因此“!=”测试成功。)

相关内容