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
    • 最新
    • 标签
主页 / ubuntu / 问题 / 1153258
Accepted
AcBap
AcBap
Asked: 2019-06-23 15:37:46 +0800 CST2019-06-23 15:37:46 +0800 CST 2019-06-23 15:37:46 +0800 CST

为什么ksh中的“[[”和“-e xxx”之间需要空格?

  • 772

例如,以下命令不起作用:

if [[-e xyz]]; then echo File exists;fi

ksh 给出以下错误

[[-e: command not found

那是因为“[[-”是模棱两可的吗?

command-line syntax ksh
  • 5 5 个回答
  • 2373 Views

5 个回答

  • Voted
  1. Best Answer
    Sergiy Kolodyazhnyy
    2019-06-23T21:41:11+08:002019-06-23T21:41:11+08:00

    最简单的解释是,因为在手册中它显示为在and和 close[[ expression ]]之间必须有空格。但当然,我们可以尝试更深入地研究它。Shell 的语法有些复杂,并且严重依赖“分词”的概念。特别是, ksh(1)手册指出:[[expression]]

    shell 通过将其分解为单词来开始解析其输入。单词是字符序列,由不带引号的空白字符(空格、制表符和换行符)或元字符(<、>、|、;、&、(和))分隔。除了分隔单词,空格和制表符被忽略,而换行符通常分隔命令。

    因此,如手册中所述,字符序列[[-e被视为外壳词。ksh将在内置命令和特殊运算符( like foror while)列表中查找此类命令,然后查找外部命令 - 瞧 - 未找到,因此出现有关未找到命令的错误消息。

    在这种情况下[[也不是特殊的元字符。如果是,则-ein 部分[[-e将被视为外壳词,而不是其[[自身的参数。但是,[[被描述为复合命令,并且手册说如下:

    复合命令是使用以下保留字创建的——这些字只有在它们没有被引用并且被用作命令的第一个字时才被识别(即,它们不能在参数分配或重定向之前):

    因此,为了[[被识别为复合命令,它必须是命令或命令列表的第一个单词,根据先前对“单词”的定义,这意味着它必须用空格、制表符或换行符与其他命令分隔词/论据。


    您在评论中问过:“为什么解析器不能在找到“[[”模式后立即停止,并将其视为条件命令的开始?” 简短的回答可能是因为 1)[语法的影响,因为它最初是一个外部命令,并且对于 POSIX 标准遵从性即使在今天也作为外部命令存在,以及 2) 因为 shell 解析器是这样构建的。Shell 解析器可以识别其他非空格分隔的特殊字符: echo $((2+2))并且(echo foobar)工作得很好。也许将来当ksh开发恢复或出现分叉或克隆(如mksh或pdksh)时,有人将在[[-e.

    也可以看看:

    • 就 POSIX 语法而言,[[称为“保留字”(参见Shell Command Language 的第 2.9 节)。根据 POSIX 定义,保留字必须用空格分隔。相比之下,运算符是运算符,第 2.9 节中的标准状态“表示包括在某些不需要s 的地方(当令牌之一是运算符时)的令牌(之间的间距”。<blank>请参阅相关讨论: POSIX Shell Grammar:为什么 Brace Group 需要第一个空格,但 Subshel​​l 不需要?
    • 15
  2. Rinzwind
    2019-06-23T16:15:13+08:002019-06-23T16:15:13+08:00

    需要注意的是,它[是一个命令,bash 和 ksh 中的 shell 关键字[[是基于[.

    [[的使用方式与[. 有时您甚至可以在行为不变的情况下[ ]替换为。[[ ]]与[任何命令一样,命令及其参数之间必须有空格。这同样适用于[[。

    我们以前只有/usr/bin/[. 现在大多数 shell 都[内置了以提高效率——但语法是相同的。在提供的shell 中,它可以[[作为.[

    以下是[bash 中的描述:

    $ help [
    [: [ arg... ]
        Evaluate conditional expression.
    
        This is a synonym for the "test" builtin, but the last argument must
        be a literal `]', to match the opening `['.
    

    所以[相当于test(除了期待]最后的争论)。help test将提供更多详细信息。您可以将其与help [[.

    还有一个外部命令[( man \[) 的手册页。

    如果是

    if [[-e
    
    • 那里既不是命令[也不[[是命令。完整的词[[-e是。
    • 这if [[-e使它成为对真/假的测试。那么命令[[-e存在吗?

    那是因为“[[-”是模棱两可的吗?

    是的。或者没有。[[-e它就是这样:shell 无法理解,因此它假定它是它自己的命令。;-)

    • 2
  3. WinEunuuchs2Unix
    2019-06-24T17:54:33+08:002019-06-24T17:54:33+08:00

    空格是分隔符,是必需的。如您所见shellcheck:

    $ shellcheck gmail-browse-msgs-algorithm.sh
    
    In gmail-browse-msgs-algorithm.sh line 847:
            [[$Today != "${DaysArr[ i + DAY_DELETED_ON_NDX ]}" ]] && continue
              ^-- SC1035: You need a space after the [[ and before the ]].
    

    (ksh 和 bash 都支持[[并且没有空格就无法工作。Shellcheck 使用包含该错误行的类似 ksh 和 bash 脚本准确地给出了该输出。)

    为什么需要分隔符与标记和词典有关。

    • 2
  4. DarkDust
    2019-06-24T22:28:06+08:002019-06-24T22:28:06+08:00

    [[可以是外部命令!也就是说,它可能是一个程序,而不是您的 shell 直接支持的某些语法。支持无空格语法是可能的,ksh但在具有外部的系统上会失败,[[因此出于兼容性原因,最好保留所需的空间。

    例如, BusyBox[[作为外部命令提供。

    • 1
  5. user968973
    2019-06-25T00:08:23+08:002019-06-25T00:08:23+08:00

    由于[[-e可以参与shell扩展(它可以像

    echo [[-e]*
    

    为了列出所有以介于[和包含之间的字母开头的文件),如果并且是特殊字符不参与正常的空格管理的分词e,那将是一团糟。[]

    • 0

相关问题

  • 如何从命令行仅安装安全更新?关于如何管理更新的一些提示

  • 如何从命令行刻录双层 dvd iso

  • 如何从命令行判断机器是否需要重新启动?

  • 文件权限如何工作?文件权限用户和组

  • 如何在 Vim 中启用全彩支持?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    如何运行 .sh 脚本?

    • 16 个回答
  • Marko Smith

    如何安装 .tar.gz(或 .tar.bz2)文件?

    • 14 个回答
  • Marko Smith

    如何列出所有已安装的软件包

    • 24 个回答
  • Marko Smith

    无法锁定管理目录 (/var/lib/dpkg/) 是另一个进程在使用它吗?

    • 25 个回答
  • Martin Hope
    Flimm 如何在没有 sudo 的情况下使用 docker? 2014-06-07 00:17:43 +0800 CST
  • Martin Hope
    Ivan 如何列出所有已安装的软件包 2010-12-17 18:08:49 +0800 CST
  • Martin Hope
    La Ode Adam Saputra 无法锁定管理目录 (/var/lib/dpkg/) 是另一个进程在使用它吗? 2010-11-30 18:12:48 +0800 CST
  • Martin Hope
    David Barry 如何从命令行确定目录(文件夹)的总大小? 2010-08-06 10:20:23 +0800 CST
  • Martin Hope
    jfoucher “以下软件包已被保留:”为什么以及如何解决? 2010-08-01 13:59:22 +0800 CST
  • Martin Hope
    David Ashford 如何删除 PPA? 2010-07-30 01:09:42 +0800 CST

热门标签

10.10 10.04 gnome networking server command-line package-management software-recommendation sound xorg

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve