Mar*_*lin 4 awk regular-expression
我有一行用冒号分隔的值,我想在 awk 中处理。如果$4
变量$3
在开头包含变量,则行的处理方式不同。
所以我写了表达式: $4 ~ /^$3/
,但不幸的是这不起作用,它永远不会匹配。怎么了,如何在正则表达式模式中使用变量?
这是完整的示例:
green="$(tput setaf 2)"
red="$(tput setaf 1)"
yellow="$(tput setaf 3)"
normal="$(tput sgr0)"
stacks=$(docker stack ls --format='{{.Name}}')
for stack in ${stacks}; do
status=$(docker stack ps --filter="desired-state=running" --format="{{.Name}}:{{.Node}}:{{.DesiredState}}:{{.CurrentState}}:{{.Error}}" ${stack})
if test -z "$status"; then
echo "${red}$stack$: disabled${normal}"
else
awk -F: '
$4 ~ /^$3/ {print "GOOD '"${green}"'" $1 ": " $4 "'"${normal}"'"}
$4 !~ /^$3/ {print "BAD '"${yellow}"'" $1 ": " $3 " ? " $4 $5 "'"${normal}"'"}
' <<<${status}
fi
done
Run Code Online (Sandbox Code Playgroud)
结果总是BAD
,例如这里,行:
bind_bind.1:urknall:Running:Running 18 hours ago:
Run Code Online (Sandbox Code Playgroud)
应该打印GOOD
,但打印:
BAD bind_bind.1: Running ? Running 18 hours ago
Run Code Online (Sandbox Code Playgroud)
您可以将正则表达式放在~
字符串中 a的右侧,它不必是一个/.../
构造。(差异可能与在运行时或编译时解析 RE 有关,但我不确定。)请记住,在 中awk
,美元符号并不意味着像 shell 或 Perl 中那样的变量扩展,因此您需要连接$3
到字符串的其余部分:
第一个匹配,第二个不匹配:
$ echo 'foo fo+' |awk '$1 ~ "^" $2'
foo fo+
$ echo 'foo o+' |awk '$1 ~ "^" $2'
$
Run Code Online (Sandbox Code Playgroud)
/^$2/
被视为带有文字的正则表达式$2
,并且$
是行尾锚点。由于在 EOL 之后您不能拥有任何东西,因此 RE 永远无法匹配。
/^$3/
是保证永远不会匹配的正则表达式,因为它匹配记录3
末尾之后的记录($
正则表达式锚运算符匹配主题末尾,不要与$
awk
用于取消引用字段的运算符混淆按编号)。
要测试第三个字段是否出现在第四个字段的开头,可以使用正则表达式匹配match()
,这将返回匹配的开始位置(如果未找到匹配,则返回 -1):
awk -F ':' 'match($4, $3) == 1 { ..."GOOD"... ; next } { ..."BAD"... }'
Run Code Online (Sandbox Code Playgroud)
或者,对于字符串比较,
awk -F ':' 'substr($4, 1, length($3)) == $3 { ..."GOOD"... ; next } { ..."BAD"... }'
Run Code Online (Sandbox Code Playgroud)