bash if -a vs -e option

前端 未结 3 1281
执笔经年
执笔经年 2020-12-08 02:09

Both about -a and -e options in Bash documentation is said:

-a file
    True if file exists.
-e file
    True if file exists. 


        
相关标签:
3条回答
  • 2020-12-08 02:18

    I researched, and this is quite hairy:

    -a is deprecated, thus isn't listed in the manpage for /usr/bin/test anymore, but still in the one for bash. Use -e . For single '[', the bash builtin behaves the same as the test bash builtin, which behaves the same as /usr/bin/[ and /usr/bin/test (the one is a symlink to the other). Note the effect of -a depends on its position: If it's at the start, it means file exists. If it's in the middle of two expressions, it means logical and.

    [ ! -a /path ] && echo exists doesn't work, as the bash manual points out that -a is considered a binary operator there, and so the above isn't parsed as a negate -a .. but as a if '!' and '/path' is true (non-empty). Thus, your script always outputs "-a" (which actually tests for files), and "! -a" which actually is a binary and here.

    For [[, -a isn't used as a binary and anymore (&& is used there), so its unique purpose is to check for a file there (although being deprecated). So, negation actually does what you expect.

    0 讨论(0)
  • 2020-12-08 02:27

    The '-a' option to the test operator has one meaning as a unary operator and another as a binary operator. As a binary operator, it is the 'and' connective (and '-o' is the 'or' connective). As a unary operator, it apparently tests for a file's existence.

    The autoconf system advises you to avoid using '-a' because it causes confusion; now I see why. Indeed, in portable shell programming, it is best to combine the conditions with '&&' or '||'.

    I think @litb is on the right track. When you have '! -a ${resin_dir}', Bash may be interpreting it as "is the string '!' non-empty and is the string in '${resin_dir}' non-empty, to which the answer is yes. The Korn shell has a different view on this, and the Bourne shell yet another view - so stay away from '-a'.

    On Solaris 10:

    $ bash -c 'x=""; if [ ! -a "$x" ] ; then echo OK ; else echo Bad; fi'
    Bad
    $ ksh -c 'x=""; if [ ! -a "$x" ] ; then echo OK ; else echo Bad; fi'
    OK
    $ sh -c 'x=""; if [ ! -a "$x" ] ; then echo OK ; else echo Bad; fi'
    sh: test: argument expected
    $
    
    0 讨论(0)
  • 2020-12-08 02:34

    The double bracket [[ exp ]] is a bash builtin. In bash -a and -e are the same, probably for some backwards compatibility.

    The single bracket [ exp ] is an alias for the external command "test". In "test", -a is a logical AND. Although [ nothing AND $STRING ] looks like it should be false, test has some syntax quirks, which is why I recommend using the bash builtin [[ exp ]], which tends to be more sane.

    Note: bash really does call /bin/[ when you use "[".

    $ [ $UNASIGNED_VAR == "bar" ]
    bash: [: ==: unary operator expected
    

    the error shows bash called [. An strace also shows "execve("/usr/bin/[", ..."

    0 讨论(0)
提交回复
热议问题