awk 中 NF 的后递减

Wil*_*ell 4 awk

我对以下内容有些困惑:

$ echo foo bar baz | awk '{printf "%d:", NF--; print NF}'
3:2
$ echo foo bar baz | awk '{printf "%d:", NF; $NF=""; NF -= 1; print NF}' 
3:2
$ echo foo bar baz | awk '{printf "%d:", NF; $(NF--)=""; print NF}' 
3:3
Run Code Online (Sandbox Code Playgroud)

我在awk version 20070501(macos) 和GNU Awk 4.0.2. 为什么第 3 种情况下 NF 的后递减不适用?这种行为是预期的,是标准规定的,还是实施的一个怪癖?


Ed Morton 编辑:FWIW 我会发现以下一个更引人注目的例子:

$ echo foo bar baz | awk '{printf "%d:", NF; NF--; $NF=""; print NF}'
3:2

$ echo foo bar baz | awk '{printf "%d:", NF; --NF; $NF=""; print NF}'
3:2

$ echo foo bar baz | awk '{printf "%d:", NF; $NF=""; NF--; print NF}'
3:2

$ echo foo bar baz | awk '{printf "%d:", NF; $NF=""; --NF; print NF}'
3:2

$ echo foo bar baz | awk '{printf "%d:", NF; $(--NF)=""; print NF}'
3:2

$ echo foo bar baz | awk '{printf "%d:", NF; $(NF--)=""; print NF}'
3:3
Run Code Online (Sandbox Code Playgroud)

问题是为什么最后一个示例(带有赋值的后递减)的行为与所有其他情况不同,无论您认为它应该与哪个相同。

Bar*_*mar 7

递减后的值是变量递减前的值。因此,最后一种情况是在 decrements 之后添加一个新字段NF,该字段会更新NF.

$(NF--) = "";
Run Code Online (Sandbox Code Playgroud)

相当于

temp = NF;  # temp == 3
NF--;       # NF == 2
$temp = ""; # adds a new field 3, so now NF == 3
Run Code Online (Sandbox Code Playgroud)