@thefourtheye is correct in saying that these variables cannot be accessed before they are declared. However, it's a bit more complicated than that.
Are variables declared with
const not hoisted? What is really going on here?
All declarations (
x = "global";
// function scope:
x; // not "global"
// block scope (not for `var`s):
x; // not "global"
This is true both for function and block scopes1.
The difference between
function* declarations and
class declarations is the initialisation.
The former are initialised with
undefined or the (generator) function right when the binding is created at the top of the scope. The lexically declared variables however stay uninitialised. This means that a
ReferenceError exception is thrown when you try to access it. It will only get initialised when the
class statement is evaluated, everything before (above) that is called the temporal dead zone.
x = y = "global";
x; // undefined
y; // Reference error: y is not defined
var x = "local";
let y = "local";
Notice that a
let y; statement initialises the variable with
let y = undefined; would have.
The temporal dead zone is not a syntactic location, but rather the time between the variable (scope) creation and the initialisation. It's not an error to reference the variable in code above the declaration as long as that code is not executed (e.g. a function body or simply dead code), and it will throw an exception if you access the variable before the initialisation even if the accessing code is below the declaration (e.g. in a hoisted function declaration that is called too early).
Is there any difference between
const in this matter?
No, they work the same as far as hoisting is regarded. The only difference between them is that a
constant must be and can only be assigned in the initialiser part of the declaration (
const one = 1;, both
const one; and later reassignments like
one = 2 are invalid).
var declarations are still working only on the function level, of course