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

溫馨提示×

溫馨提示×

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

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

如何將Laravel改成Swoole版

發布時間:2022-12-02 17:03:04 來源:億速云 閱讀:129 作者:iii 欄目:編程語言

這篇文章主要講解了“如何將Laravel改成Swoole版”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“如何將Laravel改成Swoole版”吧!

前言

不建議生產環境使用

創建一個新的 laravel 項目

laravel new swoole-laravel

將 Laravel 改成 Swoole 版

Laravel 的根目錄創建一個 swoole_server.php 文件,然后把 public/index.php 中的代碼復制過來

<?php
use Illuminate\Contracts\Http\Kernel;
use Illuminate\Http\Request;
define('LARAVEL_START', microtime(true));
require __DIR__.'/../vendor/autoload.php';
$app = require_once __DIR__.'/../bootstrap/app.php';
$kernel = $app->make(Kernel::class);
$response = $kernel->handle(
    $request = Request::capture()
)->send();
$kernel->terminate($request, $response);

第一步,框架文件的加載是肯定的,而且應該是在主進程中就加載好的,不需要子進程或者協程再去重復加載。因此,上面的 require 都不太需要動。

第二步,我們要啟動一個 HTTP 的 Swoole 服務,這個之前已經講過很多次了,注意,在 onRequest 中,我們應該將 $kernel 相關的代碼放入進去。

$http = new Swoole\Http\Server('0.0.0.0', 9501);
$http->on('Request', function ($req, $res) use($app) {
    try {
        $kernel = $app->make(Kernel::class);
        $response = $kernel->handle(
            $request = Request::capture()
        )->send();
        $kernel->terminate($request, $response);
    }catch(\Exception $e){
        print_r($e->getMessage());
    }
});
echo "服務啟動", PHP_EOL;
$http->start();

這樣就可以了嗎?要不你先試試看。正常情況下可能你是獲得不了任何的輸入和輸出的,這是為啥?

第三步,解決輸入問題,其實就是超全局變量在 Swoole 中是不起作用的,所以 $_GET 之類的變量都會失效,Laravel 中 Request 相關的對象都無法獲得數據了。這怎么辦呢?我們從 onRequest 的參數中拿這些數據,然后再放回到當前進程協程中的 $_GET 中就好啦。

$http->on('Request', function ($req, $res) use($app) {
    $_SERVER = [];
    if(isset($req->server)){
        foreach($req->server as $k => $v){
            $_SERVER[strtoupper($k)] = $v;
        }
    }
    $_GET = [];
    if(isset($req->get)){
        foreach ($req->get as $k => $v){
            $_GET[$k] = $v;
        }
    }
    $_POST = [];
    if(isset($req->post)){
        foreach ($req->post as $k => $v){
            $_POST[$k] = $v;
        }
    }
    try {
        $kernel = $app->make(Kernel::class);
        $response = $kernel->handle(
            $request = Request::capture()
        )->send();
        $kernel->terminate($request, $response);
    }catch(\Exception $e){
        print_r($e->getMessage());
    }
});

上面三段代碼,分別解決了 $_SERVER、$_GET 和 $_POST 的問題。現在你再試試,參數是可以接收到了,但輸出怎么是打印在控制臺的?

第四步,解決輸出問題,將框架中的所有輸出放到輸出緩沖區,然后再用 Swoole 的 Response 返回。

$http->on('Request', function ($req, $res) use($app) {
    $_SERVER = [];
    if(isset($req->server)){
        foreach($req->server as $k => $v){
            $_SERVER[strtoupper($k)] = $v;
        }
    }
    $_GET = [];
    if(isset($req->get)){
        foreach ($req->get as $k => $v){
            $_GET[$k] = $v;
        }
    }
    $_POST = [];
    if(isset($req->post)){
        foreach ($req->post as $k => $v){
            $_POST[$k] = $v;
        }
    }
    //把返回放到一個緩沖區里
    ob_start();
    try {
        $kernel = $app->make(Kernel::class);
        $response = $kernel->handle(
            $request = Request::capture()
        )->send();
        $kernel->terminate($request, $response);
    }catch(\Exception $e){
        print_r($e->getMessage());
    }
    $ob = ob_get_contents();
    ob_end_clean();
    $res->end($ob);
});

最后的 ob_start () 這些內容,也是我們之前學習過的內容,也就不多做解釋了。

全部代碼

<?php
use Illuminate\Contracts\Http\Kernel;
use Illuminate\Http\Request;
define('LARAVEL_START', microtime(true));
require __DIR__.'/vendor/autoload.php';
$app = require_once __DIR__.'/bootstrap/app.php';
$http = new Swoole\Http\Server('0.0.0.0', 9501);
$http->on('Request', function ($req, $res) use($app) {
    $_SERVER = [];
    if(isset($req->server)){
        foreach($req->server as $k => $v){
            $_SERVER[strtoupper($k)] = $v;
        }
    }
    $_GET = [];
    if(isset($req->get)){
        foreach ($req->get as $k => $v){
            $_GET[$k] = $v;
        }
    }
    $_POST = [];
    if(isset($req->post)){
        foreach ($req->post as $k => $v){
            $_POST[$k] = $v;
        }
    }
    //把返回放到一個緩沖區里
    ob_start();
    try {
        $kernel = $app->make(Kernel::class);
        $response = $kernel->handle(
            $request = Request::capture()
        )->send();
        $kernel->terminate($request, $response);
    }catch(\Exception $e){
        print_r($e->getMessage());
    }
    $ob = ob_get_contents();
    ob_end_clean();
    $res->end($ob);
});
echo "服務啟動", PHP_EOL;
$http->start();

