How do you use an anonymous function inside a class in PHP?

5.2k views Asked by At

How do I call $greet inside this class? I am using PHP 5.5.4.

<?PHP   
class Model
{
    public $greet = function($name)
    {
        printf("Hello %s\r\n", $name);
    };
}

$test = new Model();
$test->greet('World');
$test->greet('PHP');
?>

Parse error: syntax error, unexpected '$greet' (T_VARIABLE), expecting function (T_FUNCTION)

Also tried this,

$test = new Model();
call_user_func($test->greet('World'));
call_user_func($test->greet('PHP')) 

The anonymous function works fine outside the class (straight from the manual).

<?php
$greet = function($name)
{
    printf("Hello %s\r\n", $name);
};

$greet('World');
$greet('PHP');
?>

EDIT: I took out the dollar signs, in my call (I caught it just as an answer started to roll in. It did not help,

call_user_func($test->greet('World'));
call_user_func($test->greet('PHP'));

EDIT:

class Model
{
    public $greet;
    function __construct()
    {
        $this->greet = function($name)
        {
            printf("Hello %s\r\n", $name);
        };
    }
}

$test = new Model();
$test->greet('johnny');

Now I get,

Fatal error: Call to undefined method Model::greet() 
2

There are 2 answers

5
Arnold Daniels On BEST ANSWER

The fact that you call greet makes PHP treat it like a function and not a property. You can have the same name for both a property and method in PHP, so the distinction is relevant.

PHP 7+

In PHP7 the __call() method is no longer needed to call closures bound to properties, because of Uniform Variable Syntax. This will allow you to add parentheses around any code, just like you do in arithmetics.

class Model
{
    public $greet;
    function __construct()
    {
        $this->greet = function($name)
        {
            printf("Hello %s\r\n", $name);
        };
    }
}

$test = new Model();
($test->greet)('johnny');

PHP 5

What you can doe as a workaround is use the __call() magic method. It will catch the call to the undefined greet method.

class Model
{
    public $greet;
    function __construct()
    {
        $this->greet = function($name)
        {
            printf("Hello %s\r\n", $name);
        };
    }

    function __call($method, $args)
    {
        if (isset($this->$method) && $this->$method instanceof \Closure) {
            return call_user_func_array($this->$method, $args);
        }

        trigger_error("Call to undefined method " . get_called_class() . '::' . $method, E_USER_ERROR);
    }
}

$test = new Model();
$test->greet('johnny');
6
Marc B On

You cannot initialize object variables with the results of expressions. Only static/constant values are allowed. e.g.

class foo {
   public $bar = 1+1; // illegal - cannot use an expression
   public $baz = 2; // valid. '2' is a constant
}

You'd have to do:

class foo {
   public $bar;
   function __construct() {
        $this->bar = function() { .... }
   }
}

And to actually invoke the closure, as per this answer:

$x = new foo()
$x->bar->__invoke();