PHP中的闭包......它们究竟是什么以及何时需要使用它们?

时间:2008-09-28 21:06:09

标签: php oop closures

所以我正在以一种不错的,最新的,面向对象的方式进行编程。我经常使用PHP实现的OOP的各个方面,但我想知道何时需要使用闭包。那里的任何专家都可以说明什么时候实现闭包会有用呢?

8 个答案:

答案 0 :(得分:75)

PHP将在5.3中原生支持闭包。当你想要一个仅用于某些小的特定目的的本地函数时,闭包是很好的。 RFC for closures给出了一个很好的例子:

function replace_spaces ($text) {
    $replacement = function ($matches) {
        return str_replace ($matches[1], ' ', ' ').' ';
    };
    return preg_replace_callback ('/( +) /', $replacement, $text);
}

这允许您在replacement内本地定义replace_spaces()功能,因此不是:
1)整理全局命名空间
2)让人们三年后不知道为什么有一个全局定义的函数只用在另一个函数中

它使事情井井有条。注意函数本身没有名称,它只是被定义并被指定为对$replacement的引用。

但请记住,你必须等待PHP 5.3:)

您还可以使用关键字use将范围外的变量访问到闭包中。考虑这个例子。

// Set a multiplier  
 $multiplier = 3;

// Create a list of numbers  
 $numbers = array(1,2,3,4);

// Use array_walk to iterate  
 // through the list and multiply  
 array_walk($numbers, function($number) use($multiplier){  
 echo $number * $multiplier;  
 }); 

这里给出了一个很好的解释What are php lambdas and closures

答案 1 :(得分:15)

如果您将来需要一个执行您现在决定的任务的功能。

例如,如果您读取配置文件,其中一个参数告诉您算法的hash_methodmultiply而不是square,则可以创建一个闭包可以在任何需要哈希的地方使用。

可以在(例如)config_parser()中创建闭包;它使用do_hash_method()的本地变量(来自配置文件)创建一个名为config_parser()的函数。每当调用do_hash_method()时,它就可以访问config_parser()的局部范围内的变量,即使它没有在该范围内被调用。

一个有希望的好假设的例子:

function config_parser()
{
    // Do some code here
    // $hash_method is in config_parser() local scope
    $hash_method = 'multiply';

    if ($hashing_enabled)
    {
        function do_hash_method($var)
        {
            // $hash_method is from the parent's local scope
            if ($hash_method == 'multiply')
                return $var * $var;
            else
                return $var ^ $var;
        }
    }
}


function hashme($val)
{
    // do_hash_method still knows about $hash_method
    // even though it's not in the local scope anymore
    $val = do_hash_method($val)
}

答案 2 :(得分:15)

除了技术细节之外,闭包是编程风格(称为功能导向编程)的基本先决条件。闭包大致用于在面向对象编程中使用对象时的相同内容;它将数据(变量)与某些代码(函数)绑定在一起,然后您可以将其传递到其他位置。因此,它们会影响您编写程序的方式,或者 - 如果您不改变编写程序的方式 - 它们根本没有任何影响。

在PHP的上下文中,它们有点奇怪,因为PHP已经在基于类,面向对象的范例以及较旧的程序范例上沉重。通常,具有闭包的语言具有完整的词法范围。为了保持向后兼容性,PHP不会得到这个,所以这意味着闭包在这里会比在其他语言中有所不同。我想我们还没有看到它们将如何被使用。

答案 3 :(得分:10)

我喜欢troelskn的帖子提供的上下文。当我想在Dan中使用Dan Udey的例子时,我会使用OO策略模式。在我看来,这比引入一个新的全局函数要好得多,该函数的行为是在运行时确定的。

http://en.wikipedia.org/wiki/Strategy_pattern

您还可以使用在PHP中保存方法名称的变量来调用函数和方法,这很棒。所以Dan的例子就是这样的:

class ConfigurableEncoder{
        private $algorithm = 'multiply';  //default is multiply

        public function encode($x){
                return call_user_func(array($this,$this->algorithm),$x);
        }

        public function multiply($x){
                return $x * 5;
        }

        public function add($x){
                return $x + 5;
        }