至此,我們最簡單的框架改造就完成了,趕緊試試效果吧。

運行

php swoole_server.php

訪問

http://47.113.xxx.xx:9501/

試試協程效果

先定義一個路由。或者我們直接改造一下默認的路由。

Route::get('/', function () {
    echo Swoole\Coroutine::getCid(), "<br/>";
    print_r(Swoole\Coroutine::stats());
    Swoole\Coroutine::sleep(10);
    echo "<br/>";
    echo getmypid(), "<br/>";
//    return view('welcome');
});

打印了一堆東西,不過應該都比較熟悉吧,前兩個是協程 ID 和協程信息的輸出,然后我們 Swoole\Coroutine::sleep () 了 10 秒,再打印一下進程 ID 。

然后我們打開瀏覽器,準備兩個標簽一起訪問。

// 第一個訪問的頁面
1
Array
(
    [event_num] => 2
    [signal_listener_num] => 0
    [aio_task_num] => 0
    [aio_worker_num] => 0
    [aio_queue_size] => 0
    [c_stack_size] => 2097152
    [coroutine_num] => 1
    [coroutine_peak_num] => 1
    [coroutine_last_cid] => 1
)
1468
// 第二個訪問的頁面
2
Array
(
    [event_num] => 2
    [signal_listener_num] => 0
    [aio_task_num] => 0
    [aio_worker_num] => 0
    [aio_queue_size] => 0
    [c_stack_size] => 2097152
    [coroutine_num] => 2
    [coroutine_peak_num] => 2
    [coroutine_last_cid] => 2
)
1468

看出來了嗎?每個 onRequest 事件其實都是開了一個新的協程來處理請求所以它們的協程 ID 不同。同時,第二個請求不會因為第一個請求阻塞而等到 20 秒后才返回。最后在協程狀態中,我們還看到了第二個請求中顯示 coroutine_num 有兩個,說明當前有兩個協程在處理任務。最后,進程是相同的,它們都是走的同一個進程。

試試多進程效果

默認情況下,上面的代碼是一個主進程,一個 Worker 進程,然后再使用了協程能力。其實這樣的效果已經能秒殺普通的 PHP-FPM 效果了。但我們要充分利用多核機器的性能,也就是說,我們來開啟多進程,使用多進程 + 多協程的超強處理模式。最簡單的方式,直接設置 HTTP 服務的進程 Worker 數量即可。

$http->set(array(
    'worker_num' => 4,
      // 'worker_num' => 1,單進程
));

現在運行起服務器,可以看到多了幾個進程了。然后我們再新建一個測試路由

Route::get('/a', function () {
    echo Swoole\Coroutine::getCid(), "<br/>";
    print_r(Swoole\Coroutine::stats());
    echo "<br/>";
    echo getmypid(), "<br/>";
});

現在再次訪問首頁和這個 /a 頁面。

// 首頁一
1
Array
(
    [event_num] => 2
    [signal_listener_num] => 0
    [aio_task_num] => 0
    [aio_worker_num] => 0
    [aio_queue_size] => 0
    [c_stack_size] => 2097152
    [coroutine_num] => 1
    [coroutine_peak_num] => 1
    [coroutine_last_cid] => 1
)
1562
// 首頁二
1
Array
(
    [event_num] => 2
    [signal_listener_num] => 0
    [aio_task_num] => 0
    [aio_worker_num] => 0
    [aio_queue_size] => 0
    [c_stack_size] => 2097152
    [coroutine_num] => 1
    [coroutine_peak_num] => 1
    [coroutine_last_cid] => 1
)
1563
// /a 頁面
1
Array
(
    [event_num] => 2
    [signal_listener_num] => 0
    [aio_task_num] => 0
    [aio_worker_num] => 0
    [aio_queue_size] => 0
    [c_stack_size] => 2097152
    [coroutine_num] => 1
    [coroutine_peak_num] => 1
    [coroutine_last_cid] => 1
)
1564

發現沒有,它們的進程 ID 也都不同了吧,如果沒有阻塞,會優先切換進程,如果所有進程都有阻塞,則再循環創建協程進行進程內的處理。

感謝各位的閱讀,以上就是“如何將Laravel改成Swoole版”的內容了,經過本文的學習后,相信大家對如何將Laravel改成Swoole版這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節

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

AI

巴林右旗| 石家庄市| 酒泉市| 隆德县| 平凉市| 喜德县| 石阡县| 南丰县| 沂源县| 杭锦后旗| 安多县| 南投市| 绿春县| 商城县| 葫芦岛市| 马尔康县| 四平市| 武鸣县| 靖州| 漾濞| 浪卡子县| 怀仁县| 玉环县| 綦江县| 镇康县| 三原县| 鲁甸县| 彝良县| 曲靖市| 武胜县| 阆中市| 邵阳县| 上杭县| 徐汇区| 威宁| 海南省| 长岛县| 冀州市| 织金县| 静安区| 奉节县|