There are no Booleans in Bash. However, we can define the shell variable having value as 0 (“ False “) or 1 (“ True “) as per our needs.
Zero is used to represent false, and One is used to represent true. For interpretation, Zero is interpreted as false and anything non-zero is interpreted as true. To make life easier, C Programmers typically define the terms "true" and "false" to have values 1 and 0 respectively.
First, it's important to remember that a successful exit code (a.k.a. “return code”) on Linux systems is 0. Think of this as meaning “zero errors”. Exit codes indicating some form of failure will have values of 1 or greater.
Like in C, the integers 0 (false) and 1 (true—in fact any nonzero integer) are used.
It's a convention, but a particularly useful one when you think about it. In general, if a program succeeds that's all you need to know. If it fails, however, you might need to know all kinds of information about the failure - why it happened, how to fix it, etc. Having zero mean 'success' and non-zero mean failure lets you can check pretty easily for success, and investigate the particular error for more details if you want to. A lot of APIs and frameworks have a similar convention - functions that succeed return 0 and and those that fail give back an error code describing the particular failure case.
Bash is a programming (scripting) language, but it's also a shell and a user-interface. If 0
was error, then the program could only present one kind of error.
However in Bash, any nonzero value is an error, and we may use any number from 1-255 to represent an error. This means we can have many different kinds of errors. 1
is a general error, 126
means that a file cannot be executed, 127
means 'command not found', etc. Here's a list of Bash Exit Codes With Special Meanings showing some of the most common exit codes.
There are also many kinds of success (exit status is 0
). However, a success will allow you to proceed to the next step—you can like print results to a screen, or execute a command, etc.
There are two related issues here.
First, the OP's question, Why 0 is true but false is 1 in the shell? and the second, why do applications return 0 for success and non-zero for failure?
To answer the OP's question we need to understand the second question. The numerous answers to this post have described that this is a convention and have listed some of the niceties this convention affords. Some of these niceties are summarized below.
Why do applications return 0 for success and non-zero for failure?
Code that invokes an operation needs to know two things about the exit status of the operation. Did the operation exit successfully? [*1] And if the operation does not exit successfully why did the operation exit unsuccessfully? Any value could be used to denote success. But 0 is more convenient than any other number because it is portable between platforms. Summarizing xibo's answer to this question on 16 Aug 2011:
Zero is encoding-independent.
If we wanted to store one(1) in a 32-bit integer word, the first question would be "big-endian word or little-endian word?", followed by "how long are the bytes composing a little-endian word?", while zero will always look the same.
Also it needs to be expected that some people cast errno to char or short at some point, or even to float. (int)((char)ENOLCK) is not ENOLCK when char is not at least 8-bit long (7-bit ASCII char machines are supported by UNIX), while (int)((char)0) is 0 independent of the architectural details of char.
Once it is determined that 0 will be the return value for success, then it makes sense to use any non-zero value for failure. This allows many exit codes to answer the question why the operation failed.
Why 0 is true but false is 1 in the shell?
One of the fundamental usages of shells is to automate processes by scripting them. Usually this means invoking an operation and then doing something else conditionally based on the exit status of the operation. Philippe A. explained nicely in his answer to this post that
In bash and in unix shells in general, return values are not boolean. They are integer exit codes.
It's necessary then to interpret the exit status of these operations as a boolean value. It makes sense to map a successful (0
) exit status to true and any non-zero/failure exit status to false. Doing this allows conditional execution of chained shell commands.
Here is an example mkdir deleteme && cd $_ && pwd
. Because the shell interprets 0 as true this command conveniently works as expected. If the shell were to interpret 0 as false then you'd have to invert the interpreted exit status for each operation.
In short, it would be nonsensical for the shell to interpret 0 as false given the convention that applications return 0 for a successful exit status.
[*1]: Yes, many times operations need to return more than just a simple success message but that is beyond the scope of this thread.
See also Appendix E in the Advanced Bash-Scripting Guide
The one fundamental point I find important to understand is this. In bash and in unix shells in general, return values are not boolean. They are integer exit codes. As such, you must evaluate them according to the convention saying 0 means success, and other values mean some error.
With test
, [ ]
or [[ ]]
operators, bash conditions evaluate as true in case of an exit code of 0 (the result of /bin/true). Otherwise they evaluate as false.
Strings are evaluated differently than exit codes:
if [ 0 ] ; then echo not null ; fi
if [ $(echo 0) ] ; then echo not null ; fi
if [ -z "" ] ; then echo null ; fi
The (( ))
arithmetic operator interprets 1 and 0 as true and false. But that operator cannot be used as a complete replacement for test
, [ ]
or [[ ]]
. Here is an example showing when the arithmetic operator is useful:
for (( counter = 0 ; counter < 10 ; counter ++ )) ; do
if (( counter % 2 )) ; then echo "odd number $counter" ; fi
done
It's just a convention that a 0 exit code means success. EXIT_SUCCESS will be 0 on almost every modern system.
EDIT:
"why both test 0 and test 1 returns 0(success) ?"
That's a completely different question. The answer is that passing a single argument to test always results in success unless that argument is the null string (""). See the Open Group documentation.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With