EasySwoole 基礎入門

使用 Composer 安裝

composer require easyswoole/easyswoole=3.x
php vendor/bin/easyswoole install

啓動框架

php easyswoole start

nginx轉發

 

server {
 root /data/wwwroot/;
 server_name local.easyswoole.com;
  
 location / {
 proxy_http_version 1.1;
 proxy_set_header Connection "keep-alive";
 proxy_set_header X-Real-IP $remote_addr;
 if (!-e $request_filename) {
 proxy_pass http://127.0.0.1:9501;
 }
 if (!-f $request_filename) {
 proxy_pass http://127.0.0.1:9501;
 }
 }
}

proxy_set_header X-Real-IP $remote_addr; 獲取真實IP地址

運行你的hellword

project              項目部署目錄
----------------------------------
├─App        應用目錄
│  └─HttpController      應用的控制器目錄
│     └─Index.php    默認控制器文件
----------------------------------

Index.php

<?php
namespace App\HttpController;
use EasySwoole\Http\AbstractInterface\Controller;
class Index extends Controller
{
 function index()
 {
 // TODO: Implement index() method.
 $this->response()->write('hello world');
 }
}

編輯根目錄下的 composer.json 文件,註冊應用的命名空間

{
 "autoload": {
 "psr-4": {
 "App\\": "App/"
 }
 },
 "require": {
 "easyswoole/easyswoole": "3.x-dev"
 }
}

意思就是設置自動加載

最後執行composer dumpautoload 命令更新命名空間,可以開始編寫業務邏輯

# 更新命名空間映射
composer dumpautoload
# 啓動框架
php easyswoole start
目錄結構
project                   項目部署目錄
├─App                     應用目錄(可以有多個)
│  ├─HttpController       控制器目錄
│  │  └─Index.php         默認控制器
│  └─Model                模型文件目錄
├─Log                     日誌文件目錄
├─Temp                    臨時文件目錄
├─vendor                  第三方類庫目錄
├─composer.json           Composer架構
├─composer.lock           Composer鎖定
├─EasySwooleEvent.php     框架全局事件
├─easyswoole              框架管理腳本
├─easyswoole.install      框架安裝鎖定文件
├─dev.php                 開發配置文件
├─produce.php             生產配置文件

生命週期,也就是流程

我的官方羣點擊此處

 

配置文件說明

<?php
 /**
 * Created by PhpStorm.
 * User: yf
 * Date: 2019-01-01
 * Time: 20:06
 */
 return [
 'SERVER_NAME' => "EasySwoole",//服務名
 'MAIN_SERVER' => [
 'LISTEN_ADDRESS' => '0.0.0.0',//監聽地址
 'PORT' => 9501,//監聽端口
 'SERVER_TYPE' => EASYSWOOLE_WEB_SERVER, //可選爲 EASYSWOOLE_SERVER  EASYSWOOLE_WEB_SERVER EASYSWOOLE_WEB_SOCKET_SERVER
 'SOCK_TYPE' => SWOOLE_TCP,//該配置項當爲SERVER_TYPE值爲TYPE_SERVER時有效
 'RUN_MODEL' => SWOOLE_PROCESS,// 默認Server的運行模式
 'SETTING' => [// Swoole Server的運行配置( 完整配置可見[Swoole文檔](https://wiki.swoole.com/wiki/page/274.html) )
 'worker_num' => 8,//運行的  worker進程數量
 'max_request' => 5000,// worker 完成該數量的請求後將退出,防止內存溢出
 'task_worker_num' => 8,//運行的 task_worker 進程數量
 'task_max_request' => 1000,// task_worker 完成該數量的請求後將退出,防止內存溢出
 'reload_async' => true,//設置異步重啓開關。設置爲true時,將啓用異步安全重啓特性,Worker進程會等待異步事件完成後再退出。
 'task_enable_coroutine' => true//開啓後自動在onTask回調中創建協程
 ]
 ],
 'TEMP_DIR' => null,//臨時文件存放的目錄
 'LOG_DIR' => null,//日誌文件存放的目錄
 'CONSOLE' => [//console控制檯組件配置
 'ENABLE' => true,//是否開啓
 'LISTEN_ADDRESS' => '127.0.0.1',//監聽地址
 'PORT' => 9500,//監聽端口
 'USER' => 'root',//驗權用戶名
 'PASSWORD' => '123456'//驗權用戶名
 ],
 'FAST_CACHE' => [//fastCache組件
 'PROCESS_NUM' => 0,//進程數,大於0纔開啓
 'BACKLOG' => 256,//數據隊列緩衝區大小
 ],
 'DISPLAY_ERROR' => true,//是否開啓錯誤顯示
 ];

配置操作類

EasySwoole\Config 類

toArray 方法獲取全部配置,load 方法重載全部配置

