亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

PHP中怎么使用Closure類

發布時間:2021-03-06 09:58:59 來源:億速云 閱讀:246 作者:小新 欄目:編程語言

小編給大家分享一下PHP中怎么使用Closure類,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

Closure 類

用于代表匿名函數的類。

匿名函數(在 PHP 5.3 中被引入)會產生這個類型的對象。在過去,這個類被認為是一個實現細節,但現在可以依賴它做一些事情。自 PHP 5.4 起,
這個類帶有一些方法,允許在匿名函數創建后對其進行更多的控制。

這個類不能實例化,里面主要有兩個方法,都用來復制閉包,一個靜態一個動態,下面分別詳細講解下這兩個不好理解的方法。

Closure::bind

public static Closure Closure::bind ( Closure $closure , object $newthis [, mixed $newscope = 'static' ] )

參數說明:
closure
需要綁定的匿名函數。

newthis
需要綁定到匿名函數的對象,或者 NULL 創建未綁定的閉包。

newscope
想要綁定給閉包的類作用域,或者 'static' 表示不改變。如果傳入一個對象,則使用這個對象的類型名。 類作用域用來決定在閉包中 $this 對象的 
私有、保護方法 的可見性。 The class scope to which associate the closure is to be associated, or 'static' to keep the 
current one. If an object is given, the type of the object will be used instead. This determines the visibility of 
protected and private methods of the bound object.

上面是該方法的定義,第一個參數很好理解,就是一個閉包函數;第二個參數就不太好理解,如果要復制的閉包中包含$this,這個對象就表示這個
$this,閉包函數里面對這個對象的修改在調用結束之后也會保持一致,比如修改了一個屬性;第三個參數就不太好理解了,看官方的說明也是云里霧里的,
默認參數情況下,調用$this->訪問object $newthis中的屬性函數的時候,會有限制,只能訪問public屬性的函數,如果想訪問protected/private屬性,
就要設置為對應的類名/類實例,就要像在類里面一樣,要訪問那個類的保護/私有屬性函數。

例子
<?php
class T {
    private function show()
    {
        echo "我是T里面的私有函數:show\n";
    }

    protected  function who()
    {
        echo "我是T里面的保護函數:who\n";
    }

    public function name()
    {
        echo "我是T里面的公共函數:name\n";
    }
}

$test = new T();

$func = Closure::bind(function(){
    $this->who();
    $this->name();
    $this->show();
}, $test);

$func();

上面的代碼會報錯Fatal error: Uncaught Error: Call to protected method T::who() from  context 'Closure'。 加上bind第三個參數為t::class或者new T(),會正常輸出每一個結果。

我是T里面的保護函數:who
我是T里面的公共函數:name
我是T里面的私有函數:show

當然了,閉包也可以傳遞參數

$test = new StdClass();
var_dump($test);

$func = Closure::bind(function($obj){
    $obj->name = "燕睿濤";
}, null);

$func($test);
var_dump($test);

上面的程序跟匿名函數一樣,啥對象也沒有依賴,上面的程序會輸出:

object(stdClass)#1 (0) {
}
object(stdClass)#1 (1) {
  ["name"]=>
  string(9) "燕睿濤"
}

另外還有個特別要說明的例子

<?php
class T {
    private function show()
    {
        echo "我是T里面的私有函數:show\n";
    }

    protected  function who()
    {
        echo "我是T里面的保護函數:who\n";
    }

    public function name()
    {
        echo "我是T里面的公共函數:name\n";
    }
}

$func = Closure::bind(function ($obj) {
    $obj->show();
}, null);

$test = new T();

$func($test);

上面的情況會輸出什么呢,沒錯,會報錯,提示訪問不了私有屬性show,這個時候,加上第三個參數就可以了,看了第三個參數不光影響$this的作用域,
也可以影響參數的作用域。

Closure::bindTo

