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

在swoole中製做一款仿製laravel的框架
首先須要肯定一下思路:我但願基於swoole的擴展開發的代碼在run起來的時候,在接收到ws或是tcp等消息時,自動路由到某個類上,同時類能夠實現加載類的依賴注入功能。目前市面上佔據主流的一款框架Laravel,其中有一個依賴注入的功能很是的便捷。通常在一般的框架中拉取Class是這樣作的:php

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中則能夠省略一些實例化的步驟, 直接經過類型約束的語法在方法的形參上指定某類的命名空間就自動實例化該類進來了。laravel

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

想要實現這一點,必需要了解php的反射機制。反射是一個比較冷門的類,他能夠作到:使用namespace實例化一個類、調用類的方法等,利用這一點,能夠構造一個自動裝箱的類。web

<?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,鍵入以下內容。shell

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

下一步,咱們須要建立一個處理路由的類,這個類在常規的框架中,通常用來映射http請求到對應的類的函數上,而在swoole裏,請求會來自長鏈接。那麼在route類中則須要作相應的處理。json

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);
    }
    
}

最後一步,建立一個入口文件,引導路由類的執行。websocket

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

use App\Server\Route;

$App = new Route();
$App->start_ws();
相關文章
相關標籤/搜索