15

我正在尝试学习 PHP,现在我被困在“静态匿名函数”中。

我在教程中找到了这个(http://www.slideshare.net/melechi/php-53-part-2-lambda-functions-closures-presentation

“面向对象

  • Lambda 函数是闭包,因为它们会自动绑定到创建它们的类的范围。
  • ' $this' 并不总是需要在范围内。
  • 删除 ' $this' 可以节省内存。
  • 您可以通过将 Lambda 函数声明为静态来阻止此行为。”

这段代码有什么问题?

我收到此错误:

解析错误:解析错误,在第 11 行的 C:\wamp\www\z-final\a.php 中期待 `T_PAAMAYIM_NEKUDOTAYIM'

为什么此代码行不起作用“return static function(){var_dump($this);};” ?

class foo
{
    public function getLambda()
    {
        return function(){var_dump($this);};
    }

    public function getStaticLambda()
    {
        return static function(){var_dump($this);};
    }
}

$foo = new foo();
$lambda = $foo->getLambda();
$staticLambda = $foo->getStaticLambda();
$lambda();
$staticLambda();
4

2 回答 2

41

是的,这是 5.4+ 中完全有效的语法。

基本上,它会阻止当前类自动绑定到闭包(实际上,它会阻止所有绑定,但稍后会详细介绍)。

class Foo {
    public function bar() {
        return static function() { var_dump($this); };
    }
    public function baz() {
        return function() { var_dump($this); };
    }
}

如果我们在 5.4+ 上实例化它,闭包bar()返回将$this设置为 null。就像您对它进行了静态调用一样。但是baz()$this设置为您调用的 foo 实例baz()

所以:

$bar = $f->bar();

$bar();

结果是:

注意:未定义的变量:第 5 行 /in/Bpd3d 中的 this

无效的

$baz = $f->baz();

$baz();

结果是

对象(Foo)#1(0){

}

说得通?伟大的。

现在,如果我们采用在函数外部定义的闭包会发生什么:

$a = function() { var_dump($this); };
$a();

我们得到null(和通知)

$c = $a->bindTo(new StdClass());
$c();

我们得到StdClass了,正如你所期望的那样

$b = static function() { var_dump($this); };
$b();

我们得到null(和通知)

$d = $b->bindTo(new StdClass());
$d();

这就是事情变得有趣的地方。现在,我们收到警告、通知和 null:

警告:无法将实例绑定到第 12 行 /in/h63iF 中的静态闭包

注意:未定义的变量:第 9 行 /in/h63iF 中的 this

无效的

所以在 5.4+ 中,你可以声明一个静态闭包,这会导致它永远不会$this绑定到它,你也不能将对象绑定到它......

于 2014-05-08T23:53:55.317 回答
0

应该不需要用static关键字定义它。

<?php
class House
{
     public function paint($color)
     {
         return function() use ($color) { return "Painting the house $color..."; };
     }
}

$house = new House();
$callback = $house->paint('red');
var_dump($callback); // object(Closure)#2 (2) {..}
var_dump($callback()); // "Painting the house red..."
于 2012-09-25T09:11:12.397 回答