摘要:服務(wù)重點基本概述協(xié)議是基于的一種新的網(wǎng)絡(luò)協(xié)議。被調(diào)用者通過狀態(tài)通知機制等來通知調(diào)用者,或通過回調(diào)函數(shù)來處理結(jié)果阻塞和非阻塞關(guān)注的是調(diào)用者等待被調(diào)用者返回調(diào)用結(jié)果時的狀態(tài)。
一、PHP7源碼安裝和Swoole源碼編譯安裝 1.1 PHP7源碼安裝 1.1.1 獲取源碼與安裝
????獲取PHP7源碼:www.php.net
tar -xzvf ... # 解壓命令 ./configure --prefix=/home/study/php # 安裝至某個路徑,提前安裝gcc等 make # 編譯 make install # 安裝
????源碼執(zhí)行文件放在:bin目錄下
php -m # 查看 PHP 安裝的擴展1.1.2 簡化PHP執(zhí)行命令
????alias 命令=命令的絕對路徑
vim /.bash_profile alias php=/home/work/soft/php/bin/php # 添加 source /.bash_profile # 注意
????source FileName
????作用:在當(dāng)前bash環(huán)境下讀取并執(zhí)行FileName中的命令。????用于重新執(zhí)行剛修改的初始化文檔,如 .bash_profile 和 .profile 等等
????注:該命令通常用命令“.”來替代
????如:source /etc/profile 與 . /etc/profile是等效的
php -i | grep php.ini # 查找PHP的配置文件1.2 Swoole源碼編譯安裝
獲取swoole源碼:https://gitee.com/swoole/swoole.git
phpize是用來擴展php模塊的,通過phpize可以建立php的外掛模塊,解決沒有configure問題
/usr/local/php/bin/phpize # 在需要執(zhí)行的目錄執(zhí)行這行代碼即可 ./configure --with-php-config=/usr/local/php/bin/php-config
make make install
????最后可以在PHP的擴展目錄中看見swoole.so 擴展文件
????
1.3 雙劍合璧,PHP7支持swoole????在php.ini文件中添加:extension=swoole.so
????查看是否添加成功:php -m
????在swoole/examples/server下執(zhí)行php echo.php
????查看是否執(zhí)行端口:9501
netstat -anp|grep 9501二、玩轉(zhuǎn)網(wǎng)絡(luò)通信引擎(非常重要) 2.1 TCP服務(wù)&TCP客戶端 2.1.1 TCP服務(wù)
Swoole官網(wǎng)文檔:創(chuàng)建TCP服務(wù)器 | 創(chuàng)建UDP服務(wù)器
//創(chuàng)建Server對象,監(jiān)聽 127.0.0.1:9501端口 $serv = new swoole_server("127.0.0.1", 9501); //swoole_server->set函數(shù)用于設(shè)置swoole_server運行時的各項參數(shù) $serv->set([ "worker_num" => 6 , // worker進程數(shù),cpu 1-4倍 "max_request" => 10000, ]); /** * 監(jiān)聽連接進入事件 * $fd 客戶端連接的唯一標(biāo)示 * $reactor_id 線程id */ $serv->on("connect", function ($serv, $fd, $reactor_id) { echo "Client: {$reactor_id} - {$fd}-Connect. "; }); /** * 監(jiān)聽數(shù)據(jù)接收事件 * $reactor_id = $from_id */ $serv->on("receive", function ($serv, $fd, $reactor_id, $data) { $serv->send($fd, "Server: {$reactor_id} - {$fd}".$data); }); //監(jiān)聽連接關(guān)閉事件 $serv->on("close", function ($serv, $fd) { echo "Client: Close. "; }); //啟動服務(wù)器 $serv->start();
????測試tcp服務(wù)器方法:
netstat -anp | grep 9501
通過telnet方式登錄遠程主機:telnet 127.0.0.1 9501
tcp客戶端腳本
????查看當(dāng)前worker進程數(shù):ps -aft | grep tcp_server.php
????
????Tips:為了保證程序執(zhí)行的完整性,當(dāng)修改tcp服務(wù)器腳本后最好設(shè)置平滑重啟worker進程
????平滑重啟worker進程
????
connect("127.0.0.1", 9501)) { echo "連接失敗"; exit; } // php cli常量 fwrite(STDOUT, "請輸入消息:"); $msg = trim(fgets(STDIN)); // 發(fā)送消息給 tcp server服務(wù)器 $client->send($msg); // 接受來自server 的數(shù)據(jù) $result = $client->recv(); echo $result;2.2 HTTP服務(wù)(常用)
????
$http = new swoole_http_server("0.0.0.0", 8811); //添加測試一:獲取參數(shù)并打印出來 //$http->on("request", function ($request, $response) { // $response->cookie("singwa","xsssss", time() + 1800); // $response->end("sss".json_encode($request->get)); //}); /** * https://wiki.swoole.com/wiki/page/783.html * 配置靜態(tài)文件根目錄,與enable_static_handler配合使用。 * 設(shè)置document_root并設(shè)置enable_static_handler為true后, * 底層收到Http請求會先判斷document_root路徑下是否存在此文件, * 如果存在會直接發(fā)送文件內(nèi)容給客戶端,不再觸發(fā)onRequest回調(diào)。 */ $http->set( [ "enable_static_handler" => true, "document_root" => "/home/work/hdtocs/swoole_mooc/data", ] ); $http->on("request", function($request, $response) { //print_r($request->get); $content = [ "date:" => date("Ymd H:i:s"), "get:" => $request->get, "post:" => $request->post, "header:" => $request->header, ]; swoole_async_writefile(__DIR__."/access.log", json_encode($content).PHP_EOL, function($filename){ // todo }, FILE_APPEND); $response->cookie("singwa", "xsssss", time() + 1800); $response->end("sss". json_encode($request->get)); }); $http->start();2.3 WebSocket服務(wù)(重點) 2.3.1 基本概述
????WebSocket協(xié)議是基于TCP的一種新的網(wǎng)絡(luò)協(xié)議。它實現(xiàn)了瀏覽器與服務(wù)器全雙工(full-duplex)通信--允許服務(wù)器主動發(fā)送信息給客戶端
????為什么需要WebSocket
缺陷:HTTP的通信只能由客戶端發(fā)起
????WebSocket特點
建立在TCP協(xié)議之上
性能開銷小通信高效
客戶端可以與任意服務(wù)器通信
協(xié)議標(biāo)識符ws wss
持久化網(wǎng)絡(luò)通信協(xié)議
2.3.2 案例實現(xiàn) 2.3.2.1 服務(wù)端實現(xiàn)????1. 面向過程:procedure_ws_server.php
$server = new swoole_websocket_server("0.0.0.0", 9912); //配置靜態(tài)文件根目錄,可選 $server->set( [ "enable_static_handler" => true, "document_root" => "/home/wwwroot/www.lingyuan88.com/public/swoole/data", ] ); //監(jiān)聽websocket連接打開事件 $server->on("open", "onOpen"); function onOpen($server, $request) { print_r($request->fd); } // 監(jiān)聽ws消息事件 $server->on("message", function (swoole_websocket_server $server, $frame) { echo "receive from {$frame->fd}:{$frame->data},opcode:{$frame->opcode},fin:{$frame->finish} "; $server->push($frame->fd, "singwa-push-secesss"); }); $server->on("close", function ($ser, $fd) { echo "client {$fd} closed "; }); $server->start();
????2. WebSocket服務(wù)優(yōu)化,基礎(chǔ)類庫面向?qū)ο螅?/b>object_ws_server.php
class Ws { CONST HOST = "0.0.0.0"; CONST PORT = 9912; public $ws = null; public function __construct() { $this->ws = new swoole_websocket_server(self::HOST, self::PORT); //配置靜態(tài)文件根目錄,可選 $this->ws->set( [ "enable_static_handler" => true, "document_root" => "/home/wwwroot/www.lingyuan88.com/public/swoole/data", ] ); $this->ws->on("open", [$this, "onOpen"]); $this->ws->on("message", [$this, "onMessage"]); $this->ws->on("close", [$this, "onClose"]); $this->ws->start(); } /** * 監(jiān)聽ws連接事件 * @param $ws * @param $request */ public function onOpen($ws, $request) { print_r($request->fd); } /** * 監(jiān)聽ws消息事件 * @param $ws * @param $frame */ public function onMessage($ws, $frame) { echo "ser-push-message:{$frame->data} "; $ws->push($frame->fd, "server-push:".date("Y-m-d H:i:s")); } /** * close * @param $ws * @param $fd */ public function onClose($ws, $fd) { echo "clientid:{$fd} "; } } $obj = new Ws();2.3.2.2 客戶端實現(xiàn)
????ws_client.html
2.3.2.3 測試singwa-swoole-ws測試
????1. 通過WebSocket靜態(tài)文件目錄測試
????
????2. 通過HTTP服務(wù)測試
????
2.4 異步Task任務(wù)使用(重點)????使用場景
執(zhí)行耗時的操作(發(fā)送郵件 廣播等)
????注意:
投遞異步任務(wù)之后程序會繼續(xù)往下執(zhí)行,不會等待任務(wù)執(zhí)行完后再繼續(xù)向下執(zhí)行
class Ws { CONST HOST = "0.0.0.0"; CONST PORT = 9912; public $ws = null; public function __construct() { $this->ws = new swoole_websocket_server(self::HOST, self::PORT); $this->ws->set( [ "worker_num" => 2, "task_worker_num" => 2, ] ); //注冊Server的事件回調(diào)函數(shù) $this->ws->on("open", [$this, "onOpen"]); $this->ws->on("message", [$this, "onMessage"]); $this->ws->on("task", [$this, "onTask"]); $this->ws->on("finish", [$this, "onFinish"]); $this->ws->on("close", [$this, "onClose"]); $this->ws->start(); } /** * 監(jiān)聽ws連接事件 * @param $ws * @param $request */ public function onOpen($ws, $request) { var_dump($request->fd); } /** * 監(jiān)聽ws消息事件 * @param $ws * @param $frame */ public function onMessage($ws, $frame) { echo "ser-push-message:{$frame->data} "; // todo 10s $data = [ "task" => 1, "fd" => $frame->fd, ]; //投遞異步任務(wù) //注意:程序會繼續(xù)往下執(zhí)行,不會等待任務(wù)執(zhí)行完后再繼續(xù)向下執(zhí)行 $ws->task($data); //客戶端會馬上收到以下信息 $ws->push($frame->fd, "server-push:".date("Y-m-d H:i:s")); } /** * @param $serv * @param $taskId * @param $workerId * @param $data * @return string */ public function onTask($serv, $taskId, $workerId, $data) { print_r($data); // 耗時場景 10s sleep(10); return "on task finish"; // 告訴worker,并返回給onFinish的$data } /** * @param $serv * @param $taskId * @param $data */ public function onFinish($serv, $taskId, $data) { echo "taskId:{$taskId} "; echo "finish-data-sucess:{$data} "; } /** * close * @param $ws * @param $fd */ public function onClose($ws, $fd) { echo "clientid:{$fd} "; } } $obj = new Ws();三、異步非堵塞IO場景 3.1 異步、阻塞和IO模型(務(wù)必理解) 3.1.1 同步和異步
????關(guān)注的是消息通知機制;
同步:調(diào)用發(fā)出之后不會立即返回,但一旦返回,則返回最終結(jié)果;3.1.2 阻塞(block)和非阻塞(nonblock)異步:調(diào)用發(fā)出之后,被調(diào)用方立即返回消息,但返回的并非最終結(jié)果。被調(diào)用者通過狀態(tài)、通知機制等來通知調(diào)用者,或通過回調(diào)函數(shù)來處理結(jié)果;
????關(guān)注的是調(diào)用者等待被調(diào)用者返回調(diào)用結(jié)果時的狀態(tài)。
阻塞:調(diào)用結(jié)果返回之前,調(diào)用者會被掛起,調(diào)用者只有在得到返回結(jié)果之后才能繼續(xù)。3.1.3 IO模型非阻塞:調(diào)用者在結(jié)果返回之前,不會被掛起;
blocking IO:阻塞式IO nonblocking IO:非阻塞IO multiplexing IO:多路復(fù)用IO signal driven IO:事件驅(qū)動式IO asynchronous IO:異步IO
????真正執(zhí)行IO過程的階段是內(nèi)核內(nèi)存數(shù)據(jù)拷貝到進程內(nèi)存中
3.2 Swoole異步毫秒定時器????異步高精度定時器,粒度為毫秒級
//每隔2000ms觸發(fā)一次 swoole_timer_tick(2000, function ($timer_id) { echo "tick-2000ms "; }); //3000ms后執(zhí)行此函數(shù) swoole_timer_after(3000, function () { echo "after 3000ms. "; });3.3 異步文件系統(tǒng)IO
????Swoole官網(wǎng)文檔:異步文件系統(tǒng)IO
3.3.1 異步讀/** * 讀取文件 * __DIR__ * 文件不存在會返回false * 成功打開文件立即返回true * 數(shù)據(jù)讀取完畢后會回調(diào)指定的callback函數(shù)。 */ //函數(shù)風(fēng)格 $result = swoole_async_readfile(__DIR__."/1.txt", function($filename, $fileContent) { echo "filename:".$filename.PHP_EOL; // echo "content:".$fileContent.PHP_EOL; }); //命名空間風(fēng)格 $result = SwooleAsync::readfile(__DIR__."/1.txt", function($filename, $fileContent) { echo "filename:".$filename.PHP_EOL; // echo "content:".$fileContent.PHP_EOL; }); var_dump($result); echo "start".PHP_EOL;3.3.2 異步寫(如日志)
$http->on("request", function($request, $response) { $content = [ "date:" => date("Ymd H:i:s"), "get:" => $request->get, "post:" => $request->post, "header:" => $request->header, ]; swoole_async_writefile(__DIR__."/access.log", json_encode($content).PHP_EOL, function($filename){ // todo }, FILE_APPEND); $response->end("response:". json_encode($request->get)); });3.4 異步MySQL詳解
class AsyncMySql { /** * @var string */ public $dbSource = ""; /** * mysql的配置 * @var array */ public $dbConfig = []; public function __construct() { //new swoole_mysql; $this->dbSource = new SwooleMysql; $this->dbConfig = [ "host" => "127.0.0.1", "port" => 3306, "user" => "root", "password" => "test", "database" => "test", "charset" => "utf8", ]; } public function update() {} public function add() {} /** * mysql 執(zhí)行邏輯 * @param $id * @param $username * @return bool */ public function execute($id, $username) { $this->dbSource->connect($this->dbConfig, function($db, $result) use($id, $username) { echo "mysql-connect".PHP_EOL; if($result === false) { var_dump($db->connect_error); // todo } $sql = "select * from cmf_user where id=1"; //$sql = "update test set `username` = "".$username."" where id=".$id; // insert into // query (add select update delete) $db->query($sql, function($db, $result){ // select => result返回的是 查詢的結(jié)果內(nèi)容 if($result === false) { // todo var_dump($db->error); }elseif($result === true) {// add update delete // todo var_dump($db->affected_rows); }else { print_r($result); } $db->close(); }); }); return true; } } $obj = new AsyncMySql(); $flag = $obj->execute(1, "singwa-111112"); var_dump($flag).PHP_EOL; echo "start".PHP_EOL;
????
3.5 異步Redis 3.5.1 環(huán)境準(zhǔn)備????swoole使用redis的前置條件
redis服務(wù)
hiredis庫
編譯swoole需要加入 -enable-async-redis
????編譯安裝hiredis
使用Redis客戶端,需要安裝hiredis庫,下載hiredis源碼后,執(zhí)行
make -j sudo make install sudo ldconfig
????hiredis下載地址
????啟用異步Redis客戶端
編譯swoole時,在configure指令中加入--enable-async-redis
[root@izwz93ee3z8wdxsujiec2oz swoole]# ./configure --with-php-config=/usr/local/php/bin/php-config --enable-async-redis make clean make -j sudo make install
????查看PHP的swoole擴展:php -m
????查看hiredis是否編譯安裝成功:php --ri swoole
$redisClient = new swoole_redis;// SwooleRedis $redisClient->connect("127.0.0.1", 6379, function(swoole_redis $redisClient, $result) { echo "connect".PHP_EOL; var_dump($result); // 同步 redis (new Redis())->set("key",2); /*$redisClient->set("singwa_1", time(), function(swoole_redis $redisClient, $result) { var_dump($result); });*/ /*$redisClient->get("singwa_1", function(swoole_redis $redisClient, $result) { var_dump($result); $redisClient->close(); });*/ $redisClient->keys("*gw*", function(swoole_redis $redisClient, $result) { var_dump($result); $redisClient->close(); }); }); echo "start".PHP_EOL;下一篇:Swoole入門到實戰(zhàn)(二):進程,內(nèi)存和協(xié)程、Swoole完美支持ThinkPHP5、分發(fā)Task異步任務(wù)機制實現(xiàn)
參考教程:韓天峰力薦 Swoole入門到實戰(zhàn)打造高性能賽事直播平臺
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/28681.html
摘要:對于這一問題企業(yè)一般多會采用其他技術(shù),比如使用或者其他語言。但是有了之后,由于底層全是用語言實現(xiàn),其出色的性能深受追捧。由于以前適用場景的局限性比較大,所以會有說重新定義了的說法。到這里的環(huán)境已經(jīng)搭建完成了。 Swoole 是一個 php 的擴展,它的核心目標(biāo)就是解決 php 在實現(xiàn)一些高訪問 server 服務(wù)中可能遇到的一系列問題,這些問題用原生的 php 往往并不能很方便高效的解...
摘要:修復(fù)添加超過萬個以上定時器時發(fā)生崩潰的問題增加模塊,下高性能序列化庫修復(fù)監(jiān)聽端口設(shè)置無效的問題等。線程來處理網(wǎng)絡(luò)事件輪詢,讀取數(shù)據(jù)。當(dāng)?shù)娜挝帐殖晒α艘院?,由這個線程將連接成功的消息告訴進程,再由進程轉(zhuǎn)交給進程。此時進程觸發(fā)事件。 本文示例代碼詳見:https://github.com/52fhy/swoo...。 簡介 Swoole是一個PHP擴展,提供了PHP語言的異步多線程服務(wù)器...
摘要:目前來看等語言還難以企及和。作為一個資深的開發(fā)者,在技術(shù)上給各位程序十點未來的建議,希望對大家有所幫助。開發(fā)者應(yīng)當(dāng)學(xué)習(xí)掌握規(guī)范,在開發(fā)程序時應(yīng)當(dāng)盡量遵循規(guī)范。程序員除了寫后臺程序之外,還有很大一部分工作在展現(xiàn)層,和瀏覽器前端打交道。 PHP 從誕生到現(xiàn)在已經(jīng)有20多年歷史,從Web時代興起到移動互聯(lián)網(wǎng)退潮,互聯(lián)網(wǎng)領(lǐng)域各種編程語言和技術(shù)層出不窮, Node.js 、 GO 、 Pytho...
閱讀 3121·2023-04-26 01:58
閱讀 962·2021-11-24 09:38
閱讀 3293·2021-09-03 10:29
閱讀 723·2021-08-21 14:10
閱讀 1498·2019-08-30 15:44
閱讀 3096·2019-08-30 14:10
閱讀 3224·2019-08-29 16:32
閱讀 1486·2019-08-29 12:48