I got an unexpected result. Here's the code:
b = function c() {
console.log(c);
c = 3;
console.log(c);
}
b();
I thought the second console.log should print "3" but instead I got the function itself. Why?
Meanwhile, from the code below I got the right "3".
function ff() {
ff = 3;
console.log(ff);
}
ff();
You cannot overwrite the function's named variable inside its own declaration. NFE (Named Function Expression) name cannot be overwritten (because it is constant)
This is clear when you write in strict JS mode. Try the example below:
This pattern of function declaration shown in the first example is called NFE (named function expression), while the othher one is called function declaration. The big difference is, in case of NFE, the function's name,
i.e. c
in our case lives only incide the scope of the function. Therefore, if you will try to call it by its' name from outside wou will get the error,c is not defined
, which meansc
doesn't exist globally.Now look closely at the line
c=3
inside the function body of c. What this code block normally would have done, is create a global variable by the name c, outside the function body, which would have been accessible, outside the function as well. But, here, as thec
already lives inside the scope of the function's own body, it will not let you declare there, cause it will mean to overwrite its' own name, which is not allowed in case of NFE, (but allowed in case of function declaration, i.e, the second example in question). That is precisely why the assignment code,c=3
is not doing anything here.To realize it more closely, you can update
c=3
withvar c=3
, in which case it will let you declare a local variable by the name ofc
inside your function body, which you can then use inside the function.You are using a function expression:
So
b = function c() { ... };
is perfectly valid, strict mode or otherwise. What happens toc
is another question. According to the specs:So:
c
is visible inside the function but not outside itc
cannot be overwritten from inside the function (an "immutable" binding)This is a function declaration; different (and more obvious) rules apply here.