0

0

Workerman支持哪些数据库?Workerman数据库连接方式?

星降

星降

发布时间:2025-09-06 14:46:10

|

721人浏览过

|

来源于php中文网

原创

workerman不支持全局数据库连接,因其常驻内存特性易导致连接超时、资源泄露和并发问题;正确做法是在onworkerstart中为每个进程建立独立连接或使用连接池,并通过心跳机制与异常重连保障连接可用性。

workerman支持哪些数据库?workerman数据库连接方式?

Workerman本身并不直接“支持”特定类型的数据库,因为它是一个基于PHP的异步事件驱动框架,其核心功能是管理PHP进程和处理网络请求。换句话说,Workerman可以与任何PHP语言能够连接和操作的数据库进行交互,这包括但不限于MySQL、PostgreSQL、MongoDB、Redis等。关键在于Workerman的运行机制——常驻内存,这要求我们对数据库连接的管理方式与传统PHP-FPM模式有所不同。通常,这意味着在每个Workerman进程启动时建立独立的数据库连接,或者更推荐的做法是利用连接池来高效管理这些连接。

解决方案

Workerman环境下处理数据库连接,最核心的理念是理解其长驻内存的特性与传统Web服务器(如Nginx + PHP-FPM)的短生命周期模式之间的差异。在PHP-FPM模式下,每个请求完成后,所有资源(包括数据库连接)都会被释放,下次请求会重新初始化。但在Workerman中,进程一旦启动便会持续运行,如果我们在全局范围初始化一个数据库连接,这个连接会一直存在。

这种全局或静态的单一连接方式,在Workerman中会带来一系列问题:

  1. 连接超时与失效: 数据库服务器通常会设置连接超时时间。如果一个Workerman进程长时间没有数据库操作,数据库服务器可能会主动关闭这个空闲连接。但Workerman进程端对此毫不知情,下次尝试使用这个“死连接”时,就会抛出类似“MySQL server has gone away”的错误。
  2. 资源泄露风险: 如果不妥善管理连接的生命周期,可能导致连接句柄泄露,最终耗尽数据库服务器的连接资源。
  3. 多进程并发问题: 在Workerman的多进程架构中,如果多个进程共享同一个数据库连接,可能会引发意想不到的并发问题和数据混乱。每个进程都应该拥有自己的独立连接。

因此,正确的数据库连接策略是:在每个Worker进程启动时(即

onWorkerStart
回调函数中)为该进程建立独立的数据库连接。 这样确保了每个进程都有一个健康的、独立的连接。对于高并发或需要更高效率的场景,使用数据库连接池则是更优的选择,它能有效复用连接,减少连接建立和关闭的开销。

在Workerman中,为什么不能直接使用全局数据库连接?

这个问题,我个人觉得是很多从传统PHP-FPM背景转到Workerman的开发者最容易踩的坑。我们习惯了PHP脚本执行完就“万事大吉”,所有状态都清空。但在Workerman这种长驻内存的服务里,事情就没那么简单了。

原因其实很简单,核心在于生命周期管理。PHP-FPM模式下,一个请求进来,PHP脚本执行,连接数据库,处理完业务逻辑,脚本结束,数据库连接也随之关闭。整个过程是短暂而独立的。然而,Workerman进程一旦启动,就会持续运行,监听端口,处理无数个请求。如果你在脚本的顶层或者某个静态变量里初始化了一个数据库连接,这个连接就会伴随进程的整个生命周期。

这就导致了几个关键问题:

  • 连接“腐烂”: 数据库服务器为了节省资源,会对长时间不活动的连接进行清理。比如MySQL的
    wait_timeout
    参数,默认可能只有8小时。Workerman进程如果在这段时间内没有与数据库交互,其持有的连接就会被数据库服务器单方面关闭。当Workerman进程再次尝试使用这个连接时,就会收到
    MySQL server has gone away
    这样的错误。这就像你和朋友通电话,你以为对方还在听,但其实他已经挂断了。
  • 资源浪费与冲突: 即使没有超时,一个全局连接也可能被多个请求或任务在不恰当的时机并发访问,导致数据不一致或锁等待。更糟糕的是,如果连接在某个地方意外断开,整个进程都会受到影响。
  • 扩展性差: 全局连接难以适应动态的数据库集群变化,比如主从切换、读写分离等。

所以,我的建议是,永远不要在Workerman中直接依赖一个全局的、静态的数据库连接。这几乎是所有长驻内存应用都会面临的问题,不光是Workerman。理解这一点,能帮你少走很多弯路。

Difeye-敏捷的轻量级PHP框架
Difeye-敏捷的轻量级PHP框架

Difeye是一款超轻量级PHP框架,主要特点有: Difeye是一款超轻量级PHP框架,主要特点有: ◆数据库连接做自动主从读写分离配置,适合单机和分布式站点部署; ◆支持Smarty模板机制,可灵活配置第三方缓存组件; ◆完全分离页面和动作,仿C#页面加载自动执行Page_Load入口函数; ◆支持mysql,mongodb等第三方数据库模块,支持读写分离,分布式部署; ◆增加后台管理开发示例