bindTobind功能類似,這里只是另外一種形式,都是復制當前閉包對象,綁定指定的$this對象和類作用域。,參數比bind少了第一個,
后面兩個一樣,當然還有一個區別就是bindTo不是靜態方法,是閉包才會存在的一個屬性方法。

例子
<?php
class T {
    private function show()
    {
        echo "我是T里面的私有函數:show\n";
    }

    protected  function who()
    {
        echo "我是T里面的保護函數:who\n";
    }

    public function name()
    {
        echo "我是T里面的公共函數:name\n";
    }
}

$func = function () {
    $this->show();
    $this->who();
    $this->name();
};

$funcNew = $func->bindTo(new T(), T::class);

$funcNew();

上面函數的輸出和bind的類似

我是T里面的私有函數:show
我是T里面的保護函數:who
我是T里面的公共函數:name

一個trick

這個函數是在看composer生成的自動加載源碼的時候碰到的,在composer中用的比較特別,下面是截取部分composer中的代碼

// 文件autoload_real.php
call_user_func(\Composer\Autoload\ComposerStaticInit898ad46cb49e20577400c63254121bac::getInitializer($loader));

// 文件autoload_static.php
public static function getInitializer(ClassLoader $loader)
{
    return \Closure::bind(function () use ($loader) {
        $loader->prefixLengthsPsr4 = ComposerStaticInit25885cdf386fdaafc0bce14bb5a7d06e::$prefixLengthsPsr4;
        $loader->prefixDirsPsr4 = ComposerStaticInit25885cdf386fdaafc0bce14bb5a7d06e::$prefixDirsPsr4;
        $loader->prefixesPsr0 = ComposerStaticInit25885cdf386fdaafc0bce14bb5a7d06e::$prefixesPsr0;
        $loader->classMap = ComposerStaticInit25885cdf386fdaafc0bce14bb5a7d06e::$classMap;

    }, null, ClassLoader::class);
}

上面的代碼比較奇特,在call_user_func中,第一感覺是傳錯參數了,其實不然,這里調用了一個函數,這個函數會返回一個Closure對象,
也就是一個匿名函數,最終傳入的參數還是一個callable類型。再看看這個返回的閉包,里面使用了use,這是連接閉包和外部變量的橋梁。
至于這里為什么普通傳參數就可以,是因為php5里面,對象形參和實參數指向相同的對象,函數里面對對象的修改會反映到對象外面。

所以,上面這么做是沒問題的,還有另外一種形式也可以

call_user_func(\Composer\Autoload\ComposerStaticInit898ad46cb49e20577400c63254121bac::getInitializer(), $loader);

public static function getInitializer()
{
    return \Closure::bind(function ($loader) {
        $loader->prefixLengthsPsr4 = ComposerStaticInit25885cdf386fdaafc0bce14bb5a7d06e::$prefixLengthsPsr4;
        $loader->prefixDirsPsr4 = ComposerStaticInit25885cdf386fdaafc0bce14bb5a7d06e::$prefixDirsPsr4;
        $loader->prefixesPsr0 = ComposerStaticInit25885cdf386fdaafc0bce14bb5a7d06e::$prefixesPsr0;
        $loader->classMap = ComposerStaticInit25885cdf386fdaafc0bce14bb5a7d06e::$classMap;

    }, null, ClassLoader::class);
}

以上是“PHP中怎么使用Closure類”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

洛宁县| 宜章县| 柏乡县| 昭苏县| 大荔县| 桂阳县| 霍林郭勒市| 抚远县| 平凉市| 南城县| 云龙县| 阿巴嘎旗| 康保县| 萨迦县| 九江县| 织金县| 巴林左旗| 安图县| 乌兰浩特市| 宁陵县| 罗田县| 金沙县| 平果县| 开平市| 南昌县| 阳泉市| 永年县| 崇信县| 句容市| 靖边县| 石门县| 顺义区| 临夏县| 柘荣县| 且末县| 兴宁市| 宁武县| 怀集县| 扶风县| 南部县| 西和县|