如何在 Laravel 中创建一个简单的事件流?

简介

事件流为你提供了一种将事件发送到客户端而无需重新加载页面的方法。当对数据库进行实时更改时,这对于更新用户界面非常有用。

与使用 AJAX 请求的传统长轮询不同,在传统轮询中,多个请求被发送到服务器,每次都建立新的连接,事件流在单个请求中实时发送到客户端。

Laravel

在本文中,我将向你展示如何在 Laravel 中创建一个简单的事件流。

先决条件

在开始之前,你需要在机器上安装 Laravel。

我将在这个演示中使用 DigitalOcean Ubuntu Droplet 。如果你愿意,你可以使用我的会员代码获得免费 $100 DigitalOcean 积分来启动你自己的服务器!

如果你还没有,可以按照本教程中的步骤进行操作:

或者可以使用这个很棒的脚本进行安装:

创建一个 Controller

让我们从创建一个处理事件流的控制器开始。

使用以下命令:

php artisan make:controller EventStreamController

这将在 App\Http\Controllers 目录中创建一个新控制器。

添加事件流方法

一旦我们创建了我们的控制器,我们需要向它添加 stream 方法。 该方法将用于发送事件流。

打开 EventStreamController.php 文件并添加以下代码:

<?php

namespace App\Http\Controllers;

use Carbon\Carbon;
use App\Models\Trade;

class StreamsController extends Controller
{
    /**
     * 事件流代码
     *
     * @return \Illuminate\Http\Response
     */
    public function stream(){
        return response()->stream(function () {
            while (true) {
                echo "event: ping\n";
                $curDate = date(DATE_ISO8601);
                echo 'data: {"time": "' . $curDate . '"}';
                echo "\n\n";

                $trades = Trade::latest()->get();
                echo 'data: {"total_trades":' . $trades->count() . '}' . "\n\n";

                $latestTrades = Trade::with('user', 'stock')->latest()->first();
                if ($latestTrades) {
                    echo 'data: {"latest_trade_user":"' . $latestTrades->user->name . '", "latest_trade_stock":"' . $latestTrades->stock->symbol . '", "latest_trade_volume":"' . $latestTrades->volume . '", "latest_trade_price":"' . $latestTrades->stock->price . '", "latest_trade_type":"' . $latestTrades->type . '"}' . "\n\n";
                }

                ob_flush();
                flush();

                // 如果客户端中止连接,则中断循环(关闭页面)
                if (connection_aborted()) {break;}
                usleep(50000); // 50ms
            }
        }, 200, [
            'Cache-Control' => 'no-cache',
            'Content-Type' => 'text/event-stream',
        ]);
    }
}

这里要注意的主要事项是:

  • 我们使用 response()->stream() 方法来创建事件流。
  • 然后我们有一个无限循环,每隔50ms发送一次事件流。
  • 如果客户端中止连接,我们使用 ob_flush()flush() 来发送事件流。
  • 我们使用 sleep() 发送下一个事件之前等待50ms。
  • 我们使用 connection_aborted() 来中断循环,如果客户端中止了连接。
  • 我们使用 Carbon\Carbon 类获取当前日期。
  • 我们使用 App\Models\Trade 模型获取最新交易。这仅用于演示,你可以使用任何你想要的模型。
  • Content-Type 标头设置为 text/event-stream 以告知浏览器响应是事件流。

启用输出缓冲

为了使上述代码正常工作,我们需要在你的 PHP.ini 文件中启用输出缓冲。 这是通过将以下行添加到 php.ini 文件中完成的:

output_buffering = On

进行此更改后,可能需要重新加载 PHP-FPM 服务。 或者如果你使用的是 Apache,则可以重新启动 Apache。

添加路由

当请求 /stream 路由时,我们想调用 stream 方法。

路由将被添加到routes/web.php 文件中,如下所示:

use App\Http\Controllers\StreamsController;

Route::get('/stream', [StreamsController::class, 'stream']);

使用前端的事件流

