Laravel如何优雅的使用Swoole

Laravel如何优雅的使用Swoole

背景

正在做一个智能家居的项目(钱低的吓死人怎么办),接收下位机(就是控制智能家居硬件模块的HUB)协议解析,Web端维护硬件状态,利用APP交互。由于下位机数据是发送到服务器的XXX端口,所以必须对XXX端口进行监听。其实和聊天室的概念差不多,研究了一下workerman、swoole和其他几个开源的项目,决定采用swoole。

关于php解析下位机的16进制协议,其实相当之扯蛋,要是你最好还是用.NET或者JAVA吧。很久没碰MVC了,光是为解析协议写webservice觉得钱又太TM低了,哈哈哈,所以直接上PHP吧。网上搜搜还没见几个php这样搞的项目,我还没做完,做完了来谈谈,关键函数主要是bin2hex/pack/unpack。这一篇主要聊聊Laravel如何优雅的使用Swoole,其实只需简单3步就可以完成。

什么是Swoole

直接套用Swoole官网的介绍:PHP的异步、并行、高性能网络通信引擎,使用纯C语言编写,提供了PHP语言的异步多线程服务器,异步TCP/UDP网络客户端,异步MySQL,异步Redis,数据库连接池,AsyncTask,消息队列,毫秒定时器,异步文件读写,异步DNS查询。 Swoole内置了Http/WebSocket服务器端/客户端、Http2.0服务器端。

Swoole官网的文档不够丰富啊,这比较头疼,但大部分的问题都解释了。如果你对Swoole很感兴趣,那么看看这个Swoole入门教程。Swoole提供了多线程、长连接等很多牛逼的功能,把php上升到了一个新的台阶,具体的你可以看看入门教程,本文只限于讨论Laravel和Swoole的结合。

Swoole为了提供服务,必须以CLI模式运行,什么是CLI模式呢?如果你Swoole业务代码是写在一个叫server.php的文件中,那么在命令行下输入php server.php开启。这是比较头疼的事情,因为Laravel框架可不是这样的运转的,那如何能与Laravel结合呢?没错,自定义一条Artisan Command,就这么简单。

STEP 1-自定义Command

关于自定义Artisan Commnad,你需要了解的技术点都在这里,我自定义了一个叫做SwooleCommand的命令,直接贴关键代码:

fire是入口

在命令行(CLI)下执行php artisan swoole start即可开启Swoole服务。分析一下代码,你可以看到命令参数包括启动、重启、关闭,我图省事只实现了启动部分,如果需要关闭,在linux中利用kill命令关闭进程,步骤挺简单的:

1.执行 ps -aux|grep artisan命令,获取pid(有多个进程,杀第一个即可)
2.执行 kill pid命令,pid是第一步你获取的

关于Swoole的配置不是本文讨论的范围,请移步官网,这里把Swoole服务用$serv变量进行了保存,是为了后面Laravel发送命令交互。你可以看到,Swoole的事件响应代码是这样的:

用Handler处理事件响应

如果说fire打开了Swoole的大门,那么这里的handler就是Swoole与Laravel的传送带,利用自己写的handler,就可以把各种业务逻辑写进Laravel框架中,然后就可以使用Laravel提供的各种高效方便的功能了。“handler”是一种命名习惯,你也可以叫做"callback"、"manager"、"listener",这看你的命名习惯了。我没有采用new的方式而是用Laravel的IoC注入App::make,主要是图省事(因为handler的构造器用到了我自定义的数据处理类,往下看)。

STEP 2-自定义handler

因为是自定义的类,请遵循命名空间,并在composer.json中声明,完了执行composer dump-autoload命令更新一遍。比如我创建了一个文件夹app\handlers存放handler,那么在composer.json中看起来是这样的:

autoload不能少

那么handler里面具体干些啥,就由你来决定了。反正和写controller差不多,各种Laravel框架的功能你都能随便用,贴上我的:

上一节我提到我用IoC是因为构造器里面用到了自己的数据处理类,我把增删改查和其他数据处理的业务放到Repository中了,没其他原因,只是这样代码看起来清爽一点。如此,利用Swoole接收数据的流程就算搞定了,那么要想利用Swoole向客户端发送数据该怎么做呢?咳咳,这个稍微麻烦点,需要曲线方法实现,继续看下一节。

STEP 3-发送数据

有两种方法,但都离不开一个缓存kv结构(Laravel自带的Cache功能就够了),保存客户端的地址数据,要不你怎么知道发到哪里去。我用的是第一种,图省事,发送数据和Swoole就无关了,如果你需要长连接websocket,这种不适用,老老实实用第二种吧。如果你有更好的办法,请一定要告诉我!

第一种:fsockopen

挺简单的,和swoole就没关系了,利用Swoole的connection_info函数获取客户端的IP地址和端口,然后用fsockopen直接发送数据。

第二种:内部端口监听

Swoole支持监听多个端口,实现的思想就是利用fsockopen把数据利用内部监听的端口发送过去,然后就可以调用$serv发送消息了。这么做的好处就是不需要知道客户端的实际IP地址和端口,在Cache保存客户端的$fd标识,直接就发数据。采用这个思路,请记得iptables把端口打开。我自己并没有采用,因为不是长连接我觉得太麻烦。

