0

0

Swoole中如何实现平滑地给所有客户端发消息

下次还敢

下次还敢

发布时间:2025-09-26 16:46:01

|

403人浏览过

|

来源于php中文网

原创

使用协程分片广播并校验连接状态,结合心跳机制与消息队列解耦,可实现Swoole平滑推送。示例中每10秒将客户端分批(50个/组),通过go协程并发调用$server->push发送,避免阻塞;利用$server->isEstablished($fd)跳过无效连接,防止异常;推荐用Redis或Swoole\Table管理在线状态,Worker仅转发消息,提升扩展性;需定义onClose回调清理资源,万级连接时可引入分布式架构优化性能。核心为:分批协程+状态校验+心跳+解耦。

swoole中如何实现平滑地给所有客户端发消息

在 Swoole 中实现平滑地给所有客户端发送消息,关键在于避免阻塞主进程、处理连接状态异常,并保证广播效率。直接遍历所有连接发消息容易因个别客户端卡顿导致整体延迟,因此需要结合协程、连接验证和异步机制来优化。

使用协程分片广播,避免单次阻塞

Swoole 的 foreach + coroutine 可以并发发送消息,减少总耗时。通过 red">Swoole\Coroutine\WaitGroup 或并发协程控制,将客户端连接分批处理,防止一次性创建过多协程影响性能。

示例代码:

use Swoole\Coroutine;
use Swoole\Http\Server;

$server = new Server("0.0.0.0", 9501);
$server->set(['worker_num' => 1, 'enable_coroutine' => true]);

$server->on('workerStart', function () use ($server) {
    go(function () use ($server) {
        while (true) {
            // 模拟定时广播
            co::sleep(10);
            $clients = $server->connections;
            $chunks = array_chunk($clients, 50); // 每批50个

            foreach ($chunks as $chunk) {
                go(function () use ($chunk, $server) {
                    foreach ($chunk as $fd) {
                        // 确保连接有效且为 WebSocket 客户端
                        if ($server->isEstablished($fd)) {
                            $server->push($fd, "来自服务器的广播消息");
                        }
                    }
                });
            }
        }
    });
});

检查连接状态,跳过无效客户端

发送前必须调用 $server->isEstablished($fd) 判断连接是否正常。对于已断开但未及时清理的 $fd,跳过可避免 write error 或异常中断。

也可以结合心跳机制,在客户端定期 ping,服务端标记活跃状态,只向活跃连接推送。

利用中间件解耦消息分发逻辑

实际项目中,广播通常由业务触发(如订单通知)。建议将消息推送交给独立的“推送服务”或“消息队列”,Worker 进程只负责转发,不直接操作连接。

Heeyo
Heeyo

Heeyo:AI儿童启蒙陪伴师,风靡于硅谷的儿童AI导师和玩伴

下载

例如:使用 Redis Pub/Sub 或 Swoole\Table 记录在线用户,另一个协程监听频道并执行广播,实现解耦与扩展性。

注意内存与连接管理

长时间运行下,$server->connections 可能包含已关闭的连接(仅在 onClose 回调后移除)。确保定义 onClose 回调,及时清理资源。

若客户端量大(万级),考虑引入分布式方案或多进程共享 Swoole\Table 存储 fd 与用户映射,提升查找效率。

基本上就这些。核心是:分批协程发送 + 连接状态校验 + 心跳维护 + 架构解耦,才能做到真正平滑广播。

相关专题

更多
swoole为什么能常驻内存
swoole为什么能常驻内存

swoole常驻内存的特性:1. 事件驱动模型减少内存消耗;2. 协程并行执行任务占用更少内存;3. 协程池预分配协程消除创建开销;4. 静态变量保留状态减少内存分配;5. 共享内存跨协程共享数据降低内存开销。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

291

2024.04.10

什么是分布式
什么是分布式

分布式是一种计算和数据处理的方式,将计算任务或数据分散到多个计算机或节点中进行处理。本专题为大家提供分布式相关的文章、下载、课程内容,供大家免费下载体验。

325

2023.08.11

分布式和微服务的区别
分布式和微服务的区别

分布式和微服务的区别在定义和概念、设计思想、粒度和复杂性、服务边界和自治性、技术栈和部署方式等。本专题为大家提供分布式和微服务相关的文章、下载、课程内容,供大家免费下载体验。

233

2023.10.07

什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

178

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

213

2025.12.18

php中foreach用法
php中foreach用法

本专题整合了php中foreach用法的相关介绍,阅读专题下面的文章了解更多详细教程。

42

2025.12.04

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

187

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

288

2023.10.25

PS使用蒙版相关教程
PS使用蒙版相关教程

本专题整合了ps使用蒙版相关教程,阅读专题下面的文章了解更多详细内容。

23

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
swoole进程树解析
swoole进程树解析

共4课时 | 0.2万人学习

Swoole系列-从0到1-新手进阶
Swoole系列-从0到1-新手进阶

共29课时 | 1.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号