AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 435413
Accepted
Heath Raftery
Heath Raftery
Asked: 2018-04-04 19:42:34 +0800 CST2018-04-04 19:42:34 +0800 CST 2018-04-04 19:42:34 +0800 CST

在管道链中使用 jq 不会产生输出

  • 772

当输出被重定向时需要一个显式过滤器的问题jq在整个网络上都有讨论。但是如果是管道链的一部分,我无法重定向输出jq,即使在使用显式过滤器时也是如此。

考虑:

touch in.txt
tail -f in.txt | jq '.f1'
# in a different terminal:
echo '{"f1":1,"f2":2}' >> in.txt
echo '{"f1":3,"f2":2}' >> in.txt

正如预期的那样,该命令在原始终端中的输出jq是:

1
3

但是,如果我在命令末尾添加任何类型的重定向或管道jq,输出将变为静音:

rm in.txt
touch in.txt
tail -f in.txt | jq '.f1' | tee out.txt
# in a different terminal:
echo '{"f1":1,"f2":2}' >> in.txt
echo '{"f1":3,"f2":2}' >> in.txt

第一个终端没有输出,out.txt 为空。

我已经尝试了数百种变体,但这是一个难以捉摸的问题。我发现的唯一解决方法是通过mosquitto_sub和 The Things Network 发现的(我也发现了这个问题),是将 tail和jq 函数包装在 shell 脚本中:

#!/bin/bash
tail -f $1 | while IFS='' read line; do
echo $line | jq '.f1'
done

然后:

./tail_and_jq.sh | tee out.txt
# in a different terminal:
echo '{"f1":1,"f2":2}' >> in.txt
echo '{"f1":3,"f2":2}' >> in.txt

果然,输出出现了:

1
3

这是jq通过 Homebrew 安装的最新版本:

$ echo $SHELL
/bin/bash
$ jq --version
jq-1.5
$ brew install jq
Warning: jq 1.5_3 is already installed and up-to-date

这是jq我对管道链的理解中的(大部分未记录的)错误吗?

pipe io-redirection
  • 2 2 个回答
  • 13970 Views

2 个回答

  • Voted
  1. Best Answer
    Kusalananda
    2018-04-04T21:31:19+08:002018-04-04T21:31:19+08:00

    jq当其标准输出通过管道传输时,来自的输出被缓冲。

    要请求jq在每个对象之后刷新其输出缓冲区,请使用其--unbuffered选项,例如

    tail -f in.txt | jq --unbuffered '.f1' | tee out.txt
    

    从jq手册:

    --unbuffered

    在打印每个 JSON 对象后刷新输出(如果您将慢速数据源传输到其他地方jq并jq在其他地方传输 的输出,这很有用)。

    • 36
  2. filbranden
    2018-04-04T20:47:13+08:002018-04-04T20:47:13+08:00

    您在这里看到的是 C stdio 缓冲在起作用。它将输出存储在缓冲区中,直到达到某个限制(可能是 512 字节,或 4KB 或更大),然后一次性发送所有内容。

    如果 stdout 连接到终端,则此缓冲会自动禁用,但是当它连接到管道时(例如在您的情况下),它将启用此缓冲行为。

    禁用/控制缓冲的常用方法是使用该setvbuf()函数(有关更多详细信息,请参见此答案),但这需要在其jq自身的源代码中完成,因此对您来说可能不实用......

    有一种解决方法......(有人可能会说是黑客。)有一个名为“unbuffer”的程序,它与“expect”一起分发,可以创建一个伪终端并将其连接到一个程序。因此,即使jq仍将写入管道,它也会认为它正在写入终端,并且缓冲效果将被禁用。

    安装“expect”包,如果你还没有它,它应该带有“unbuffer”......例如,在 Debian(或 Ubuntu)上:

    $ sudo apt-get install expect
    

    然后你可以使用这个命令:

    $ tail -f in.txt | unbuffer -p jq '.f1' | tee out.txt
    

    有关“unbuffer”的更多详细信息,另请参阅此答案,您也可以在此处找到手册页。

    • 7

相关问题

  • 如何将 airodump-ng 的输出保存到文件中?

  • 将句柄传递给标准输入下行管道

  • 如何通过管道传输 bash 命令并保持 Ctrl+C 正常工作?

  • 为什么管道`mysql`到'tail'会改变输出格式?

  • `tee` 和 `bash` 进程替换顺序

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    ssh 无法协商:“找不到匹配的密码”,正在拒绝 cbc

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    如何卸载内核模块“nvidia-drm”?

    • 13 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Wong Jia Hau ssh-add 返回:“连接代理时出错:没有这样的文件或目录” 2018-08-24 23:28:13 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST
  • Martin Hope
    Bagas Sanjaya 为什么 Linux 使用 LF 作为换行符? 2017-12-20 05:48:21 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve