假设我只需要输出的前 5 行用于记录目的。我还需要知道日志是否以及何时被截断。
我正在尝试使用head
来完成这项工作,seq
下面的命令输出 20 行被截断的 20 行head
,并且我echo
是截断信息:
> seq -f 'log line %.0f' 20 | head -n 5 && echo '...Output truncated. Only showing first 5 lines...'
log line 1
log line 2
log line 3
log line 4
log line 5
...Output truncated. Only showing first 5 lines...
但是,如果seq
命令输出少于 5 行,使用与上述相同的构造,我会得到错误的“截断”状态:
seq -f ' log line %.0f' 3 | head -n 5 && echo '...Output truncated. Only showing first 5 lines...'
log line 1
log line 2
log line 3
...Output truncated. Only showing first 5 lines...
命令(或其他工具)有没有办法head
告诉我它是否截断了任何内容,以便我只在需要时显示“...截断...”消息?
警告说明:
当你这样做时:
如果输出被截断,则可能导致
cmd
被 SIGPIPE 杀死,如果它在head
退出后写入更多行。如果它不是您想要的,如果您想cmd
在之后继续运行,即使它的输出被丢弃,您也需要读取但丢弃剩余的行而不是在输出 10 行后退出(例如,使用sed '1,10!d'
或awk 'NR<=10'
代替head
)。因此,对于两种不同的方法:
输出被截断,cmd 可能被杀死
请注意,在开始处理之前,它的
mawk
实现会累积一个充满缓冲区的输入,因此在写入一个充满缓冲区(我的系统 AFAICT 上为 8KiB)的数据之前,它可能不会被杀死。这可以通过使用该选项来解决。awk
cmd
-Winteractive
一些
sed
实现还提前读取了一行(以便在使用地址时能够知道哪一行是最后一行$
),因此只有在输出第 7行cmd
后才能将其杀死。输出被截断,其余被丢弃,因此 cmd 不会被杀死
您可以使用 AWK:
这会按原样打印前五行(如果有);如果它看到超出此范围的行,则输出截断消息并退出。
如果要实现条件处理,可以指定要使用的退出代码: