中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

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

ThinkPHP容器之容器是如何返回實例的

發布時間:2020-12-10 10:23:32 來源:億速云 閱讀:193 作者:小新 欄目:編程語言

這篇文章主要介紹ThinkPHP容器之容器是如何返回實例的,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!

Container實例調用make方法

本文沒有太多文字解析,都在代碼注釋中說明了執行過程。

代碼static::getInstance()返回了Container的實例后,就會去調用本類的make方法,接下來就是對make方法進行詳解了。

ThinkPHP容器之容器是如何返回實例的在開始閱讀make方法里邊的源碼之前,我們需要先對幾個屬性進行簡單的梳理一下。

這四個屬性一定要有點印象,并且一定要區別instance和instances。

這倆個屬性一個是單例模式返回當前類的實例,一個是容器中的所有的實例。

ThinkPHP容器之容器是如何返回實例的第一次執行結果

   /**     * 創建類的實例     * @access public     * @param  string        $abstract       類名或者標識     * @param  array|true    $vars           變量     * @param  bool          $newInstance    是否每次創建新的實例     * @return object     */    public function make($abstract, $vars = [], $newInstance = false)    {        // 判斷$vars這個變量是否為true        if (true === $vars) {            // 總是創建新的實例化對象            $newInstance = true;            $vars        = [];        }        // app  這里就是在容器別名里獲取傳遞過來的app    如果沒有則就是app        $abstract = isset($this->name[$abstract]) ? $this->name[$abstract] : $abstract;                // 從容器實例中獲取  如果存在則直接返回對應的實例  也就是使用注冊樹模式        if (isset($this->instances[$abstract]) && !$newInstance) {            return $this->instances[$abstract];        }        // think\App 從容器標識中獲取        if (isset($this->bind[$abstract])) {            // 將think\App 復制給$concrete變量            $concrete = $this->bind[$abstract];            // 用于代表匿名函數的類  判斷是不是閉包            if ($concrete instanceof Closure) {                $object = $this->invokeFunction($concrete, $vars);            } else {                // $this->name['app'] = think\App                $this->name[$abstract] = $concrete;                // 在執行一次本類的make方法,也就是本方法                return $this->make($concrete, $vars, $newInstance);            }        } else {            $object = $this->invokeClass($abstract, $vars);        }        if (!$newInstance) {            $this->instances[$abstract] = $object;        }        return $object;    }

這是第二次執行流程

    public function make($abstract, $vars = [], $newInstance = false)    {        // 判斷$vars這個變量是否為true        if (true === $vars) {            // 總是創建新的實例化對象            $newInstance = true;            $vars        = [];        }        // app  這里就是在容器別名里獲取傳遞過來的app    如果沒有則就是app        // 第二次執行時 $abstract = think\App        $abstract = isset($this->name[$abstract]) ? $this->name[$abstract] : $abstract;        // 從容器實例中獲取  如果存在則直接返回對應的實例  也就是使用注冊樹模式        if (isset($this->instances[$abstract]) && !$newInstance) {            return $this->instances[$abstract];        }        // think\App 從容器標識中獲取        // 第二次執行$this->bind['think\App']不存在走else        if (isset($this->bind[$abstract])) {            // 將think\App 復制給$concrete變量            $concrete = $this->bind[$abstract];            // 用于代表匿名函數的類  判斷是不是閉包            if ($concrete instanceof Closure) {                $object = $this->invokeFunction($concrete, $vars);            } else {                // $this->name['app'] = think\App                $this->name[$abstract] = $concrete;                // 在執行一次本類的make方法,也就是本方法                // think\App                return $this->make($concrete, $vars, $newInstance);            }        } else {            // think\App            $object = $this->invokeClass($abstract, $vars);        }        if (!$newInstance) {            // 把創建的容器存起來            //$this->instances['think\App'] = $object;            $this->instances[$abstract] = $object;        }        return $object;    }
public function invokeClass($class, $vars = [])    {        try {            /**             * ReflectionClass Object                (                [name] => think\App                )             */            // 這里就是之前文章提到的反射            $reflect = new ReflectionClass($class);            if ($reflect->hasMethod('__make')) {                $method = new ReflectionMethod($class, '__make');                if ($method->isPublic() && $method->isStatic()) {                    $args = $this->bindParams($method, $vars);                    return $method->invokeArgs(null, $args);                }            }            // 通過反射獲取think\App的構造函數            $constructor = $reflect->getConstructor();            $args = $constructor ? $this->bindParams($constructor, $vars) : [];            // 從給出的參數創建一個新的類實例            return $reflect->newInstanceArgs($args);        } catch (ReflectionException $e) {            throw new ClassNotFoundException('class not exists: ' . $class, $class);        }    }

執行流程圖

既然把代碼都理清楚了,這時來理一下執行的流程圖可以看的更清晰。

以上是“ThinkPHP容器之容器是如何返回實例的”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

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

AI

洛扎县| 来安县| 四子王旗| 博野县| 札达县| 崇左市| 红河县| 肃宁县| 琼海市| 镇宁| 定南县| 容城县| 根河市| 景德镇市| 精河县| 昭平县| 诸城市| 沙湾县| 凤城市| 花垣县| 德江县| 古丈县| 眉山市| 泰安市| 淮安市| 文水县| 鹤庆县| 个旧市| 闽清县| 航空| 乡城县| 吉水县| 正宁县| 泸西县| 千阳县| 灵宝市| 务川| 黄山市| 西安市| 青冈县| 广水市|