DIY 實現 ThinkPHP 核心框架 (九)Container 類

cn-five發表於2020-09-07

ThinkPHP 的 Container 類提供了靜態方法 get() ,可以根據類名或類的別名獲取例項,會保持建立完成的例項,避免重複建立。下面實現這個方法,修改 Container.php ,新增以下程式碼。

//     * ThinkPHP 5 與 6 在此處引數一致
//     * @param string $abstract
//     * @param array $vars
//     * @param bool $newInstance
//     */
    public static function get(string $abstract, array $vars = [], bool $newInstance = false)
    {
        return static::getInstance()->make($abstract, $vars, $newInstance);
    }

編寫 getInstance() 方法,並新增靜態屬性 $instance 儲存自身例項。

protected static $instance;

public static function getInstance()
    {
        // 建立自身例項
        if (is_null(static::$instance)) {
            static::$instance = new static;
        }
        return static::$instance;
    }

編寫 make() 方法。

public function make (string $abstract, array $vars = [], bool $newInstance = false)
    {
        // 這裡的 $abstract 是包含有名稱空間的類名
        if (isset($this->bind[$abstract])) {
            $abstract = $this->bind[$abstract];
        }

        // 如果已經例項化直接返回
        if (isset($this->instances[$abstract]) && !$newInstance) {
            return $this->instances[$abstract];
        }

        // 如果就建立
        $object = $this->invokeClass($abstract, $vars);

        // 儲存例項
        if (!$newInstance) {
        $this->instances[$abstract] = $object;
        }

        return $object;
    }

建立儲存屬性的別名陣列 $bind

    protected $bind = [
        'app' => App::class,
        'config' => Config::class,
        'request' => Request::class
    ];

編寫 invokeClass() 方法

public function invokeClass (string $class, array $vars = [])
    {
        // $vars 為建構函式的引數
        return new $class();
    }

修改入口檔案 index.php

require __DIR__ . '/../core/base.php';
use think\Request;

$req = \think\Container::get('request');
var_dump($req instanceof Request);

輸出 bool(true) 表示 get() 方法功能正常。

也可以利用魔術方法 __get()__set() ,實現外部物件直接操作容器例項。

    public function __get($abstract)
    {
        // 返回容器的類例項
        return $this->make($abstract);
    }

public function __set($abstract, $instance)
    {
        if (isset($this->bind[$abstract])) {
            $abstract = $this->bind[$abstract];
        }
        // 裝入容器
        $this->instances[$abstract] = $instance;
    }

index.php 進行測試

$container = \think\Container::getInstance();
// 獲取容器中的例項,輸出物件
var_dump($container->request);

// 裝入容器
$container->contianerName = $container;
var_dump($container->contianerName);

輸出 object(think\Request) 表示成功

本作品採用《CC 協議》,轉載必須註明作者和本文連結

相關文章