在swoole中製作一款仿製laravel的框架

VE視訊引擎發表於2021-03-15

首先需要確定一下思路:我希望基於swoole的擴充套件開發的程式碼在run起來的時候,在接收到ws或是tcp等訊息時,自動路由到某個類上,同時類可以實現載入類的依賴注入功能。目前市面上佔據主流的一款框架Laravel,其中有一個依賴注入的功能非常的便捷。一般在通常的框架中拉取Class是這樣做的:

class a {
    public $bClassInstance;
    public function __construct(Class b) {
        $classInstance = new b();
    }
    public function doSth() {
        return $this->bClassInstance->xxx();
    }
}

$b = new b();
$a = new a($b)
$a->doSth();

而在Laravel中則可以省略一些例項化的步驟, 直接通過型別約束的語法在方法的形參上指定某類的名稱空間就自動例項化該類進來了。

class a {
    public function doSth(b $b) {
        return $b->xxx();
    }
}

想要實現這一點,必須要了解php的反射機制。反射是一個比較冷門的類,他可以做到:使用namespace例項化一個類、呼叫類的方法等,利用這一點,可以構造一個自動裝箱的類。

<?php
/***
 *  依賴注入容器,若要執行依賴注入,請確保類包含建構函式!
 */
namespace App\Server;

class Container
{
    public $config;
    public $reflection;
    public function __construct($namespace)
    {
        try
        {
            $this->reflection = new \ReflectionClass($namespace);
        }
        catch (Exception $e)
        {
            echo $namespace;
        }
    }
    public function builderController($fn, $server, $frame, $userMessage)
    {
        //從route中得到的control名稱
        $this->reflection->getMethod($fn)->invoke($this->autoBuilder(), $server, $frame, $userMessage);
    }

    public function builderTask($fn, $server, $userMessage)
    {
        $this->reflection->getMethod($fn)->invoke($this->autoBuilder(), $server, $userMessage);
    }

    public function autoBuilder()
    {
        #對建構函式賦值
        return $this->batchInstantiation($this->getPrototypeController($this->reflection)#獲得字串
        );
    }

    protected final function getPrototypeController(\ReflectionClass $object)
    {
        $prototype = false;
        //批量從反射類中獲取原型字串
        foreach ($object->getConstructor()->getParameters() as $parameter)
        {
            $prototype[] = $parameter->getClass()->name;
        }

        return $prototype ?: [];
    }

    protected final function batchInstantiation(array $prototypeArr)
    {
        foreach ($prototypeArr as $item)
        {
            $container = new container($item);
            $insArr[]  = $container->autoBuilder();//進行遞迴注入
        }

        return empty($prototypeArr) ? $this->reflection->newInstance() : $this->reflection->newInstanceArgs($insArr);
    }
}

有了這個簡易的裝箱類後,可以著手實現類的路由功能,我們首先建立composer.json,鍵入如下內容。

{
    "require": {
 
    },
    "autoload": {
        "psr-4": {
        "App\\": "App/"
        }
   }
}

下一步,我們需要建立一個處理路由的類,這個類在常規的框架中,一般用來對映http請求到對應的類的函式上,而在swoole裡,請求會來自長連線。那麼在route類中則需要做相應的處理。

class Route
{
    public $websocketServer;
    public $model;
    public $cache;
    public function __construct() {
        $this->websocketServer = new \swoole_websocket_server("0.0.0.0", "8002");
    }
    public function start_ws() {
        // 這裡設定一些swoole的引數 ...
        // 最後執行啟動swoole
        $this->websocketServer->start();
    }
   
    public function ws_onMessage(\swoole_websocket_server $server, $frame)
    {
        $userMessage = $this->filter_arr(json_decode($frame->data, true));
        if (!$userMessage) {
            return false;
        }
        
        if (!$userMessage['type'] || !$userMessage['action']) {
            return $this->call_shell("Type or action not found! ");
        }
        //使用依賴注入容器做偽路由
        $App = new Container('\App\Controller\\'.$userMessage['type']);
        return $App->builderController($userMessage['action'], $server, $frame,$userMessage);
    }
    
}

最後一步,建立一個入口檔案,引導路由類的執行。

<?php
require "vendor/autoload.php";

use App\Server\Route;

$App = new Route();
$App->start_ws();

相關文章