当命令是if
语句的一部分时,如何在shell中的多行上拆分命令?
这有效:
if ! fab --fabfile=.deploy/fabfile.py --forward-agent --disable-known-hosts deploy:$target; then rc=1
fi
这不起作用:
# does not work:
if ! fab --fabfile=.deploy/fabfile.py \
--forward-agent \
--disable-known-hosts deploy:$target; then
rc=1
fi
而不是执行整个命令,我得到:
./script.sh: line 73: --forward-agent: command not found
更重要的是,我对Bash的理解中缺少哪些内容可以帮助我理解未来的类似问题?
答案 0 :(得分:483)
如果在反斜杠之后和换行符之前有空格(空格或制表符),则行继续将失败。没有这样的空格,你的例子对我来说很好:
$ cat test.sh
if ! fab --fabfile=.deploy/fabfile.py \
--forward-agent \
--disable-known-hosts deploy:$target; then
echo failed
else
echo succeeded
fi
$ alias fab=true; . ./test.sh
succeeded
$ alias fab=false; . ./test.sh
failed
从评论中提出了一些细节:shell中的行继续反斜杠并不是特殊情况;它只是一般规则的一个实例,反斜杠“引用”紧随其后的字符,防止它通常会受到任何特殊处理。在这种情况下,下一个字符是换行符,并且阻止的特殊处理是终止命令。通常情况下,一个引用的字符在字面上包含在命令中;反向删除换行符将被完全删除。但除此之外,机制是一样的。反斜杠只引用紧随其后的字符;如果该字符是空格或制表符,您只需获得带引号的空格或制表符,任何后续换行符都不会被加引号。
答案 1 :(得分:43)
对于Windows / WSL / Cygwin等用户:
确保您的行结尾是标准的Unix换行符,即\n
(LF)。
使用Windows行结尾\r\n
(CRLF)行结尾将破坏命令行中断。
这是因为在Windows行结尾的行末尾有\
会转换为
\
\r
\n
。
正如Mark上面正确解释的那样:
如果在反斜杠之后和换行符之前有空格,则行继续将失败。
这不仅包括空格()或制表符(
\t
),还包括回车符(\r
)。