下载

Workerman推荐的数据库连接池方案有哪些?

在Workerman这样的长驻内存应用中,数据库连接池几乎是一个标准配置,它能极大地提升数据库操作的效率和稳定性。它主要解决了频繁建立/关闭连接的开销,并能更好地管理连接的生命周期。

我通常会推荐以下几种方案:

  1. 基于

    onWorkerStart
    的单连接模式(简易版连接池): 这其实是最基础、最直接的实现方式,对于并发量不是特别高,或者每个Worker进程独立处理请求的场景非常适用。每个Worker进程在启动时,都会建立一个独立的数据库连接,并将其存储在进程的全局变量中。

    use Workerman\Worker;
    use PDO;
    
    $worker = new Worker('websocket://0.0.0.0:2346');
    $worker->count = 4; // 启动4个Worker进程
    
    // 在每个Worker进程启动时,建立独立的数据库连接
    $worker->onWorkerStart = function($worker) {
        global $db; // 使用全局变量存储连接
        try {
            $db = new PDO('mysql:host=localhost;dbname=test;charset=utf8', 'user', 'password', [
                PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
                PDO::ATTR_PERSISTENT => false, // 明确设置为非持久化连接
                PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8"
            ]);
            // 可以在这里设置一些连接属性,比如长连接的心跳检测间隔
            // $db->setAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY, false);
        } catch (PDOException $e) {
            echo "Worker {$worker->id} 数据库连接失败: " . $e->getMessage() . "\n";
            // 连接失败通常是致命错误,让进程退出,Workerman会自动拉起新的进程
            exit(250);
        }
    };
    
    $worker->onMessage = function($connection, $data) {
        global $db;
        // 在这里直接使用$db进行数据库操作
        try {
            $stmt = $db->prepare("SELECT * FROM users WHERE id = ?");
            $stmt->execute([1]);
            $user = $stmt->fetch(PDO::FETCH_ASSOC);
            $connection->send(json_encode($user));
        } catch (PDOException $e) {
            // 数据库操作失败,可能是连接断开,需要进一步处理
            $connection->send("数据库操作失败: " . $e->getMessage());
            // 考虑在此处尝试重连或记录日志
        }
    };
    
    Worker::runAll();

    这种方式虽然简单,但每个进程只持有一个连接,严格来说它不是一个“池”,但它解决了每个进程独立连接的问题。

  2. 使用成熟的PHP ORM/DB库的连接管理: 如果你在使用Laravel、ThinkPHP、Yii等框架,它们通常有自己的数据库抽象层和连接管理机制。在Workerman环境下,这些框架的数据库组件通常也能通过一些配置或适配器来很好地工作。例如,Laravel的Eloquent ORM在Workerman中可以配置为按需获取连接或使用更高级的连接池。许多这些库内部已经考虑了连接的重用和状态管理。

  3. Workerman生态内的异步数据库客户端/连接池: Workerman社区提供了一些专门为Workerman设计的异步客户端,它们通常内置了连接池功能。例如,

    workerman/mysql
    (注意,它是一个异步MySQL客户端,不是简单的PHP MySQL扩展封装),它提供了一个非阻塞的MySQL客户端,可以更好地配合Workerman的异步特性,并且通常会自带连接池的管理能力。这种方案对于需要高性能、高并发的异步应用来说是理想选择。

    // 假设使用 workerman/mysql 异步客户端
    use Workerman\Worker;
    use Workerman\MySQL\Connection; // 假设这是 workerman/mysql 的类
    
    $worker = new Worker('websocket://0.0.0.0:2346');
    $worker->count = 4;
    
    $worker->onWorkerStart = function($worker) {
        global $db_pool;
        // 这里可以初始化一个连接池实例,或者直接在onMessage中按需获取
        // workerman/mysql 自身就支持连接池模式
        $db_pool = new Connection('127.0.0.1', 3306, 'user', 'password', 'test');
        // $db_pool 实际上是一个连接池,可以从中获取连接
    };
    
    $worker->onMessage = function($connection, $data) {
        global $db_pool;
        // 从连接池中获取一个连接并执行查询
        $db_pool->query("SELECT * FROM users LIMIT 1", function($result) use ($connection) {
            $connection->send(json_encode($result));
        });
    };
    
    Worker::runAll();

    使用异步客户端的好处是,当数据库查询耗时较长时,不会阻塞Workerman进程,提高了整体的并发处理能力。

选择哪种方案,取决于你的项目规模、性能要求以及你对异步编程的熟悉程度。对于大多数中小项目,第一种

onWorkerStart
的单连接模式已经足够稳定和高效。

如何处理Workerman数据库连接断开与重连问题?

数据库连接断开,这是在任何长驻内存应用中都无法避免的问题,它可能由多种原因引起:数据库服务器重启、网络波动、数据库服务器主动关闭空闲连接(

wait_timeout
)、甚至是Workerman进程本身长时间不活跃。处理这个问题,需要一套健壮的机制。

