设为首页 - 加入收藏 ASP站长网(Aspzz.Cn)- 科技、建站、经验、云计算、5G、大数据,站长网!
热搜: 创业者 手机 数据
当前位置: 首页 > 站长学院 > PHP教程 > 正文

PHP闭包和匿名函数使用解析

发布时间:2022-02-24 13:37 所属栏目:121 来源:互联网
导读:1、概述 闭包和匿名函数在PHP 5.3.0中引入,这两个特性非常有用,每个PHP开发者都应该掌握。 闭包是指在创建时封装周围状态的函数,即使闭包所在的环境的不存在了,闭包中封装的状态依然存在。 匿名函数其实就是没有名称的函数,匿名函数可以赋值给变量,还
  1、概述
 
  闭包和匿名函数在PHP 5.3.0中引入,这两个特性非常有用,每个PHP开发者都应该掌握。
 
  闭包是指在创建时封装周围状态的函数,即使闭包所在的环境的不存在了,闭包中封装的状态依然存在。
 
  匿名函数其实就是没有名称的函数,匿名函数可以赋值给变量,还能像其他任何PHP函数对象那样传递。不过匿名函数仍然是函数,因此可以调用,还可以传入参数,适合作为函数或方法的回调。
 
  注:理论上讲闭包和匿名函数是不同的概念,不过PHP将其视作相同的概念(匿名函数在PHP中也叫作闭包函数),所以下面提到闭包时指的也是匿名函数;反之亦然。
 
  2、创建闭包
 
  创建闭包很简单:
 
  $greet = function ($name) {
      return sprintf("Hello %s/r/n", $name);
  };
  
  echo $greet('Cuoxin.com');
  结果打印:
 
  Hello Cuoxin.com
 
  闭包和普通的PHP函数很像:常用的句法相同,也接受参数,而且能返回值。不过闭包没有函数名。
   print_r($numberPlusOne);
  在闭包出现之前,要实现这样的功能,PHP开发者只能单独创建具名函数,然后使用名称引用这个函数:
 
  function incrementNumner ($number) {
      return $number += 1;
  }
  
  $numberPlusOne = array_map(‘incrementNumber’,  [1, 2, 3]);
  print_r($numberPlusOne);
  这样做把回调的实现和使用场所隔离开了,而且使用闭包实现代码更加简洁。
 
  3、从父作用域继承变量
 
  在PHP中必须手动调用闭包对象的bindTo方法或使用use关键字把父作用域的变量及状态附加到PHP闭包中。而实际应用中,又以使用use关键字实现居多。
 
  use关键字
 
  实际上,Laravel框架中也大量使用了闭包,最常见的比如路由定义:
 
  Route::group(['domain' => '{account}.myapp.com'], function () {
      Route::get('user/{id}', function ($account, $id) {
          //
      });
  });
  这里面的两个function都是闭包。而从父作用域继承变量的使用场景在Laravel底层源码中也是俯拾即是,比如Model.php(Illuminate/Database/Eloquent)的saveOrFail方法:
 
  closure-use
 
  该方法的作用是使用事务将模型数据保存到数据库,这里面我们使用闭包返回保存状态,同时使用use关键字将父作用域的$options传递给该闭包以便其能够访问这个数据。
 
  此外,还支持传递多个父作用域变量到闭包,比如还是在Model类中的forceFill方法:
 
  closure-use-multi
 
  多个变量以逗号分隔即可。
 
  
  你会发现,PHP框架经常使用bindTo方法把路由URL映射到匿名回调函数上,框架会把匿名回调函数绑定到应用对象上,这样在匿名函数中就可以使用$this关键字引用重要的应用对象:
 
  class App {
      protected $routes = [];
      protected $responseStatus = '200 OK';
      protected $responseContentType = 'text/html';
      protected $responseBody = 'Laravel学院';
  
      public function addRoute($routePath, $routeCallback) {
          $this->routes[$routePath] = $routeCallback->bindTo($this, __CLASS__);
      }
  
      public function dispatch($currentPath) {
          foreach ($this->routes as $routePath => $callback) {
              if( $routePath === $currentPath) {
                  $callback();
              } //Cuoxin.com
          }
          header('HTTP/1.1 ' . $this->responseStatus);
          header('Content-Type: ' . $this->responseContentType);
          header('Content-Length: ' . mb_strlen($this->responseBody));
          echo $this->responseBody;
      }
  
  }
  这里我们需要重点关注addRoute方法,这个方法的参数分别是一个路由路径和一个路由回调,dispatch方法的参数是当前HTTP请求的路径,它会调用匹配的路由回调。第9行是重点所在,我们将路由回调绑定到了当前的App实例上。这么做能够在回调函数中处理App实例的状态:
 
  $app = new App();
  $app->addRoute(‘user/nonfu’, function(){
      $this->responseContentType = ‘application/json;charset=utf8’;
      $this->responseBody = ‘{“name”:”LaravelAcademy"}';
  });
  $app->dispatch(‘user/nonfu');
  在Larval底层也有用到bindTo方法,详见Illuminate/Support/Traits/Macroable的__call方法。

(编辑:ASP站长网)

    网友评论
    推荐文章
      热点阅读