我有一个youtube-dl
命令,当运行时输出两个新行,我试图将每一个捕获到变量中。
这是我到目前为止所拥有的,但由于某种原因我无法理解,它只捕获第一行输出而丢弃第二行($audio
为空)。
read -r video audio <<< "$(youtube-dl -g --youtube-skip-dash-manifest https://www.youtube.com/watch?v=-gcvAxJJiGo)"
如何修复它以使其捕获两个输入,或者,是否有更可靠的方法可以做到这一点?
read
设计读取一行。读取两行的最简单方法是使用read
两次。read -u
不是便携式的,也不是<(…)
。他们在 Bash 中工作,你的问题被标记为bash,所以他们应该为你工作。?
请注意,如果您在其中有目录https:/www.youtube.com/
和匹配的文件,我会单引号包含包含的字符串。这只是远程可能的,仍然引用不应被视为通配符的通配符在任何 shell 中都是很好的做法。比较Zsh 中发生的情况。当您想阅读一行时,通常空
IFS
是正确的。这个其他答案不使用
exec
,很好。在将使用它的工具之前以视觉方式设置输入是我个人的偏好(比较这个答案)。不幸的是<(…) { …; }
不起作用。有了exec
我就可以做到。或者我可以通过管道连接
youtube-dl
到脚本的其余部分来做到这一点。通常 Bash
{ … }
在子 shell 中运行片段,因此变量在}
.shopt -s lastpipe
可以更改此设置(如果作业控制未激活)。没有lastpipe
(不可移植),整个方法是可移植的,但您需要运行使用{ }
块内变量的所有内容,因为您不知道它是否是子外壳(POSIX 允许这两种行为)。注意块内的标准输入来自youtube-dl
; 如果块内的任何内容需要使用整个脚本的标准输入,那么您可能需要exec
无论如何都要使用来处理描述符。另一种便携式方法是这样的:
代替
head
andtail
您可以使用sed
orawk
来隔离正确的行。请注意,此解决方案不使用
read
. 如果您需要功能,read
那么这不是方法。read
不过,在这种特殊情况下,您似乎不需要 的功能。即使
youtube-dl
非常快,我也不建议video="$(youtube-dl … | head -n 1)"
plusaudio="$(youtube-dl … | …)"
。如果video
和audio
来自相同的input
,来自 的相同调用youtube-dl
,那么它们在工具允许的范围内是连贯的。否则,它们可能不连贯。坦率地说,我不知道他们可能有多不连贯(如果有的话);但我知道一般来说最好查询一次(例如date +%H:+%M
应该始终连贯;date +%H; date +%M
如果您在一个完整小时之前运行它会令您感到惊讶)。read -r video audio
会将第一个($IFS 分隔的)单词捕获到“video”中,并将该行的其余部分捕获到“audio”中捕获 2 行的另一种方法是使用命令分组和进程替换:
之间的空格
< <(
很重要:a<(process substitution)
就像文件一样,因此您<
用于重定向来自文件的输入。我突然想到:另一种读取输出行的方法是
mapfile
命令: