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 / 问题 / 495324
Accepted
JoL
JoL
Asked: 2019-01-19 09:42:31 +0800 CST2019-01-19 09:42:31 +0800 CST 2019-01-19 09:42:31 +0800 CST

为什么`... | 当 `... > >(sed 's/^/stdout: /')` 不打印时,sed 's/^/stdout: /'` 在空标准输入上打印?

  • 772

我试图了解是什么导致了我认为在功能上等效的这两个构造的差异:

$ ( echo foo >&2 ) 2> >(sed 's/^/stderr: /') | sed 's/^/stdout: /'   
stdout: stderr: foo
$ ( echo foo >&2 ) 2> >(sed 's/^/stderr: /') > >(sed 's/^/stdout: /')
stderr: foo

编辑:如果我理解 user1133275 正确,他建议>(sed 's/^/stdout: /')不要运行,除非子外壳( echo foo >&2 )输出到标准输出。但是,这将意味着以下内容:

$ ( echo foo >&2 ) 2> >(sed 's/^/stderr: /') > >(echo baz)   
baz
stderr: foo

不应该显示baz。

编辑 2:也许也很有趣, sed 不会stdout:在空输入上输出,即使是通过管道传输的:

$ sed 's/^/stdout: /' < /dev/null
$ printf "" | sed 's/^/stdout: /'
$
bash sed
  • 4 4 个回答
  • 1857 Views

