资讯专栏INFORMATION COLUMN

PHP 闭包那些事儿

winterdawn / 434人阅读

摘要:注理论上讲闭包和匿名函数是不同的概念,不过将其视作相同的概念。匿名函数可以从父作用域继承变量,而这个父作用域是定义该闭包的函数不一定是调用它的函数。

匿名函数

匿名函数,也叫闭包函数,说白了就是“没有名字的函数”,和一般函数结构一样,只是少了函数名以及最后需要加上分号;

</>复制代码

  1. 注:理论上讲闭包和匿名函数是不同的概念,不过PHP将其视作相同的概念。

</>复制代码

  1. $func = function()
  2. {
  3. echo "Hello World" . PHP_EOL;
  4. };
  5. $func();

匿名函数和普通函数的区分有:

匿名函数也可以作为变量的值来使用。

匿名函数可以从父作用域继承变量,而这个父作用域是定义该闭包的函数(不一定是调用它的函数)。

</>复制代码

  1. $message = "hello";
  2. $example = function () use ($message) {
  3. return $message;
  4. };
  5. $message = "world";
  6. echo $example();
  7. 输出:hello

注意:必须使用use关键字将变量传递进去才行,具体见官方文档。

闭包类

定义一个闭包函数,其实就是实例化一个闭包类(Closure)对象:

</>复制代码

  1. $func = function()
  2. {
  3. echo "hello world" . PHP_EOL;
  4. };
  5. var_dump($func);
  6. 输出:
  7. object(Closure)#1 (0) {
  8. }

类摘要:

</>复制代码

  1. Closure {
  2. __construct ( void )
  3. public static Closure bind ( Closure $closure , object $newthis [, mixed $newscope = "static" ] )
  4. public Closure bindTo ( object $newthis [, mixed $newscope = "static" ] )
  5. }

除了以上方法,闭包还实现了一个__invoke()魔术方法,当尝试以调用函数的方式调用一个对象时,__invoke() 方法会被自动调用。

bindTo 方法

接下来我们来看看bindTo方法,通过该方法,我们可以把闭包的内部状态绑定到其他对象上。这里bindTo方法的第二个参数显得尤为重要,其作用是指定绑定闭包的那个对象所属的PHP类,这样,闭包就可以在其他地方访问绑定闭包的对象中受保护和私有的成员变量。

你会发现,PHP框架经常使用bindTo方法把路由URL映射到匿名回调函数上,框架会把匿名回调函数绑定到应用对象上,这样在匿名函数中就可以使用$this关键字引用重要的应用对象:

</>复制代码

  1. class App {
  2. protected $routes = [];
  3. protected $responseStatus = "200 OK";
  4. protected $responseContentType = "text/html";
  5. protected $responseBody = "Hello World";
  6. public function addRoute($path, $callback) {
  7. $this->routes[$path] = $callback->bindTo($this, __CLASS__);
  8. }
  9. public function dispatch($path) {
  10. foreach ($this->routes as $routePath => $callback) {
  11. if( $routePath === $path) {
  12. $callback();
  13. }
  14. }
  15. header("HTTP/1.1 " . $this->responseStatus);
  16. header("Content-Type: " . $this->responseContentType);
  17. header("Content-Length: " . mb_strlen($this->responseBody));
  18. echo $this->responseBody;
  19. }
  20. }

这里我们需要重点关注addRoute方法,这个方法的参数分别是一个路由路径和一个路由回调,dispatch方法的参数是当前HTTP请求的路径,它会调用匹配的路由回调。第9行是重点所在,我们将路由回调绑定到了当前的App实例上。这么做能够在回调函数中处理App实例的状态:

</>复制代码

  1. $app = new App();
  2. $app->addRoute(‘/user’, function(){
  3. $this->responseContentType = ‘application/json;charset=utf8’;
  4. $this->responseBody = "世界你好";
  5. });
  6. $app->dispatch("/user");
IoC 容器

匿名函数可以从父作用域继承变量,而这个父作用域是定义该闭包的函数(不一定是调用它的函数)。

利用这个特性,我们可以实现一个简单的控制反转IoC容器:

</>复制代码

  1. class Container
  2. {
  3. protected static $bindings;
  4. public static function bind($abstract, Closure $concrete)
  5. {
  6. static::$bindings[$abstract] = $concrete;
  7. }
  8. public static function make($abstract)
  9. {
  10. return call_user_func(static::$bindings[$abstract]);
  11. }
  12. }
  13. class talk
  14. {
  15. public function greet($target)
  16. {
  17. echo "Hello " . $target->getName();
  18. }
  19. }
  20. class A
  21. {
  22. public function getName()
  23. {
  24. return "World";
  25. }
  26. }
  27. // 创建一个talk类的实例
  28. $talk = new talk();
  29. // 将A类绑定至容器,命名为foo
  30. Container::bind("foo", function() {
  31. return new A;
  32. });
  33. // 通过容器取出实例
  34. $talk->greet(Container::make("foo")); // Hello World

上述例子中,只有在通过make方法获取实例的时候,实例才被创建,这样使得我们可以实现容器。

Laravel框架底层也大量使用了闭包以及bindTo方法,利用好闭包可以实现更多的高级特性如事件触发等。


以上为闭包学习笔记,部分参考了网上的一些文章。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/28403.html

相关文章

  • 2019前端面试那些事儿

    摘要:虽然今年没有换工作的打算但为了跟上时代的脚步还是忍不住整理了一份最新前端知识点知识点汇总新特性,语义化浏览器的标准模式和怪异模式和的区别使用的好处标签废弃的标签,和一些定位写法放置位置和原因什么是渐进式渲染模板语言原理盒模型,新特性,伪 虽然今年没有换工作的打算 但为了跟上时代的脚步 还是忍不住整理了一份最新前端知识点 知识点汇总1.HTMLHTML5新特性,语义化浏览器的标准模式和怪...

    JeOam 评论0 收藏0
  • 2019前端面试那些事儿

    摘要:虽然今年没有换工作的打算但为了跟上时代的脚步还是忍不住整理了一份最新前端知识点知识点汇总新特性,语义化浏览器的标准模式和怪异模式和的区别使用的好处标签废弃的标签,和一些定位写法放置位置和原因什么是渐进式渲染模板语言原理盒模型,新特性,伪 虽然今年没有换工作的打算 但为了跟上时代的脚步 还是忍不住整理了一份最新前端知识点 知识点汇总1.HTMLHTML5新特性,语义化浏览器的标准模式和怪...

    QLQ 评论0 收藏0
  • PHP那些事儿

    摘要:封禁策略为一个自然分钟内请求签到接口次则封禁该分钟,如何操作设置两个来支撑此问题获取封禁获取次数只提供内存,现在要做一个活动,参与活动的用户为,请问如何设计可考虑用的和命令来实现此需求,对其进行占位,并且各的占位才占有的空间,所有空间 1、封禁策略为一个自然分钟内请求签到接口500次则封禁该IP10分钟,如何操作? 2、只提供10M内存,现在要做一个活动,参与活动的用户userId为...

    X_AirDu 评论0 收藏0
  • 【nginx】nginx 配置那些事儿

    摘要:是一款具有高负载能力的服务器,也是架构的主要角色之一。多站点设置前面我们修改配置文件的代码位置,都是在下的里。如果想项目和项目均适用端口,则需要利用做反向代理设置。 nginx 是一款具有高负载能力的 web 服务器,也是 LNMP 架构的主要角色之一。现在越来越多的开发者选择 nginx 作为 php 的好搭档,替代 apache 的位置。下面我以 Mac 系统为例,介绍下 ngin...

    lunaticf 评论0 收藏0
  • 【nginx】nginx 配置那些事儿

    摘要:是一款具有高负载能力的服务器,也是架构的主要角色之一。多站点设置前面我们修改配置文件的代码位置,都是在下的里。如果想项目和项目均适用端口,则需要利用做反向代理设置。 nginx 是一款具有高负载能力的 web 服务器,也是 LNMP 架构的主要角色之一。现在越来越多的开发者选择 nginx 作为 php 的好搭档,替代 apache 的位置。下面我以 Mac 系统为例,介绍下 ngin...

    PrototypeZ 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<