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 / 问题 / 748078
Accepted
furynerd
furynerd
Asked: 2023-06-06 15:59:54 +0800 CST2023-06-06 15:59:54 +0800 CST 2023-06-06 15:59:54 +0800 CST

检索标准输出重定向到 /dev/null

  • 772

有没有办法检索重定向到 /dev/null 的标准输出?我试过了tail -f /proc/{PID}/fd/1,除了重定向到/dev/null.

IE

tail -f /proc/${cmd_pid}/fd/1

工作cmd > log.txt但不cmd > /dev/null

- - - - - - 更新 - - - - - -

事实上,log.txt如果总是记录标准输出,我的问题将会非常大。如果我可以控制何时记录或停止日志(而不停止进程cmd本身),那将是最好的。

所以如果可以将标准输出重定向到某个tmp不占用资源的文件系统,当我需要时我可以检索标准输出?

process
  • 3 3 个回答
  • 65 Views

3 个回答

  • Voted
  1. symcbean
    2023-06-06T16:35:24+08:002023-06-06T16:35:24+08:00

    不,您不能更改文件句柄。

    所以如果可以将标准输出重定向到一些不占用资源的 tmp 文件系统,当我需要时我可以检索标准输出?

    不,如果不将数据存储在某处,就无法存储数据。

    可能有人写了一个循环缓冲区——它可以让你看到(比如说)最后 100k 的数据。或者,如果程序通常无头运行,您可以将输出发送到屏幕并通过screen.

    • 0
  2. muru
    2023-06-06T17:09:18+08:002023-06-06T17:09:18+08:00

    当我不检索它时,标准输出可以被丢弃,当我开始检索时,标准输出可以重定向到日志文件。

    这可以通过将输出通过管道传输到 shell 脚本来完成,如下所示:

    #! /bin/sh
    # a-script.sh
    while IFS= read -d input
    do
      printf "%s\n" input >> /some/file
    done 
    

    请注意,重定向是在循环内完成的,特别是为了在每次迭代中重新打开输出文件。然后,开始:

    ln -s /dev/null /some/file
    cmd | a-script.sh
    

    然后当你想要日志时:

    ln -sf /some/log/file /some/file
    

    循环的下一次迭代将开始写入/some/log/file而不是写入到/dev/null。

    如果您的命令大量记录,这可能会导致严重的性能损失。

    • 0
  3. Best Answer
    Stéphane Chazelas
    2023-06-06T17:16:43+08:002023-06-06T17:16:43+08:00

    当你这样做时cmd > /dev/null,shell 在一个新的子进程中打开 fd 1/dev/null并执行cmd。

    cmd执行一些操作write(1, "output"...)以将数据写入 stdout(或者可能是其他写入系统调用,例如send,sendmsg或pwrite),而不管 fd 是在什么上打开的。

    在 Linux 上,/proc/$pid_of_cmd/fd/1将只是一个神奇的符号链接/dev/null,文件 fd 1 在其中打开,因此tail -f that与tail -f /dev/null.

    您可以拦截这些write()系统调用strace并解码结果:

    strace -zqqs99999 -xxa0 -o /dev/stdout -e write -p "$pid" |
       perl -ne 'print chr(hex($_)) for /\\x(..)/g'
    

    或者:

    strace -zqqs99999 -xxa0 -o /dev/stdout -e write -p "$pid" |
       perl -ne 'if (/^write\(1,/) {print chr(hex($_)) for /\\x(..)/g}'
    

    仅用于write()标准输出 (fd 1) 的 s。或者:

    strace -yzqqs99999 -xxa0 -o /dev/stdout -e write -p "$pid" |
       perl -ne 'if (/^\Qwrite(1<\x2f\x64\x65\x76\x2f\x6e\x75\x6c\x6c>,\E "(.*)"/) {
         print chr(hex($_)) for $1 =~ /\\x(..)/g}'
    

    仅用于write()stdout (fd 1) 的 s 并且仅当 stdout 在/dev/null.

    请注意,如果您将 的输出重定向perl到 tty 设备以外的其他设备,它将开始按块而不是按行进行缓冲,因此您只会看到几千字节的批量输出。$|您可以通过将变量设置为来禁用该缓冲1:

    strace -zqqs99999 -xxa0 -o /dev/stdout -e write -p "$pid" |
       perl -ne 'BEGIN{$| = 1}
                 print chr(hex($_)) for /\\x(..)/g' > file
    

    cmd > /dev/null或者,不要做,而是做cmd | cat > /dev/null。

    Thencmd的标准输出将是一个管道,在 Linux 上,/proc/$pid_of_cmd/fd/1它的行为就像一个命名管道,所以如果你这样做:

    kill -s STOP "$pid_of_cat"
    cat "/proc/$pid_of_cmd/fd/1"
    

    您实际上会将管道重定向到您的新cat命令。

    如果你终止那个新的cat,你会想要恢复另一个 ( kill -s CONT "$pid_of_cat") 以重新打开/dev/null那个管道上的水龙头。

    要实际更改进程的标准输出位置,您可以附加一个调试器:

    gdb --pid "$pid_of_cmd"
    

    然后在 gdb 中:

    call close(1)
    p open("/path/to/some/file", 0x241, 0600)
    detach
    

    (O_WRONLY|O_TRUNC|O_CREAT 为 0x241)

    检查是否open()返回 fd 1。如果没有,您可能需要调用dup2()和一个额外的close().

    • 0

相关问题

  • 如何用名称而不是“ps”中的数字替换用户 ID 和组 ID?

  • 如何获取进程的补充组 ID?

  • `/proc/irq/.../spurious` 包含什么?

  • 父进程可以改变其子进程的环境吗?

  • “弹出”如何让进程关闭文件句柄?

Sidebar

Stats

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

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

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

    • 4 个回答
  • Marko Smith

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

    • 5 个回答
  • Marko Smith

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

    • 3 个回答
  • 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
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +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

热门标签

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