4 个回答

  • Voted
  1. Best Answer
    Kusalananda
    2019-01-19T10:23:41+08:002019-01-19T10:23:41+08:00

    你的第一个命令,

    ( echo foo >&2 ) 2> >(sed 's/^/stderr: /') | sed 's/^/stdout: /'
    

    以简化形式(使用临时文件保存由 生成的数据echo):

    { echo foo 2>file >&2; sed 's/^/stderr: /' file; } | sed 's/^/stdout: /'
    

    即,第一个sed读取从标准错误产生的内容echo并写入标准输出,第二个sed读取并修改它。

    你的第二个命令,

    ( echo foo >&2 ) 2> >(sed 's/^/stderr: /') > >(sed 's/^/stdout: /')
    

    以简化的形式,

    echo foo 2>file >&2; sed 's/^/stderr: /' file; sed 's/^/stdout: /' </dev/null
    

    在这里,sed获得标准错误输出的那个产生输出,而另一个sed获得标准输出输出(什么都没有)的那个不产生任何输出(因为它没有得到任何输入并且因为它没有插入或附加任何数据) .

    另一种表述方式:

    第一个命令:

    ( echo foo >&2 ) 2>file
    sed 's/^/stderr: /' file | sed 's/^/stdout: /'
    

    第二条命令:

    ( echo foo >&2 ) 2>file >otherfile
    sed 's/^/stderr: /' file
    sed 's/^/stdout: /' otherfile
    

    简而言之,第二sed个命令中的第二个从不读取任何内容。特别是,它不像第sed一个命令那样从第一个命令中读取输出。


    使用极其简化的符号,第一个命令类似于

    utility-writing-to-stderr 2> >(something1) | something2
    

    wheresomething1写入标准输出,由something2.

    第二个命令,使用相同的符号,

    utility-writing-to-stderr 2> >(something1) >(something2)
    

    即something1,something2甚至没有相互连接,并且something2无法以任何方式读取something1正在产生的内容。此外,由于utility-writing-to-stderr不会在其标准输出流上产生任何内容,something2因此不会从其标准输入中读取任何内容。

    • 3
  2. user1133275
    2019-01-19T09:52:04+08:002019-01-19T09:52:04+08:00

    >由于 shell 中的重定向顺序,正在运行( echo foo >&2 )并且 |正在运行。>(sed 's/^/stderr: /')例如

    $ ( echo foo >&2 ) 2> >(sed 's/^/stderr: /' > >(sed 's/^/stdout: /') )
    stdout: stderr: foo
    

    或者

    $ ( echo foo >&2 ) 2> >(sed 's/^/stderr: /') | cat > >(sed 's/^/stdout: /')
    stdout: stderr: foo
    

    vs 明确的例子问题中的顺序

    $ ( ( echo foo >&2 ) > >(sed 's/^/stdout: /') ) 2> >(sed 's/^/stderr: /')
    stderr: foo
    

    或者

    $ ( ( echo foo >&2 )  | sed 's/^/stdout: /' ) 2> >(sed 's/^/stderr: /')
    stderr: foo
    
    • 1
  3. user232326
    2019-01-19T13:55:20+08:002019-01-19T13:55:20+08:00

    这两个命令是不等价的。

    命令(1):

    ( one ) 2> >(two) | three 
    

    将 (std) 输出发送two到three(更多关于 stderr 下面)。

    命令(2):

    ( one ) 2> >(two) > >(three)
    

    将 (std) 输出发送one到three(stderr go to two)。


    痛苦的细节是:

    命令 1:( one ) 2> >(two) | three

    • 命令three已启动 ( sed 's/^/stdout: /') 等待标准输入上的输入。
    • 2将( stderr)重定向到stdinof two。
    • 发生内部重定向one( echo foo >&2)。
    • 该命令被执行发送foo到stderr(没有到标准输出)。
    • 单词从 ( of )foo重定向到 ( of )。stderronestdintwo
    • 两个中的命令被执行 ( sed 's/^/stderr: /')
    • two( )现在修改后的输出stderr: foo被发送到 的标准输出two。
    • 当前的标准输出two通过管道传输到three.
    • three从一开始就等待输入的命令得到输出。
    • 输出被修改添加一个前导stdout:。
    • 最后一个字符串stdout: stderr: foo转到 tty。

    命令2:(一)2>>(二)>>(三)

    • 最后一个重定向 ( >) 首先构建。
    • 该命令仅在( )three上开始等待输入。stdin >()
    • 该命令仅three获取.one
    • 命令two开始等待输入。
    • 该命令two(仅)获取one.
    • one 的内部重定向将 stdout 连接one到其stderr.
    • 该命令one被执行发送foo到stderr(of one)。
    • foo去two
    • two更改接收到的字符串stderr: foo并将其发送到tty。
    • three从 的标准输出接收一个空输入one。
    • three什么都不打印(因为没有要处理的行)。

    理解这个命令:

    printf '' | sed 's/^/initial: /'
    

    没有输出(也没有办法让它工作)。


    更改顺序:命令 3:(一)>>(三)2>>(二)

    two将go to three(而不是)详细信息的输出留作tty练习。

    • 1
  4. Cbhihe
    2019-01-20T11:18:08+08:002019-01-20T11:18:08+08:00

    我赞成@Kusalananda 非常明确的答案,但我冒险(迟来)将其添加到他的答案中,作为工作中流程替换的一个小例子:

    您的第二个命题将适用于一个小的印刷变化(及其错误逻辑的相对较大的变化):

      $ ( echo foo >&2 ) 2> >(sed 's/^/stderr: /') > >(sed 's/^/stdout: /')
      stderr: foo    # for the well explained reason above
    

    但:

        v                                          vv
      $ ( (echo foo >&2) 2> >(sed 's/^/stderr: /') )> >(sed 's/^/stdout: /')  
      stdout: stderr: foo
    

    因为( )>如图所示添加(或者>( ),两者都可以正常工作)实际上允许您通过进程替换创建一个文件,其输出可以作为输入重定向到>(sed 's/^/stdout: /').

    我考虑添加这个来说明如何将流程替换嵌入到另一个流程替换中。为了可读性,我不会将其推向更多嵌入级别。但是如果出于任何原因您需要避免使用管道和相关的shell,这是一个很好的解决方案。

    • 0

相关问题

  • 通过命令的标准输出以编程方式导出环境变量[重复]

  • 从文本文件传递变量的奇怪问题

  • 虽然行读取保持转义空间?

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

  • 运行一个非常慢的脚本直到它成功

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