我通常会从以下几个层面来考虑和实现:

  1. 心跳检测(Keep-Alive): 这是预防连接断开的有效手段。你可以设置一个定时器(例如,每隔几分钟),向数据库发送一个非常轻量级的查询,比如

    SELECT 1
    。这个查询本身不产生大的开销,但能让数据库服务器知道这个连接是活跃的,从而避免因空闲而被关闭。

    use Workerman\Worker;
    use Workerman\Timer;
    use PDO;
    
    // ... (onWorkerStart 部分,确保 $db 是全局变量)
    
    $worker->onWorkerStart = function($worker) {
        global $db;
        // ... 建立 $db 连接的代码 ...
    
        // 每60秒发送一次心跳
        Timer::add(60, function() use (&$db, $worker) {
            try {
                $db->query("SELECT 1");
            } catch (PDOException $e) {
                echo "Worker {$worker->id} 数据库心跳失败,尝试重连...\n";
                // 尝试重连
                try {
                    $db = new PDO('mysql:host=localhost;dbname=test;charset=utf8', 'user', 'password', [
                        PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
                        PDO::ATTR_PERSISTENT => false,
                        PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8"
                    ]);
                    echo "Worker {$worker->id} 数据库重连成功。\n";
                } catch (PDOException $reconnectE) {
                    echo "Worker {$worker->id} 数据库重连失败: " . $reconnectE->getMessage() . "\n";
                    // 重连失败,考虑让进程退出,Workerman会拉起新进程
                    exit(250);
                }
            }
        });
    };
    // ...
  2. 异常捕获与按需重连: 这是最直接的应对方式。在每次执行数据库操作时,都用

    try-catch
    块包裹起来。如果捕获到
    PDOException
    ,并且错误信息表明是连接断开(例如包含
    server has gone away
    或特定的SQLSTATE错误码),那么就尝试重新建立连接。

    // ... (接上面的onWorkerStart部分)
    
    $worker->onMessage = function($connection, $data) {
        global $db;
        $maxRetries = 3; // 最大重试次数
        for ($i = 0; $i < $maxRetries; $i++) {
            try {
                $stmt = $db->prepare("SELECT * FROM users WHERE id = ?");
                $stmt->execute([1]);
                $user = $stmt->fetch(PDO::FETCH_ASSOC);
                $connection->send(json_encode($user));
                return; // 成功则返回
            } catch (PDOException $e) {
                // 判断是否是连接断开错误
                if (strpos($e->getMessage(), 'server has gone away') !== false ||
                    strpos($e->getMessage(), 'SQLSTATE[HY000]') !== false ||
                    strpos($e->getMessage(), 'Broken pipe') !== false) {
    
                    echo "数据库连接断开,尝试重连... (Worker {$connection->worker->id}, 尝试次数: " . ($i + 1) . ")\n";
                    // 尝试重新建立连接
                    try {
                        $db = new PDO('mysql:host=localhost;dbname=test;charset=utf8', 'user', 'password', [
                            PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
                            PDO::ATTR_PERSISTENT => false,
                            PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8"
                        ]);
                        echo "数据库重连成功。\n";
                        // 重连成功后,继续尝试执行原始查询

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
laravel组件介绍
laravel组件介绍

laravel 提供了丰富的组件,包括身份验证、模板引擎、缓存、命令行工具、数据库交互、对象关系映射器、事件处理、文件操作、电子邮件发送、队列管理和数据验证。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

339

2024.04.09

laravel中间件介绍
laravel中间件介绍

laravel 中间件分为五种类型:全局、路由、组、终止和自定。想了解更多laravel中间件的相关内容,可以阅读本专题下面的文章。

293

2024.04.09

laravel使用的设计模式有哪些
laravel使用的设计模式有哪些

laravel使用的设计模式有:1、单例模式;2、工厂方法模式;3、建造者模式;4、适配器模式;5、装饰器模式;6、策略模式;7、观察者模式。想了解更多laravel的相关内容,可以阅读本专题下面的文章。

772

2024.04.09

thinkphp和laravel哪个简单
thinkphp和laravel哪个简单

对于初学者来说,laravel 的入门门槛较低,更易上手,原因包括:1. 更简单的安装和配置;2. 丰富的文档和社区支持;3. 简洁易懂的语法和 api;4. 平缓的学习曲线。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

385

2024.04.10

laravel入门教程
laravel入门教程

本专题整合了laravel入门教程,想了解更多详细内容,请阅读专题下面的文章。

140

2025.08.05

laravel实战教程
laravel实战教程

本专题整合了laravel实战教程,阅读专题下面的文章了解更多详细内容。

85

2025.08.05

laravel面试题
laravel面试题

本专题整合了laravel面试题相关内容,阅读专题下面的文章了解更多详细内容。

79

2025.08.05

PHP高性能API设计与Laravel服务架构实践
PHP高性能API设计与Laravel服务架构实践

本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

418

2026.03.04

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 2.5万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 846人学习

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

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