28

I get what I expected when doing this in bash:

[ "a" == "a" ] && echo yes

It gave me yes.

But when I do this in zsh, I get the following:

zsh: = not found

Why does the same command (/usr/bin/[) behave differently in different shells?

1
  • 5
    it's not the same command - its a builtin found before $PATH is searched. and == isn't valid test syntax for the /usr/bin/[ anway. Just = is fine. Commented Jan 15, 2016 at 8:46

4 Answers 4

33

It's not /usr/bin/[ in either of the shells. In Bash, you're using the built-in test/[ command, and similarly in zsh.

The difference is that zsh also has an = expansion: =foo expands to the path to the foo executable. That means == is treated as trying to find a command called = in your PATH. Since that command doesn't exist, you get the error

zsh: = not found

that you saw (and in fact, this same thing would happen even if you actually were using /usr/bin/[).


You can use == here if you really want. This works as you expected in zsh:

[ "a" "==" "a" ] && echo yes

because the quoting prevents =word expansion running. You could also disable the equals option with setopt noequals.


However, you'd be better off either:

  • Using single =, the POSIX-compatible equality test; or
  • Better still, using the [[ conditionals with == in both Bash and zsh. In general, [[ is just better and safer all around, including avoiding this kind of issue (and others) by having special parsing rules inside.
11
  • What's the exact difference between [ and [[? (Searching on Google for [ vs [[ gives me results for vs only, LOL) Commented Jan 15, 2016 at 9:04
  • 2
    [[ supports a wider range of tests in both cases, and has custom parsing rules that avoid the need for quoting variables, operators, and so on. If you're specifically using either Bash or zsh, use [[. If you're writing a portable script, write to the POSIX-compatible [/test command (which may or may not be a real command on your running system). Commented Jan 15, 2016 at 9:06
  • 1
    Just use [[ there and forget [ exists. Commented Jan 15, 2016 at 9:09
  • 1
    you know... i disagree with your recommendation. [[ is differently capable. try this with it: for f in *; do [ -e "$f" ] && for a in f d h p S b c; do [ "-$a" "$f" ] && for p in r w x u g; do [ "-$p" "$f" ] || p=-; a=$a$p; done && break; done && printf "%s:\t%s\n" "$a" "$f"; done. Commented Jan 15, 2016 at 9:42
  • 3
    Well, =/== is arguably one case where [[ is not better than [, as [[ a == b ]] is does "a" match the "b" pattern and not is "a" equal to "b" as you'd expect. That means you need to write [[ $a == "$b" ]] for instance. At least, with the [ command, if you know how command parsing works, you know you need to write it [ "$a" = "$b" ] to prevent the split+glob operator like in other commands. With that in mind and if you don't use -a or -o, [ is safe in POSIX conformant implementations. Commented Jan 15, 2016 at 10:07
4

[ is a shell builtin command in bash and in zsh:

$ type [
[ is a shell builtin

From the Shell Builtin Commands documentation:

Builtin commands are contained within the shell itself. When the name of a builtin command is used as the first word of a simple command (see Simple Commands), the shell executes the command directly, without invoking another program. Builtin commands are necessary to implement functionality impossible or inconvenient to obtain with separate utilities.

The official documentation ($ help test) only allows to use =:

STRING1 = STRING2

True if the strings are equal.

So, the correct expression would be:

$ [ "a" = "a" ] && echo yes
yes

What happens is that bash is a bit less strict. Supporting the == operator with [ seems to be a bash extension and it is no recommended to use it:

string1 == string2

string1 = string2

True if the strings are equal. When used with the [[ command, this performs pattern matching as described above (see Conditional Constructs).

‘=’ should be used with the test command for POSIX conformance.

If you want to use ==, you should use the [[ keyword:

$ [[ "a" == "a" ]] && echo yes
yes

Keep in mind that [[ is less portable (is not POSIX). But both bash and zsh support it.

2

And zsh, and bash give the same answer (type is builtin too for both shells):

$ type -a [
[ is a shell builtin
[ is /usr/bin/[
1

In both shells, bash and zsh, the [ utility is a shell builtin. This is the shells implementation of that tool, it is used in preference to the binary /usr/bin/[. The different results you encounter is caused by different implementations.


In bash, the [ utility accepts CONDITIONAL EXPRESSIONS as the [[ compound command. According to bashs man page both = and == are valid:

string1 == string2
string1 = string2
        True if the strings are equal.  = should be used with the test command for POSIX
        conformance.

In zsh, the [ utility attempts to implement POSIX and its extensions where these are specified. In the specification of the POSIX test utility there is no == operator defined.

You must log in to answer this question.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.