将字符串发送到标准输入

问题描述 投票:0回答:8

有没有办法在 bash 中有效地做到这一点:

/my/bash/script < echo 'This string will be sent to stdin.'

我知道我可以通过管道传输 echo 的输出,如下所示:

echo 'This string will be piped to stdin.' | /my/bash/script
bash redirect stdin io-redirection
8个回答
351
投票

您可以使用一行

heredoc

cat <<< "This is coming from the stdin"

以上相同

cat <<EOF
This is coming from the stdin
EOF

或者您可以重定向命令的输出,例如

diff <(ls /bin) <(ls /usr/bin)

或者你可以读作

while read line
do
   echo =$line=
done < some_file

或者简单地

echo something | read param

79
投票

你很接近

/my/bash/script <<< 'This string will be sent to stdin.'

对于多行输入,here-docs 适合:

/my/bash/script <<STDIN -o other --options
line 1
line 2
STDIN

编辑 致评论:

要实现二进制输入,请说

xxd -r -p <<BINARY | iconv -f UCS-4BE -t UTF-8 | /my/bash/script
0000 79c1 0000 306f 0000 3061 0000 3093 0000 3077 0000 3093 0000 304b 0000 3093 0000 3077 0000 3093 0000 306a 0000 8a71 0000 306b 0000 30ca 0000 30f3 0000 30bb
0000 30f3 0000 30b9 0000 3092 0000 7ffb 0000 8a33 0000 3059 0000 308b 0000 3053 0000 3068 0000 304c 0000 3067 0000 304d 0000 000a
BINARY

如果您将

cat
替换为
/my/bash/script
(或者确实删除了最后一个管子),则会打印:

私はちんぷんかんぷんな話にナンセンスを翻訳することができ

或者,如果你想要一些更极客的东西:

0000000: 0000 0000 bef9 0e3c 59f8 8e3c 0a71 d63c  .......<Y..<.q.<
0000010: c6f2 0e3d 3eaa 323d 3a5e 563d 090e 7a3d  ...=>.2=:^V=..z=
0000020: 7bdc 8e3d 2aaf a03d b67e b23d c74a c43d  {..=*..=.~.=.J.=
0000030: 0513 d63d 16d7 e73d a296 f93d a8a8 053e  ...=...=...=...>
0000040: 6583 0e3e 5a5b 173e 5b30 203e 3d02 293e  e..>Z[.>[0 >=.)>
0000050: d4d0 313e f39b 3a3e 6f63 433e 1c27 4c3e  ..1>..:>ocC>.'L>
0000060: cde6 543e 59a2 5d3e 9259 663e 4d0c 6f3e  ..T>Y.]>.Yf>M.o>
0000070: 60ba 773e cf31 803e ee83 843e 78d3 883e  `.w>.1.>...>x..>
0000080: 5720 8d3e 766a 913e beb1 953e 1cf6 993e  W .>vj.>...>...>
0000090: 7a37 9e3e c275 a23e dfb0 a63e bce8 aa3e  z7.>.u.>...>...>
00000a0: 441d af3e 624e b33e 017c b73e 0ca6 bb3e  D..>bN.>.|.>...>
00000b0: 6fcc bf3e 15ef c33e e90d c83e d728 cc3e  o..>...>...>.(.>
00000c0: c93f d03e ac52 d43e 6c61 d83e f36b dc3e  .?.>.R.>la.>.k.>
00000d0: 2f72 e03e 0a74 e43e 7171 e83e 506a ec3e  /r.>.t.>qq.>Pj.>
00000e0: 945e f03e 274e f43e f738 f83e f11e fc3e  .^.>'N.>.8.>...>
00000f0: 0000 003f 09ee 013f 89d9 033f 77c2 053f  ...?...?...?w..?
0000100: caa8 073f 788c 093f 776d 0b3f be4b 0d3f  ...?x..?wm.?.K.?
0000110: 4427 0f3f 0000 113f e8d5 123f f3a8 143f  D'.?...?...?...?
0000120: 1879 163f 4e46 183f 8d10 1a3f cad7 1b3f  .y.?NF.?...?...?
0000130: fe9b 1d3f 1f5d 1f3f 241b 213f 06d6 223f  ...?.].?$.!?.."?
0000140: bb8d 243f 3a42 263f 7cf3 273f 78a1 293f  ..$?:B&?|.'?x.)?
0000150: 254c 2b3f 7bf3 2c3f 7297 2e3f 0138 303f  %L+?{.,?r..?.80?
0000160: 22d5 313f ca6e 333f                      ".1?.n3?

4 字节二进制浮点数中前 90 度的正弦值是多少


24
投票

解决方案

您想要 (1) 在一个进程中创建 stdout 输出(如

echo '…'
),并 (2) 将该输出重定向到另一进程的 stdin 输入,但 (3) 不使用 bash 管道机制。这是满足所有三个条件的解决方案:

/my/bash/script < <(echo 'This string will be sent to stdin.')

<
是标准输入的正常输入重定向。
<(…)
是 bash 进程替换。粗略地说,它使用替换命令的输出创建一个
/dev/fd/…
文件,并传递该文件名来代替
<(…)
,例如此处的
script < /dev/fd/123
。详情请参阅此答案

与其他解决方案的比较

  • 发送到标准输入的单行heredoc仅允许发送静态字符串,而不是其他命令的输出。

    
    

  • 单独的进程替换(例如
  • script <<< 'string'

    )不会向标准输入发送任何内容。相反,进程输出被保存到文件中,并且其路径作为命令行参数传递。对于上面的示例,这相当于

    diff <(ls /bin) <(ls /usr/bin)
    ,这是一个命令,其中
    diff /dev/fd/10 /dev/fd/11
    不接收来自 stdin 的输入。
    
    

  • 使用案例

我喜欢这一点,与管道机制不同,

diff

机制允许将命令放在前面,然后将所有输入放在后面,这是从命令行选项输入的标准。

但是,除了命令行美观之外,在某些情况下无法使用管道机制。例如,当像 

< <(…)

ssh
这样的特定命令被期望作为另一个命令的参数时,例如在
 这个带有 
scp
 的示例中。


3
投票
sshpass


read



2
投票
使用户能够在程序中键入多行,而该输入不会保存在历史记录中,也不会在 
cat | /my/bash/script

中可见。以换行符结束,然后按 Ctrl + D 结束

ps
    


0
投票

我用:

cat

Bash 在子 shell 中运行命令并将输出传递到原始命令的 stdin。


0
投票

/my/bash/script $(echo 'This string will be sent to stdin.')



-4
投票
别名可以也不能处理管道标准输入...

这里我们创建 3 行输出

$ read x <<< "x-value > " $ echo $x x-value
然后我们将输出通过管道传输到 sed 命令的标准输入,将它们全部放在
一行

$ echo -e "line 1\nline 2\nline 3" line 1 line 2 line 3
如果我们定义同一个 sed 命令的别名

$ echo -e "line 1\nline 2\nline 3" | sed -e ":a;N;\$!ba ;s?\n? ?g" line 1 line 2 line 3
我们可以将输出通过管道传输到别名的标准输入,它的行为
一模一样

$ alias oline='sed -e ":a;N;\$!ba ;s?\n? ?g"'
当我们尝试将别名定义为函数时,问题就出现了

$ echo -e "line 1\nline 2\nline 3" | oline line 1 line 2 line 3
将别名定义为函数会破坏管道

$ alias oline='function _oline(){ sed -e ":a;N;\$!ba ;s?\n? ?g";}_oline'
© www.soinside.com 2019 - 2024. All rights reserved.