but still I can't get why it shows 1 instead of undefined.
It's not just you. This is a deep, dark part of the specification. :-)
The key here is that there are two x
s. Yes, really. There's the parameter x
, and there's the variable x
.
A parameter list containing expressions (like f
's default value) has its own scope separate from the function body's scope. But prior to parameter lists possibly having expressions, having var x
within a function with an x
parameter had no effect (x
was still the parameter, with the parameter's value). So to preserve that, when there's a parameter list with expressions in it, a separate variable is created and the value of the parameter is copied to the variable at the beginning of the function body. Which is the reason for this seemingly-odd (no, not just seemingly) odd behavior. (If you're the kind who likes to dive into the spec, this copying is Step 28 of FunctionDeclarationInstantiation.)
Since f
's default value, () => x
, is created within the parameter list scope, it refers to the parameter x
, not the var.
So the first solution, [2, 1, 1]
is correct, because:
2
was assigned to the var x
in the function body. So at the end of the function, the var x
is 2
.
1
was assigned to y
from the var x
before x
got the value 2
, so at the end of the function, y
is 1
.
- The parameter
x
's value has never changed, so f()
results in 1
at the end of the function
It's as though the code were written like this instead (I've removed unnecessary parens and added missing semicolons):
console.log(function(param_x, f = () => param_x) {
var var_x = param_x;
var y = var_x;
var_x = 2;
return [var_x, y, f()];
}(1));
...I removed var x from the function body, I found that the response changed to #3...
#3 is [2, 1, 2]
. That's correct, because when you remove the var x
from the function, there's only one x
, the parameter (inherited by the function body from the parmeter list). So assigning 2
to x
changes the parameter's value, which f
returns.
Taking the earier example with param_x
and var_x
, here's what it looks like if you remove the var x;
from it:
console.log(function(param_x, f = () => param_x) {
var y = param_x;
param_x = 2;
return [param_x, y, f()];
}(1));
Here's an annotated description of the original code (with the extraneous parentheses removed and missing semicolons added):
// /---- the parameter "x"
// v vvvvvvvvvvv--- the parameter "f" with a default value
console.log(function(x, f = () => x) {
var x; // <=== the *variable* x, which gets its initial value from the
// parameter x
var y = x; // <=== sets y to 1 (x's current value)
x = 2; // <=== changes the *variable* x's value to 2
// +---------- 2, because this is the *variable* x
// | +------- 1, because this is the variable y
// | | +--- 1, because f is () => x, but that x is the *parameter* x,
// | | | whose value is still 1
// v v vvv
return [x, y, f()];
}(1));
Final note regarding your title:
declaring a variable twice in IIFE
The variable is only declared once. The other thing is a parameter, not a variable. The distinction is rarely important...this being one of those rare times. :-)
var x;
doesn't define a new variable within the function scope, wherevar x = somevalue;
would – Ensnare