        public function setAlgorithm($algName){
                switch(strtolower($algName)){
                        case 'add':
                                $this->algorithm = 'add';
                                break;
                        case 'multiply':        //fall through
                        default:                //default is multiply
                                $this->algorithm = 'multiply';
                                break;
                }
        }
}

$raw = 5;
$encoder = new ConfigurableEncoder();                           // set to multiply
echo "raw: $raw\n";                                             // 5
echo "multiply: " . $encoder->encode($raw) . "\n";              // 25
$encoder->setAlgorithm('add');
echo "add: " . $encoder->encode($raw) . "\n";                   // 10

当然,如果你想让它随处可用,你可以让一切都变得静止......

答案 4 :(得分:2)

闭包基本上是一个函数,您可以在一个上下文中编写定义但在另一个上下文中运行。 Javascript通过理解这些帮助了我很多,因为它们在整个地方用于JavaScript。

在PHP中,由于函数内“全局”(或“外部”)变量的范围和可访问性不同,它们的效率低于JavaScript。然而,从PHP 5.4开始,闭包可以在对象内部运行时访问$ this对象,这使它们更有效。

这就是关闭的内容,它应该足以理解上面写的内容。

这意味着应该可以在某处编写函数定义,并在函数定义中使用$ this变量,然后将函数定义分配给变量(其他人已经给出了语法示例),然后传递此变量对于一个对象并在对象上下文中调用它,该函数然后可以通过$ this访问和操作该对象,就像它只是它的另一个方法一样,实际上它并没有在该对象的类定义中定义,而是在某个地方其他

如果不是很清楚,那么不用担心,一旦你开始使用它们就会变得清晰。

答案 5 :(得分:1)

基本上,闭包是tat的内部函数,可以访问外部变量,并且用作匿名函数(没有任何名称的函数)的回调函数。

 <?php
      $param='ironman';
      function sayhello(){
          $param='captain';
          $func=function () use ($param){
                $param='spiderman';
          };
       $func();
       echo  $param;
       }
      sayhello();
?>

//output captain

//and if we pass variable as a reference as(&$param) then output would be spider man;

答案 6 :(得分:0)

以下是php

中闭包的示例
// Author: HishamDalal@gamil.com
// Publish on: 2017-08-28

class users
{
    private $users = null;
    private $i = 5;

    function __construct(){
        // Get users from database
        $this->users = array('a', 'b', 'c', 'd', 'e', 'f');
    }

    function displayUsers($callback){
        for($n=0; $n<=$this->i; $n++){
            echo  $callback($this->users[$n], $n);
        }
    }

    function showUsers($callback){
        return $callback($this->users);

    }

    function getUserByID($id, $callback){
        $user = isset($this->users[$id]) ? $this->users[$id] : null;
        return $callback($user);
    }

}

$u = new users();

$u->displayUsers(function($username, $userID){
    echo "$userID -> $username<br>";
});

$u->showUsers(function($users){
    foreach($users as $user){
        echo strtoupper($user).' ';
    }

});

$x = $u->getUserByID(2, function($user){

    return "<h1>$user</h1>";
});

echo ($x);

<强>输出:

0 -> a
1 -> b
2 -> c
3 -> d
4 -> e
5 -> f

A B C D E F 

c

答案 7 :(得分:0)

关闭

MDN是IMO的最佳解释:

闭包是捆绑在一起的功能的组合(闭包) 并参考其周围状态(词汇环境)。在 换句话说,闭包使您可以访问外部函数的作用域 来自内部函数。

即闭包是可以访问父作用域中变量的函数。闭包使我们能够方便地动态创建函数,因为在某些情况下,仅在一个地方需要函数(回调,可调用参数)。

示例:

$arr = [1,2,3,3];
$outersScopeNr = 2;

// The second arg in array_filter is a closure
// It would be inconvenient to have this function in global namespace
// The use keyword lets us access a variable in an outer scope
$newArr = array_filter($arr, function ($el) use ($outersScopeNr) {
    return $el === 3 || $el === $outersScopeNr;
});

var_dump($newArr);
// array (size=3)
//  1 => int 2
//  2 => int 3
//  3 => int 3