PHP进阶之闭包

分享于:2019-09-05 21:58:20

说起来“闭包”,我们在学习JS时必会了解的知识。相关链接《第十二章:第12节JavaScript对象——闭包


从PHP5.3也引入了闭包的概念,想了解PHP的闭包,首先得知道PHP的匿名函数相关知识《匿名函数》。


1)use关键字





上图是JS的一个闭包示例,我们用PHP仿写一下。

function f1(){
	$age = 20;
	$height = 170;
	$f2 = function(){
		echo "年龄:".$age."-身高:".$height;
	};
	return $f2;
}
$ff = f1();
$ff();

运行后报错了:

1.png


这个错误不难理解,原因是PHP函数里的变量作用域不受函数外影响,它与JS不一样。所以(函数f1里的)匿名函数$f2里的$age、$height无法获取到$f2外的相关变量值。不能获取,那就不算的是闭包函数,那PHP怎么可以像JS一样实现闭包呢?


此时需要用到use关键字

function f1(){
	$age = 20;
	$height = 170;
	$f2 = function() use($age,$height) {
		echo "年龄:".$age."-身高:".$height;
	};
	return $f2;
}

$ff = f1();
$ff();


1.png


use意思是连接闭包和外界变量。


2)use中使用引用传值


再来看下一个JS闭包示例:




用PHP仿写一下:

function f1(&$add){
	$n = 999;
	$add = function() use ($n){
		$n++;
	};

	$f2 = function() use ($n){
		echo $n.'<br />';
	};

	return $f2;
}

$add= null;
$result = f1($add);
$result();
$add();
$result();


得到结果:

1.png


这里又和JS的闭包出现差异了,原因是use里传入的变量都是拷贝传值,所以闭包函数里的变量影响不到闭包函数外的变量,改正它也简单,把拷贝传值改成引用传值即可。

function f1(&$add){
	$n = 999;
	$add = function() use (&$n){
		$n++;
	};

	$f2 = function() use (&$n){
		echo $n.'<br />';
	};

	return $f2;
}

$add= null;
$result = f1($add);
$result();
$add();
$result();


1.png


3)继承变量的时机


继承变量的行为是在函数定义时产生还是在函数调用时产生?我们调整下上例中代码的顺序,将$msg置于函数定义之后

$func = function()use($msg){
    print_r($msg);
};  

$msg = [1,2,3];

$func();
?>

运行输出
PHP Notice:  Undefined variable: msg in /search/ballqiu/c.php on line 4


可见,继承变量的行为是在函数定义时产生的。上例中定义msg,所以函数运行时$msg就是未定义变量。


4)引用传值与拷贝传值的区别


如果传递object类型的变量,即使是普通的拷贝传值,匿名函数中变量值的改变同样会影响到外部相关变量。《PHP对象的传值

向use传递object变量时,使用引用与拷贝传值到底有没有区别呢?

$func = function()use($msg){
    echo $msg[0],"\n";
};

$msg = new ArrayObject([1,2,3], ArrayObject::ARRAY_AS_PROPS);
$func();
?>

运行输出
PHP Notice:  Undefined variable: msg


改为引用传递


$func = function()use($msg){
    echo $msg[0],"\n";
};

$msg = new ArrayObject([1,2,3], ArrayObject::ARRAY_AS_PROPS);
$func();
?>

运行输出
1


可见使用引用传递时,即使变量滞后于函数定义,函数内部还是可以找到外部相应的变量,不会出现变量未定义的情况。两者还是有区别的。


5)class中匿名函数里的this


class C{
    protected $_num = 0;

    public function mkFunc(){
        $func = function(){
            echo $this->_num++, "\n";
        };

        return $func;
    }

    public function get(){
        echo $this->_num,"\n";
    }
}

$obj = new C();
$func = $obj->mkFunc();
$func();

$obj->get();
?>

运行结果
0
1


可见匿名函数里的this就是指当前对象,不需要使用use就可以直接找到

还是上面的例子,如果一定要使用use会是什么效果呢? 
将mkFunc改为

public function mkFunc(){
    //唯一改动是此处加了use
    $func = function()use($this){
        echo $this->_num++, "\n";
    };

    return $func;
}

运行输出
PHP Fatal error:  Cannot use $this as lexical variable


换一种方式


public function mkFunc(){
    $self = $this;
    $func = function()use($self){
        echo $this->_num++, "\n";
    };

    return $func;
}

运行结果
0
1