如果設置了修改,需要更新配置的意思

<?php
$instance = \EasySwoole\EasySwoole\Config::getInstance();
// 獲取配置 按層級用點號分隔
$instance->getConf('MAIN_SERVER.SETTING.task_worker_num');
// 設置配置 按層級用點號分隔
$instance->setConf('DATABASE.host', 'localhost');
// 獲取全部配置
$conf = $instance->getConf();
// 用一個數組覆蓋當前配置項
$conf['DATABASE'] = [
 'host' => '127.0.0.1',
 'port' => 13306
];
$instance->load($conf);

添加用戶配置項

'MYSQL' => [
 'host' => '192.168.75.1',
 'port' => '3306',
 'user' => 'root',
 'timeout' => '5',
 'charset' => 'utf8mb4',
 'password' => 'root',
 'database' => 'cry',
 'POOL_MAX_NUM' => '20',
 'POOL_TIME_OUT' => '0.1',
],
/*################ REDIS CONFIG ##################*/
'REDIS' => [
 'host' => '127.0.0.1',
 'port' => '6379',
 'auth' => '',
 'POOL_MAX_NUM' => '20',
 'POOL_MIN_NUM' => '5',
 'POOL_TIME_OUT' => '0.1',
],

生產與開發配置分離

默認爲開發模式,加載 dev.php

生成

php easyswoole start produce

DI注入配置

也就是依賴注入

<?php
Di::getInstance()->set(SysConst::ERROR_HANDLER,function (){});//配置錯誤處理回調
Di::getInstance()->set(SysConst::SHUTDOWN_FUNCTION,function (){});//配置腳本結束回調
Di::getInstance()->set(SysConst::HTTP_CONTROLLER_NAMESPACE,'App\\HttpController\\');//配置控制器命名空間
Di::getInstance()->set(SysConst::HTTP_CONTROLLER_MAX_DEPTH,5);//配置http控制器最大解析層級
Di::getInstance()->set(SysConst::HTTP_EXCEPTION_HANDLER,function (){});//配置http控制器異常回調
Di::getInstance()->set(SysConst::HTTP_CONTROLLER_POOL_MAX_NUM,15);//http控制器對象池最大數量

動態配置

每次開始了,是上一次的進程,比如你打開了舊版,現在更新了新版,但是舊版還是開着,沒有重啓動,也就是一直舊版,現在有個動態配置,表示可以平滑的修改

<?php
 Config::getInstance()->setDynamicConf('test_config_value', 0);//配置一個動態配置項
 $test_config_value_1 = Config::getInstance()->getDynamicConf('test_config_value');//獲取一個配置
 Config::getInstance()->delDynamicConf('test_config_value');//刪除一個配置

服務管理腳本

php easyswoole
 install       安裝easySwoole
 start         啓動easySwoole
 stop          停止easySwoole(守護模式下使用)
 reload        重啓easySwoole(守護模式下使用)
 help          查看命令的幫助信息
easyswoole help -start

守護模式啓動

php easyswoole start d 

線上

php easyswoole start produce

停止

php easyswoole stop 

重啓服務

php easyswoole reload 只重啓task進程
php easyswoole reload all  重啓task + worker進程

文件熱加載

由於 swoole 常駐內存的特性,修改文件後需要重啓worker進程才能將被修改的文件重新載入內存中

解決:Process的方式實現文件變動自動進行服務重載

新建文件 App/Process/HotReload.php 並添加如下內容,也可以放在其他位置,請對應命名空間

<?php
/**
 * Created by PhpStorm.
 * User: evalor
 * Date: 2018-11-26
 * Time: 23:18
 */
namespace App\Process;
use EasySwoole\Component\Process\AbstractProcess;
use EasySwoole\EasySwoole\ServerManager;
use EasySwoole\Utility\File;
use Swoole\Process;
use Swoole\Table;
use Swoole\Timer;
/**
 * 暴力熱重載
 * Class HotReload
 * @package App\Process
 */
