摘要:注理論上講閉包和匿名函數是不同的概念,不過將其視作相同的概念。匿名函數可以從父作用域繼承變量,而這個父作用域是定義該閉包的函數不一定是調用它的函數。
匿名函數
匿名函數,也叫閉包函數,說白了就是“沒有名字的函數”,和一般函數結構一樣,只是少了函數名以及最后需要加上分號;。
注:理論上講閉包和匿名函數是不同的概念,不過PHP將其視作相同的概念。
$func = function() { echo "Hello World" . PHP_EOL; }; $func();
匿名函數和普通函數的區分有:
匿名函數也可以作為變量的值來使用。
匿名函數可以從父作用域繼承變量,而這個父作用域是定義該閉包的函數(不一定是調用它的函數)。
$message = "hello"; $example = function () use ($message) { return $message; }; $message = "world"; echo $example(); 輸出:hello
注意:必須使用use關鍵字將變量傳遞進去才行,具體見官方文檔。
閉包類定義一個閉包函數,其實就是實例化一個閉包類(Closure)對象:
$func = function() { echo "hello world" . PHP_EOL; }; var_dump($func); 輸出: object(Closure)#1 (0) { }
類摘要:
Closure { __construct ( void ) public static Closure bind ( Closure $closure , object $newthis [, mixed $newscope = "static" ] ) public Closure bindTo ( object $newthis [, mixed $newscope = "static" ] ) }
除了以上方法,閉包還實現了一個__invoke()魔術方法,當嘗試以調用函數的方式調用一個對象時,__invoke() 方法會被自動調用。
bindTo 方法接下來我們來看看bindTo方法,通過該方法,我們可以把閉包的內部狀態綁定到其他對象上。這里bindTo方法的第二個參數顯得尤為重要,其作用是指定綁定閉包的那個對象所屬的PHP類,這樣,閉包就可以在其他地方訪問綁定閉包的對象中受保護和私有的成員變量。
你會發現,PHP框架經常使用bindTo方法把路由URL映射到匿名回調函數上,框架會把匿名回調函數綁定到應用對象上,這樣在匿名函數中就可以使用$this關鍵字引用重要的應用對象:
class App { protected $routes = []; protected $responseStatus = "200 OK"; protected $responseContentType = "text/html"; protected $responseBody = "Hello World"; public function addRoute($path, $callback) { $this->routes[$path] = $callback->bindTo($this, __CLASS__); } public function dispatch($path) { foreach ($this->routes as $routePath => $callback) { if( $routePath === $path) { $callback(); } } 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’, function(){ $this->responseContentType = ‘application/json;charset=utf8’; $this->responseBody = "世界你好"; }); $app->dispatch("/user");IoC 容器
匿名函數可以從父作用域繼承變量,而這個父作用域是定義該閉包的函數(不一定是調用它的函數)。
利用這個特性,我們可以實現一個簡單的控制反轉IoC容器:
class Container { protected static $bindings; public static function bind($abstract, Closure $concrete) { static::$bindings[$abstract] = $concrete; } public static function make($abstract) { return call_user_func(static::$bindings[$abstract]); } } class talk { public function greet($target) { echo "Hello " . $target->getName(); } } class A { public function getName() { return "World"; } } // 創建一個talk類的實例 $talk = new talk(); // 將A類綁定至容器,命名為foo Container::bind("foo", function() { return new A; }); // 通過容器取出實例 $talk->greet(Container::make("foo")); // Hello World
上述例子中,只有在通過make方法獲取實例的時候,實例才被創建,這樣使得我們可以實現容器。
在Laravel框架底層也大量使用了閉包以及bindTo方法,利用好閉包可以實現更多的高級特性如事件觸發等。
以上為閉包學習筆記,部分參考了網上的一些文章。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/28403.html
摘要:雖然今年沒有換工作的打算但為了跟上時代的腳步還是忍不住整理了一份最新前端知識點知識點匯總新特性,語義化瀏覽器的標準模式和怪異模式和的區別使用的好處標簽廢棄的標簽,和一些定位寫法放置位置和原因什么是漸進式渲染模板語言原理盒模型,新特性,偽 雖然今年沒有換工作的打算 但為了跟上時代的腳步 還是忍不住整理了一份最新前端知識點 知識點匯總1.HTMLHTML5新特性,語義化瀏覽器的標準模式和怪...
摘要:雖然今年沒有換工作的打算但為了跟上時代的腳步還是忍不住整理了一份最新前端知識點知識點匯總新特性,語義化瀏覽器的標準模式和怪異模式和的區別使用的好處標簽廢棄的標簽,和一些定位寫法放置位置和原因什么是漸進式渲染模板語言原理盒模型,新特性,偽 雖然今年沒有換工作的打算 但為了跟上時代的腳步 還是忍不住整理了一份最新前端知識點 知識點匯總1.HTMLHTML5新特性,語義化瀏覽器的標準模式和怪...
摘要:是一款具有高負載能力的服務器,也是架構的主要角色之一。多站點設置前面我們修改配置文件的代碼位置,都是在下的里。如果想項目和項目均適用端口,則需要利用做反向代理設置。 nginx 是一款具有高負載能力的 web 服務器,也是 LNMP 架構的主要角色之一。現在越來越多的開發者選擇 nginx 作為 php 的好搭檔,替代 apache 的位置。下面我以 Mac 系統為例,介紹下 ngin...
摘要:是一款具有高負載能力的服務器,也是架構的主要角色之一。多站點設置前面我們修改配置文件的代碼位置,都是在下的里。如果想項目和項目均適用端口,則需要利用做反向代理設置。 nginx 是一款具有高負載能力的 web 服務器,也是 LNMP 架構的主要角色之一。現在越來越多的開發者選擇 nginx 作為 php 的好搭檔,替代 apache 的位置。下面我以 Mac 系統為例,介紹下 ngin...
閱讀 3925·2021-10-12 10:12
閱讀 2886·2021-09-10 11:18
閱讀 3673·2019-08-30 15:54
閱讀 2808·2019-08-30 15:53
閱讀 638·2019-08-30 13:54
閱讀 965·2019-08-30 13:21
閱讀 2260·2019-08-30 12:57
閱讀 1687·2019-08-30 11:10