总结

Swoole非常棒,其实都没怎么用上(项目钱给够再说吧)。你还可以参考官网的配置,将Swoole作为nginx承载代理,据说性能提升很大。qq群533838427

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用Laravel8和Swoole实现直播,你需要遵循以下步骤: 1. 首先,你需要在Laravel项目中安装Swoole扩展。可以使用Composer运行以下命令来安装它: ``` composer require swooletw/laravel-swoole ``` 2. 在项目的`config/app.php`文件中添加以下行,将Swoole提供程序添加到Laravel中: ``` 'providers' => [ ... SwooleTW\Http\LaravelServiceProvider::class, ... ], 'aliases' => [ ... 'Swoole' => SwooleTW\Http\Facades\Swoole::class, ... ], ``` 3. 创建Swoole服务器的配置文件。可以在项目的根目录下创建`swoole.php`文件,并添加以下内容: ``` return [ 'host' => env('SWOOLE_HOST', '127.0.0.1'), 'port' => env('SWOOLE_PORT', '9501'), 'options' => [ 'worker_num' => 8, 'daemonize' => env('SWOOLE_DAEMONIZE', false), 'task_worker_num' => 8, ], ]; ``` 4. 在`.env`文件中添加以下行,以使用上述配置: ``` SWOOLE_HOST=127.0.0.1 SWOOLE_PORT=9501 SWOOLE_DAEMONIZE=false ``` 5. 创建一个Swoole服务器的启动脚本。在项目的`app/Console/Commands`目录下创建一个`SwooleServer.php`脚本,并添加以下内容: ``` <?php namespace App\Console\Commands; use Illuminate\Console\Command; use SwooleTW\Http\Server\Facades\Server; class SwooleServer extends Command { protected $signature = 'swoole:server {action}'; protected $description = 'Start or stop swoole http server.'; public function handle() { $action = $this->argument('action'); if ($action === 'start') { Server::start(); } elseif ($action === 'stop') { Server::stop(); } else { $this->error('Invalid argument.'); } } } ``` 6. 创建控制器来处理直播流。在项目中创建一个控制器,例如`LiveController.php`,并添加以下内容: ``` <?php namespace App\Http\Controllers; use Illuminate\Http\Request; use SwooleTW\Http\Websocket\Facades\Websocket; class LiveController extends Controller { public function index(Request $request) { return view('live.index'); } public function handle(Request $request) { // 处理直播流 Websocket::broadcast()->emit('live', $request->getContent()); } } ``` 7. 创建一个WebSocket事件监听器。在项目的`app/Providers`目录下创建一个`WebsocketServiceProvider.php`文件,并添加以下内容: ``` <?php namespace App\Providers; use Illuminate\Support\ServiceProvider; use SwooleTW\Http\Websocket\Server; use App\Events\LiveStream; class WebsocketServiceProvider extends ServiceProvider { public function boot(Server $server) { $server->on('message', function ($server, $frame) { $payload = json_decode($frame->data, true); if ($payload['event'] === 'live') { event(new LiveStream($payload['data'])); } }); } } ``` 8. 创建一个事件类来处理直播流。在项目中创建一个事件类,例如`LiveStream.php`,并添加以下内容: ``` <?php namespace App\Events; use Illuminate\Broadcasting\InteractsWithSockets; use Illuminate\Foundation\Events\Dispatchable; use Illuminate\Queue\SerializesModels; class LiveStream { use Dispatchable, InteractsWithSockets, SerializesModels; public $stream; public function __construct($stream) { $this->stream = $stream; } public function broadcastOn() { return ['live']; } } ``` 9. 创建一个事件监听器来推送直播流。在项目的`app/Listeners`目录下创建一个`LiveStreamListener.php`文件,并添加以下内容: ``` <?php namespace App\Listeners; use App\Events\LiveStream; use SwooleTW\Http\Websocket\Facades\Websocket; class LiveStreamListener { public function handle(LiveStream $event) { Websocket::broadcast()->emit('live', $event->stream); } } ``` 10. 在`EventServiceProvider.php`中注册事件和监听器。在项目的`app/Providers`目录下找到`EventServiceProvider.php`文件,并将以下行添加到`$listen`数组中: ``` 'App\Events\LiveStream' => [ 'App\Listeners\LiveStreamListener', ], ``` 11. 创建一个前端页面来显示直播流。在项目的`resources/views`目录下创建一个`live/index.blade.php`文件,并添加以下内容: ``` <!DOCTYPE html> <html> <head> <title>Live Stream</title> <meta name="csrf-token" content="{{ csrf_token() }}"> </head> <body> <div id="live"></div> <script src="{{ asset('js/app.js') }}"></script> <script> const ws = new WebSocket('ws://localhost:9501'); ws.onmessage = function (event) { const data = JSON.parse(event.data); if (data.event === 'live') { document.getElementById('live').innerHTML = data.data; } }; </script> </body> </html> ``` 现在你已经完成了Laravel8和Swoole的配置,可以使用以上步骤来实现直播功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值