Saturday, April 25, 2015

Closures - PHP vs Javascript

In my previous post Exploring Lambda Functions and Closures in PHP, I gave you example of function that can calculate factorial of given number:

$factorial = function( $n ) use ( &$factorial ) {
    if( $n == 1 ) return 1;
    return $factorial( $n - 1 ) * $n;
};

echo $factorial( 5 ); // 120

In that post I simply said it is important to pass $factorial by reference otherwise it won't work. Let's explore why it is so in this post.

Closures are pretty common in Javascript so let's create a simple closure in it and see how it behaves:

var addOne = function(num) {
    return function() { 
        console.log(num++); 
    };
};

var f = addOne(1);

f(); // 1
f(); // 2
f(); // 3

As can be seen, each time we call the function, it increments the number. Let's now convert the same in PHP:

$addOne = function($num) {
    return function() { 
        echo $num++; 
    };
};

$f = $addOne(1);

$f();
$f();
$f();

This would result in error:

Notice: Undefined variable: num

Why is it so ? Because variable scope is different in PHP and Javascript. In PHP, all variables go in local scope by default unless you explicitly declare them in global scope. However in Javascript all variables go in global scope unless you explicitly declare them in local scope using var keyword. Since we are talking about functions and closures here, we can say that in PHP a variable declared in a function can only be available to that function where it is declared. On the other hand, in Javascript a variable can not only be available to function where it is declared but also to inner functions of that parent function. Let's see an example:

function test() {
  $variable = 'value';

  return function() {
    echo $variable;
  };
}

$f = test();
$f(); // Notice: Undefined variable: variable

So PHP gave error as inner function didn't had access to outer variable $variable. We could have made this variable available to inner function either by creating closure using use keyword or declaring that variable in global scope. Le't see same example in Javascript:

function test() {
  var variable = 'value';

  return function() {
    console.log(variable);
  };
}

var f = test();
f(); // value

In Javascript though, the inner function was able to get variable from outer scope eg parent function in this example.

This is reason why Javascript inner function shown above didn't give error whereas PHP's did. Therefore to have access to outer variable in PHP, we need to create a closure by using the use keyword. Of course in Javascript, you don't need something like use keyword because as I mentioned in Javascript a variable of a parent function is available to all its inner functions as well. So we need to use the use keyword in PHP due to difference of variable scoping in PHP and Javascript.

So let's fix that error we received earlier in PHP:

$addOne = function($num) {
    return function() use ($num) { 
        echo $num++; 
    };
};

$f = $addOne(1);

$f(); // 1
$f(); // 1
$f(); // 1

Now we have created the closure and error is gone and $num variable is now available to inner function but there is a problem, each time we call the function it says 1 unlike Javascript example we saw earlier which said 12 and 3. That's because unlike Javascript, PHP creates a new local variable inside inner function eg passes it by value not by reference. Conceptually, PHP sees this code like this:

$addOne = function($num) {
    return function() use ($num) { 
        $num = 1;  // so PHP kind of creates a new variable and assigns to 1 each time you call this
        echo $num++; 
    };
};

So to get similar output as in Javascript, we need to pass the variable using reference which can be done by using the & symbol:

$addOne = function($num) {
    return function() use (&$num) { // passed by reference 
        echo $num++; 
    };
};

$f = $addOne(1);

$f();
$f();
$f();

and now we get the output 12, and 3.


And that's also reason why you needed to pass variable by reference for the factorial example I had given in my previous post. Hope this post clears the confusion of closures and scoping especially for someone who comes from Javascript to PHP.

No comments:

Post a Comment

Popular Posts