&& 的逻辑运算符不能在 awk if 正则表达式模式中工作,这与 || 的逻辑运算符相比似乎有点奇怪。可以正常工作!
注意:但是 && 的逻辑运算符可以在纯简单的 awk 正则表达式模式中工作,那为什么呢?
#
#
# cat /etc/passwd|awk '$0 ~/[Ss]ystem/ || /puls/ {print $0}'
dbus:x:81:81:System message bus:/:/sbin/nologin
pulse:x:497:495:PulseAudio System Daemon:/var/run/pulse:/sbin/nologin
#
#
#
# cat /etc/passwd|awk '$0 ~/[Ss]ystem/ && /puls/ {print $0}'
pulse:x:497:495:PulseAudio System Daemon:/var/run/pulse:/sbin/nologin
#
#
#
# cat /etc/passwd|awk '$0 ~/[Ss]ystem|puls/ {print $0}'
dbus:x:81:81:System message bus:/:/sbin/nologin
pulse:x:497:495:PulseAudio System Daemon:/var/run/pulse:/sbin/nologin
#
#
#
# cat /etc/passwd|awk '{k1[NR]=$0}END{for(i=1;i<=NR;i++)if(k1[i] ~/[Ss]ystem/ || /puls/){print k1[i]}}'
dbus:x:81:81:System message bus:/:/sbin/nologin
pulse:x:497:495:PulseAudio System Daemon:/var/run/pulse:/sbin/nologin
#
#
# cat /etc/passwd|awk '{k1[NR]=$0}END{for(i=1;i<=NR;i++)if(k1[i] ~/[Ss]ystem/ && /puls/){print k1[i]}}'
#
#
# cat /etc/passwd|awk '{k1[NR]=$0}END{for(i=1;i<=NR;i++)if(k1[i] ~/[Ss]ystem/ && /puls/){print k1[i]}}'
#
#
# cat /etc/passwd|awk '$0 ~/[Ss]ystem/ && /puls/ {print $0}'
pulse:x:497:495:PulseAudio System Daemon:/var/run/pulse:/sbin/nologin
#
#
现在我稍作修改后,最新的情况可能会变得比以前奇怪得多!
为什么下面带有数组变量和逻辑运算符 && 的相同正则表达式模式可以工作?
注意:这应该可以解释某些 if 正则表达式模式中的 && 问题至少与数组变量本身无关。
#
#
# cat /etc/passwd|awk '{k0[NR]=$0;if(k0[NR] ~/[Ss]ystem/ && /puls/){print k0[NR]}}'
pulse:x:497:495:PulseAudio System Daemon:/var/run/pulse:/sbin/nologin
#
#
# cat /etc/passwd|awk '{k0[NR]=$0;if(k0[NR] ~/[Ss]ystem/ || /puls/){print k0[NR]}}'
dbus:x:81:81:System message bus:/:/sbin/nologin
pulse:x:497:495:PulseAudio System Daemon:/var/run/pulse:/sbin/nologin
#
#
# cat /etc/passwd|awk '{k0[NR]=$0;if(k0[NR] ~/[Ss]ystem/ && /puls/){print k0[NR]}}'
pulse:x:497:495:PulseAudio System Daemon:/var/run/pulse:/sbin/nologin
#
#
在第一部分
隐含于
所以
应该写成
/pusl/
,很可能(*) 与 的最后一行相匹配/etc/passwd
。man awk
但~
(和!~
)优先于&&
(*) 确认,如果我将 PulseAudio 行移动到最后一行,则匹配完成。
您假设运算符的关联错误,可能还有它们的含义。给你的测试用例编号:
$0 ~/[Ss]ystem/
检查“$0 是否匹配 /RE/”。该|| /puls/
条件甚至没有经过测试:它是“短路的”,因为已知整体条件为真。检查两个条件是否都为
&&
真,并且只匹配一行。合并的 RE(带有 的两个替代案例
|
)匹配两行。这与包含
System
.这什么都不匹配。它测试数组中的存储行,但是“/puls/”在运行完文件末尾后测试 $0 中的值——它不引用 k1[i]。$0 仍将包含文件的最后一行,我猜想完整的输入文件末尾没有这样的行。要求两个条件都为真,因此
&&
没有行匹配。是 (5) 的副本。
是 (2) 的副本。
这是对您最初问题的“最新情况”的解构,以简化问题。
cat 不向代码添加任何内容,因为 awk 可以读取文件本身。(谷歌 UUOC 了解更多。)
将输入行存储在一个数组中,然后在同一语句中测试该值,不会添加任何内容(因为没有 END 块)。所以我们可以简化为:
使用
~
模式匹配与普通 RE 相同,因此简化为:使用 an
if
与大括号外的模式相同,因此:print $0
与 相同print
,并且{ print }
作为默认操作,因此:你最初的断言
&&
不能工作,但||
工作正常,是错误的。这读作“如果模式 A和模式 B 都出现在该输入行中”:
这读作“如果此输入行中存在模式 A或模式 B”: