gpt4 book ai didi

linux - BASH 语法检查 Debug模式故障?

转载 作者:可可西里 更新时间:2023-11-01 11:48:35 30 4
gpt4 key购买 nike

我们可以使用 bash -n script.sh 来验证 shell 脚本的语法。然而,当我试图测试这个功能时,我注意到并不是所有的语法错误都可以通过这个选项找到。

例如:

root@ubuntu:~/testenv# cat test 
#!/bin/bash
SEND=1
if [ "$SEND" -eq 0 ]
echo no
fi

现在,让我们测试脚本:

root@ubuntu:~/testenv# bash -n test 
test: line 5: syntax error near unexpected token `fi'
test: line 5: `fi'

它工作正常。但是,如果我只是删除其中一个括号:

root@ubuntu:~/testenv# cat test     
#!/bin/bash
SEND=1
if [ "$SEND" -eq 0
then
echo no
fi
root@ubuntu:~/testenv# bash -n test
root@ubuntu:~/testenv#

什么都没发生!

我还查看了 bash 的手册页,它描述了“-n”是:

 -n               Read  commands  but  do not execute them.  This may be used to check a
shell script for syntax errors. This is ignored by interactive
shells.

它是一个脚本文件,所以它不应该是一个“交互式 shell”,对吧?那么,这怎么会发生呢?

最佳答案

我猜您遇到了 shell 实现单括号条件的方式的一个非常奇怪的怪癖:[ 是一个命令,而不是一个特殊字符。查看您的系统可执行目录(可能是 /usr/bin),您会发现一个名为 [ 的可执行文件,它实现了这个命令。当你写类似的东西时

[ "$SEND" -eq 0 ]

那么您实际上是在使用四个参数调用命令 [:

  1. $SEND的值
  2. 字符串-eq
  3. 字符串0
  4. 字符串]

命令[检查最后一个参数是](否则看起来会很奇怪),然后将剩余的参数放在一起形成一个条件并返回结果测试条件。

现在,因为 [ 是一个命令,所以使用您喜欢的任何参数集调用该命令都不是语法错误。当然,如果您省略尾随的 ],您会得到一个错误,但该错误来自 shell 命令 [不是 .这意味着你必须实际运行脚本才能得到错误——语法检查器不会发现任何错误。就 bash 而言,[ 只是一个命令名称,与 my_custom_conditional_test 没有区别,如果您要编写

my_custom_conditional_test "$SEND" -eq 0

很明显这很好,对吧? Bash 以同样的方式思考 [

我应该指出,为了提高效率,bash 实际上并不使用 可执行文件/usr/bin/[;它有自己的 [ 内置实现。但是人们希望 [ 以相同的方式运行,而不管它是否内置在 shell 中,因此 Bash 语法检查器不能给它自己的 [ 特殊待遇。由于调用没有尾随 ]/usr/bin/[ 不会是语法错误,因此调用内置 不会是语法错误>[ 没有 ]

您可以将它与 [[ 进行对比,它或多或少地做同样的事情(测试条件),但是 shell 赋予了特殊含义。 [[ 是 shell 语法中的特殊标记,而不是命令。如果你写 [[ 而不是 [,并且你省略了相应的尾随 ]],你打赌 Bash 会提示语法错误.

关于linux - BASH 语法检查 Debug模式故障?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41438848/

30 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com