In JavaScript, you have the ability to call a function, or use a variable, before it’s declared. This concept is called hoisting, and it sounds like magic, right? Well, maybe at first, but understanding how JavaScript code is executed will help to demystify this concept.
When JavaScript code is run, there are two phases that occur in relation to hoisting. In the first phase, variable and function declarations are saved in memory. At this point, declared variables are considered either uninitialized or undefined. In the second phase, variables are initialized, or in other words, assigned values, and code is executed. Hoisting is possible because variables and functions are given references in memory before code is executed in the second phase.
Functions and variables each have their own nuances in regards to hoisting. It’s important to understand these nuances and be mindful of them in order to utilize hoisting effectively.
Functions
- Declarations are hoisted to the top of their scope.
- Expressions are NOT hoisted.
In the code above, I tried hoisting a function declaration, findApples()
, and a function expression, findOranges()
. Even though findApples()
was called before it was declared, the function call in line one still logged, “I found the apples!” to the console. However, the function expression, findOranges()
, returned an error and was not hoisted.
Variables
- Declarations are hoisted to the top of their scope.
- Assignments are NOT hoisted.
JavaScript only hoists variable declarations, not assignments. The initialization of variables is handled differently, depending on whether var, let, or const is used to declare them.
In the example above, let was used to declare the honeycrisp variable, and it threw a reference error when it was hoisted on line one. The same reference error would occur if honeycrisp was declared using const. However, in the case of var, variables are initially set to undefined
upon declaration. If honeycrisp was declared using var, the return value of line one would have been undefined
.
Regardless of whether you use var, let, or const, the variable assignment on line five will never be hoisted.
- If a variable is assigned a value, but not declared, the variable will become global once the assignment code is executed.
In the eatingApple()
function above, the grannySmith variable is assigned a value, but it is not declared using var, let, or const. Usually, when variables are declared and assigned values inside of a function, they are only accessible with that function’s scope. In the case above, the undeclared grannySmith variable becomes global when the eatApple()
function is executed.
Hoisting order of priority
JavaScript follows the following order of priority when hoisting eligible code to the top of the stack:
- Variable assignment
- Function declarations
- Variable declarations
Takeaways
Although hoisting is a useful feature of JavaScript, it can cause confusion and unexpected bugs. The following guidelines will help you to write clear code and avoid negative side effects of hoisting.
- Always declare variables at the top of their scope.
- In addition to declaration, assign values to variables at the top of their scope whenever possible.
- Declare and define all functions before using them.
- Try to declare functions in order of their use.