你可以使用 Vue.js 之类的前端框架来处理事件流。 但是对于这个演示,我将使用纯 Javascript。

添加到 blade 模板中的 JavaScript 片段如下所示:

const eventSource = new EventSource('/stream');

eventSource.onmessage = function(event) {
    const data = JSON.parse(event.data);
    if (data.time) {
        document.getElementById('time').innerHTML = data.time;
    }
    const newElement = document.createElement("li");
    const eventList = document.getElementById("list");

    newElement.textContent = "message: " + event.data;
    eventList.appendChild(newElement);
}

要查看此操作,你可以尝试以下演示!

演示项目

如果你想了解事件流是如何工作的,可以查看我创建的演示项目:

Laravel EventStream:使用 Laravel 和 Materialize 进行实时股票交易仪表板

演示项目不仅显示事件流,还具有简单的前端仪表板,并使用 Materialize 作为流数据库。

Laravel EventStream

SSE vs WebSockets

事件流很棒且易于使用,但与 WebSockets 等其他流协议相比,它也有一些优点和缺点。

例如,SSE 是单向的,这意味着一旦建立连接,服务器只会向客户端发送数据,而客户端不能将数据发送回服务器。

与长轮询不同,使用 WebSockets,你只有一个与服务器的连接,类似于 SSE(服务器发送事件)。 连接是双工的,这意味着你可以从服务器发送和接收数据。

如果想了解有关 SSE 和 WebSockets 之间差异的更多信息,请观看 Martin Chaov 的精彩视频:链接

结论

有关事件流的更多信息,请在此处查看 Mozilla 的此文档:

在那里,你会找到对事件流及其工作方式的更深入的解释。

有关 Materialise 的更多信息,请在此处观看此视频:链接

希望你喜欢这个教程!

本文中的所有译文仅用于学习和交流目的,转载请务必注明文章译者、出处、和本文链接
我们的翻译工作遵照 CC 协议,如果我们的工作有侵犯到您的权益,请及时联系我们。

原文地址:https://devdojo.com/bobbyiliev/how-to-cr...

译文地址:https://learnku.com/laravel/t/71389

本帖已被设为精华帖!
本文为协同翻译文章,如您发现瑕疵请点击「改进」按钮提交优化建议
《L05 电商实战》
从零开发一个电商项目,功能包括电商后台、商品 & SKU 管理、购物车、订单管理、支付宝支付、微信支付、订单退款流程、优惠券等
《G01 Go 实战入门》
从零开始带你一步步开发一个 Go 博客项目,让你在最短的时间内学会使用 Go 进行编码。项目结构很大程度上参考了 Laravel。
讨论数量: 18

孤陋寡闻了,第一次听说,果断了解下,,,

1年前 评论

laravel的广播可以用这个代替websocket吗,这样就不用去维护一个push服务了

1年前 评论
小李世界 1年前

弱弱的问一句 这样会不会fpm 进程一直在,连接的人多了,直接就挂了呀

1年前 评论
哪吒的狗腿子 1年前

本地测试了一下,header 应该设置成:

[
    'Cache-Control' => 'no-cache', 
    'Content-Type' => 'text/event-stream',
    'X-Accel-Buffering' => 'no' , // 关闭缓存
]

我用了sleep(1)而不是文中的usleep,本地开了6个页面之后就明显出现了阻塞,也可能是我姿势不对,不知道有没有大佬给一个实际演示。

1年前 评论

chatGpt后台界面好像用的就是这个

1年前 评论

如果php-fpm前面用了nginx,需要把nginx 的 fastcgi 模块的缓存给关掉。不然得等缓存满了才会将累积的响应一下输出。

11个月前 评论

为什么我的不行,会重新发送?

11个月前 评论

while true。 一个页面占用一个fpm进程,假设一台4c8g的机器开100个进程,一台机器最多能打开100个这样的链接。

结论:对于fpm模式的php项目是鸡肋,聊胜于无。go/swoole的还可以一战

11个月前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!