class HotReload extends AbstractProcess
{
 /** @var \swoole_table $table */
 protected $table;
 protected $isReady = false;
 protected $monitorDir; // 需要監控的目錄
 protected $monitorExt; // 需要監控的後綴
 /**
 * 啓動定時器進行循環掃描
 */
 public function run($arg)
 {
 // 此處指定需要監視的目錄 建議只監視App目錄下的文件變更
 $this->monitorDir = !empty($arg['monitorDir']) ? $arg['monitorDir'] : EASYSWOOLE_ROOT . '/App';
 // 指定需要監控的擴展名 不屬於指定類型的的文件 無視變更 不重啓
 $this->monitorExt = !empty($arg['monitorExt']) && is_array($arg['monitorExt']) ? $arg['monitorExt'] : ['php'];
 if (extension_loaded('inotify') && empty($arg['disableInotify'])) {
 // 擴展可用 優先使用擴展進行處理
 $this->registerInotifyEvent();
 echo "server hot reload start : use inotify\n";
 } else {
 // 擴展不可用時 進行暴力掃描
 $this->table = new Table(512);
 $this->table->column('mtime', Table::TYPE_INT, 4);
 $this->table->create();
 $this->runComparison();
 Timer::tick(1000, function () {
 $this->runComparison();
 });
 echo "server hot reload start : use timer tick comparison\n";
 }
 }
 /**
 * 掃描文件變更
 */
 private function runComparison()
 {
 $startTime = microtime(true);
 $doReload = false;
 $dirIterator = new \RecursiveDirectoryIterator($this->monitorDir);
 $iterator = new \RecursiveIteratorIterator($dirIterator);
 $inodeList = array();
 // 迭代目錄全部文件進行檢查
 foreach ($iterator as $file) {
 /** @var \SplFileInfo $file */
 $ext = $file->getExtension();
 if (!in_array($ext, $this->monitorExt)) {
 continue; // 只檢查指定類型
 } else {
 // 由於修改文件名稱 並不需要重新載入 可以基於inode進行監控
 $inode = $file->getInode();
 $mtime = $file->getMTime();
 array_push($inodeList, $inode);
 if (!$this->table->exist($inode)) {
 // 新建文件或修改文件 變更了inode
 $this->table->set($inode, ['mtime' => $mtime]);
 $doReload = true;
 } else {
 // 修改文件 但未發生inode變更
 $oldTime = $this->table->get($inode)['mtime'];
 if ($oldTime != $mtime) {
 $this->table->set($inode, ['mtime' => $mtime]);
 $doReload = true;
 }
 }
 }
 }
 foreach ($this->table as $inode => $value) {
 // 迭代table尋找需要刪除的inode
 if (!in_array(intval($inode), $inodeList)) {
 $this->table->del($inode);
 $doReload = true;
 }
 }
 if ($doReload) {
 $count = $this->table->count();
 $time = date('Y-m-d H:i:s');
 $usage = round(microtime(true) - $startTime, 3);
 if (!$this->isReady == false) {
 // 監測到需要進行熱重啓
 echo "severReload at {$time} use : {$usage} s total: {$count} files\n";
 ServerManager::getInstance()->getSwooleServer()->reload();
 } else {
 // 首次掃描不需要進行重啓操作
 echo "hot reload ready at {$time} use : {$usage} s total: {$count} files\n";
 $this->isReady = true;
 }
 }
 }
 /**
 * 註冊Inotify監聽事件
 */
 private function registerInotifyEvent()
 {
 // 因爲進程獨立 且當前是自定義進程 全局變量只有該進程使用
 // 在確定不會造成污染的情況下 也可以合理使用全局變量
 global $lastReloadTime;
 global $inotifyResource;
 $lastReloadTime = 0;
 $files = File::scanDirectory(EASYSWOOLE_ROOT . '/App');
 $files = array_merge($files['files'], $files['dirs']);
 $inotifyResource = inotify_init();
 // 爲當前所有的目錄和文件添加事件監聽
 foreach ($files as $item) {
 inotify_add_watch($inotifyResource, $item, IN_CREATE | IN_DELETE | IN_MODIFY);
 }
 // 加入事件循環
 swoole_event_add($inotifyResource, function () {
 global $lastReloadTime;
 global $inotifyResource;
 $events = inotify_read($inotifyResource);
 if ($lastReloadTime < time() && !empty($events)) { // 限制1s內不能進行重複reload
 $lastReloadTime = time();
 ServerManager::getInstance()->getSwooleServer()->reload();
 }
 });
 }
 public function onShutDown()
 {
 // TODO: Implement onShutDown() method.
 }
 public function onReceive(string $str)
 {
 // TODO: Implement onReceive() method.
 }
}

添加好後在全局的 EasySwooleEvent.php 中,註冊該自定義進程

public static function mainServerCreate(EventRegister $register)
{
 $swooleServer = ServerManager::getInstance()->getSwooleServer();
 $swooleServer->addProcess((new HotReload('HotReload', ['disableInotify' => false]))->getProcess());
} 

更多學習內容請訪問:

騰訊T3-T4標準精品PHP架構師教程目錄大全,只要你看完保證薪資上升一個臺階(持續更新)圖標

以上內容希望幫助到大家,很多PHPer在進階的時候總會遇到一些問題和瓶頸,業務代碼寫多了沒有方向感,不知道該從那裏入手去提升,對此我整理了一些資料,包括但不限於:分佈式架構、高可擴展、高性能、高併發、服務器性能調優、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql優化、shell腳本、Docker、微服務、Nginx等多個知識點高級進階乾貨需要的可以免費分享給大家,需要的可以加入我的官方羣點擊此處

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章