站長資訊網
最全最豐富的資訊網站

嘗試怎么將Laravel改成Swoole版

本篇文章給大家Laravel和Swoole的相關知識,主要內容是教大家怎么將Laravel改成Swoole版(嘗試學習,不建議更改現有項目),下面一起來看看吧,希望對大家有所幫助!

前言

不建議生產環境使用

創建一個新的 laravel 項目

laravel new swoole-laravel
登錄后復制

將 Laravel 改成 Swoole 版

Laravel 的根目錄創建一個 swoole_server.php 文件,然后把 public/index.php 中的代碼復制過來【推薦學習:laravel視頻教程】

<?php use IlluminateContractsHttpKernel; use IlluminateHttpRequest; 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 SwooleHttpServer('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 IlluminateContractsHttpKernel; use IlluminateHttpRequest; define('LARAVEL_START', microtime(true)); require __DIR__.'/vendor/autoload.php'; $app = require_once __DIR__.'/bootstrap/app.php'; $http = new SwooleHttpServer('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 SwooleCoroutine::getCid(), "<br/>";     print_r(SwooleCoroutine::stats());     SwooleCoroutine::sleep(10);     echo "<br/>";     echo getmypid(), "<br/>"; //    return view('welcome'); });
登錄后復制

打印了一堆東西,不過應該都比較熟悉吧,前兩個是協程 ID 和協程信息的輸出,然后我們 SwooleCoroutine::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 SwooleCoroutine::getCid(), "<br/>";     print_r(SwooleCoroutine::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 也都不同了吧,如果沒有阻塞,會優先切換進程,如果所有進程都有阻塞,則再循環創建協程進行進程內的處理。

贊(0)
分享到: 更多 (0)
網站地圖   滬ICP備18035694號-2    滬公網安備31011702889846號
97热久久免费频精品99| 国产美女久久精品香蕉69| 精品亚洲成a人在线观看| 777精品成人影院| 久久国产精品无码HDAV| 久久福利青草精品资源站免费| 国产成人亚洲精品91专区手机 | 国产成人久久久精品二区三区| 欧美日韩久久久精品A片| 国产亚洲精品第一综合| 日韩AV无码精品一二三区| 久9视频这里只有精品8| 久久99国产精品久久99小说| 久久久久久久精品成人热色戒| 久久精品九九亚洲精品| 无码国产精品一区二区免费3p| 国产精品内射后入合集| 精品九九久久国内精品| 国产999精品2卡3卡4卡| 国模精品一区二区三区| 国模和精品嫩模私拍视频| 中国大陆精品视频XXXX| 中文字幕在线久热精品| 中文字幕日韩专区精品系列| 国产在线观看一区二区三区精品| 久久99精品久久久久久齐齐| 国产精品午夜福利在线无码| 国产精品电影在线| 国产偷伦精品视频| 国产内地精品毛片视频| 久久国产成人精品国产成人亚洲| 精品国产91久久久久久久a| 国产麻豆精品入口在线观看| 久久久久亚洲精品天堂久久久久久| 国产网红主播无码精品| 久久精品国产72国产精福利| 国产亚洲精品线观看动态图| 亚洲精品无码精品mV在线观看| 国产精品无套内射迪丽热巴| 国产成人精品电影在线观看 | 久久成人国产精品|