- $ cat file
- 1
- 2
- 3
- 4
- 5
- 6
- $ awk 'NR%2' file
- 1
- 3
- 5
- $ awk '!(NR%2)' file
- 2
- 4
- 6
[解析]
这应该是最标准的答案,NR是行号,awk的内建函数,当第一行,NR==1时,对2取余,结果是1,在awk的执行模式里,这个1就是pattern,为真,执行默认的{print},这个action操作。那么打印出该行,到第2行则余数是0,pattern为0,则为假,不会执行默认的{print},则不会输出偶数行。就达到了只输出奇数行的效果,反之则输出偶数行了。我们再看看一个更新颖的办法。
- $ awk '++i%2' file
- 1
- 3
- 5
- $ awk 'i++%2' file
- 2
- 4
- 6
[解析]
这行命令充分的利用了C语言中 ++i 与 i++ 的区别,i++ 中i的值自加一次后这个表达式的结果等于1,但此时i的值仍然为0,为什么呢?i的值在当前表达式中不会改变,++i中i的值自加一次后,i的值是1,听起来很矛盾?确实比较饶口,我们再调用一个C程序来说明例子:
- # include <stdio.h>
- int main(void)
- {
- int i, j, a, b;
- i = j = 0;
- a = i++;
- b = ++j;
- printf ("a = %d , i = %d , b = %d , j = %d\n",a,i,b,j);
- return 0;
- }
- /*
- 运行结果:
- a = 0 , i = 1 , b = 1 , j = 1
- */
[解析]
在这个C语言程序中不难看出,a的值是0,i的值却是1,i++ 是把自加前的值赋予了a,则 ++i 不一样。这就可以解释为什么 i++ 输出的是偶数行了,第一行时,因为对2取余的时候 i++ 自加一次后的值是1,但 i++ 是把i的自加前的值去对2取余的,0对2取余结果为0,条件为假,到第二行的时候i的值才是上次自加后的值为1,对2取余数为1,pattern为真,则输出第2行,以此类推。++i 则是自加后i的值是1,所以输出的奇数行。++i 和 i++ 他们自加的结果都是i为1,但是在赋值和比较的时候,却是有区别的。看到这里我想聪明的您已经明白了它们的区别。我们再看看第三中方法。
- $ awk 'i=!i' file
- 1
- 3
- 5
- $ awk '!(i=!i)' file
- 2
- 4
- 6
[解析]
这句还需要我解释吗?阁下想必已经明白其中的奥妙了吧。^_^
- $ awk 'and(FNR,1)' file
- 1
- 3
- 5
- $ awk '!and(FNR,1)' file
- 2
- 4
- 6
[解析]
awk里面有三个built-in function,分别是:and(a, b)按位与,or(a, b)按位或,xor(a, b)按位异或,当FNR为1的时候,就是第一行,1的2进制为0001,与0001按位与,得到结果是0001,0001的结果为真,就打印改行,当FNR为2时二进制表示为0010,那么又与0001相与,结果为0000,条件为假,则不打印。只有当FNR为奇数时,二进制的尾数为1,与0001相与结果才为真,否则结果都为假。即只输出奇数行,非则输出偶数行。
- $ sed -n 'p;n' file
- 1
- 3
- 5
- $ sed -n 'n;p' file
- 2
- 4
- 6
[解析]
sed也不难解释,好好看看手册吧。
- $ seq 6 | sed -n '1~2p'
- 1
- 3
- 5
- $ seq 6 | sed -n '0~2p'
- 2
- 4
- 6
[解析]
FIRST~STEP
This GNU extension matches every STEPth line starting with line FIRST.In particular, lines will be selected when there exists a non-negative N such that the current line-number equals FIRST + (N * STEP). Thus, to select the odd-numbered lines, one would use`1~2'.
- seq 10 | sed '0~3a\=='
-
- seq 10 | awk 'ORS=NR%3?"\n":"\n==\n"
[解析]
每隔三行就在后面添加一个分割符。