[This post is part of a
series on JavaScript quirks.]
JavaScript is very tolerant when it comes to accepting values. For example, everywhere it expects a number, it does not reject values from other types, but tries to convert them:
> '5' - '2'
3
> '5' * '2'
10
Automatic conversion to boolean is rarely problematic and often useful. It is covered here as a preparation for later – we’ll use it to work around quirks. Automatic conversion to string, however, can cause problems.
Implicit conversion to boolean: “truthy” and “falsy” values
Whenever JavaScript expects a boolean value (e.g. for the condition of an
if statement), any value can be used. It will be interpreted as either
true or
false. The following values are interpreted as
false:
- undefined, null
- Boolean: false
- Number: -0, +0, NaN
- String: ''
All other values are considered
true. Values interpreted as
false are called
falsy, values interpreted as
true are called
truthy. You can use
Boolean as a function to test how a value is interpreted. It converts its argument to boolean:
> Boolean(undefined)
false
> Boolean(0)
false
> Boolean(3)
true
Implicit conversion of strings
In web development, you often get values as strings that are actually numbers or booleans. For example, when users enter this kind of data in a form. If you forget to explicitly convert these strings then JavaScript will surprise you negatively in two ways: First, there will be no warning. Second, the values will be converted automatically, but wrongly. The plus operator (
+), for instance, is problematic, because it concatenates strings as soon as one of its operands is a string. During the following interaction with JavaScript, the assumption is that we are adding 1 to 5. Instead, we are concatenating the strings
'5' and
'1'.
> var x = '5'; // wrong assumption: x is a number
> x + 1
'51'
Furthermore, there are a few falsy values that are truthy if converted to string. Example:
false.
> Boolean(false) // truthy?
false
> String(false)
'false'
> Boolean('false') // truthy?
true
Example:
undefined.
> Boolean(undefined) // truthy?
false
> String(undefined)
'undefined'
> Boolean('undefined') // truthy?
true
Implicit conversion of objects
Objects are only implicitly converted if JavaScript expects a number or a string. In the former case, the conversion takes three steps:
- Call valueOf(). If the result is primitive (not an object) then use it and convert it to a number.
- Otherwise, call toString(). If the result is primitive, use it and convert it to a number.
- Otherwise, throw a TypeError.
Example for step 1:
> 3 * { valueOf: function () { return 5 } }
15
Example for step 3:
> function returnObject() { return {} }
> 3 * { valueOf: returnObject, toString: returnObject }
TypeError: Cannot convert object to primitive value
If JavaScript converts to string, steps 1 and 2 are swapped:
toString() is tried first,
valueOf() second.
Best practice: explicit conversion
It is best to explicitly convert values to the desired types before using them. A minimal solution is to use the functions
Boolean(),
Number() and
String():
function handleFormData(formData) {
var givenName = String(formData.givenName);
var age = Number(formData.age);
...
}
These functions always return a value (they never throw an exception). However,
Number() returns the error value
NaN [1] if it can’t convert a value:
> Number('xyz')
NaN
> Number(undefined)
NaN
> Number(null)
0
> Number(true)
1
A more elaborate solution would be to first check whether the value you are converting has the correct format (e.g. someone shouldn’t enter
'xyz' as their age) and to take appropriate measures if it doesn’t.
Reference
- NaN and Infinity in JavaScript