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

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中会带来一系列问题:

连接超时与失效: 数据库服务器通常会设置连接超时时间。如果一个Workerman进程长时间没有数据库操作,数据库服务器可能会主动关闭这个空闲连接。但Workerman进程端对此毫不知情,下次尝试使用这个“死连接”时,就会抛出类似“MySQL server has gone away”的错误。资源泄露风险: 如果不妥善管理连接的生命周期,可能导致连接句柄泄露,最终耗尽数据库服务器的连接资源。多进程并发问题: 在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。理解这一点,能帮你少走很多弯路。

绘蛙AI修图 绘蛙AI修图

绘蛙平台AI修图工具,支持手脚修复、商品重绘、AI扩图、AI换色

绘蛙AI修图 279 查看详情 绘蛙AI修图

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

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

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

基于

onWorkerStart

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

use WorkermanWorker;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();

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

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

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

workerman/mysql

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

// 假设使用 workerman/mysql 异步客户端use WorkermanWorker;use WorkermanMySQLConnection; // 假设这是 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进程本身长时间不活跃。处理这个问题,需要一套健壮的机制。

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

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

SELECT 1

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

use WorkermanWorker;use WorkermanTimer;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);            }        }    });};// ...

异常捕获与按需重连:这是最直接的应对方式。在每次执行数据库操作时,都用

try-catch

块包裹起来。如果捕获到

PDOException

,并且错误信息表明是连接断开(例如包含

server has gone away

或特定的SQLSTATE错误码),那么就尝试重新建立连接。

// ... (接上面的onWorkerStart部分)$worker->onMessage = function($connection, $data) {    global $db;    $maxRetries = 3; // 最大重试次数    for ($i = 0; $i 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";                    // 重连成功后,继续尝试执行原始查询

以上就是Workerman支持哪些数据库?Workerman数据库连接方式?的详细内容,更多请关注php中文网其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/714797.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月24日 12:06:20
下一篇 2025年11月24日 12:12:35

相关推荐

  • HTMLrev 上的免费 HTML 网站模板

    HTMLrev 是唯一的人工策划的库专门专注于免费 HTML 模板,适用于由来自世界各地慷慨的模板创建者制作的网站、登陆页面、投资组合、博客、电子商务和管理仪表板世界。 这个人就是我自己 Devluc,我已经工作了 1 年多来构建、改进和更新这个很棒的免费资源。我自己就是一名模板制作者,所以我知道如…

    2025年12月24日
    300
  • 如何使用 Laravel 框架轻松整合微信支付与支付宝支付?

    如何通过 laravel 框架整合微信支付与支付宝支付 在 laravel 开发中,为电商网站或应用程序整合支付网关至关重要。其中,微信支付和支付宝是中国最流行的支付平台。本文将介绍如何使用 laravel 框架封装这两大支付平台。 一个简单有效的方法是使用业内认可的 easywechat lara…

    2025年12月24日
    000
  • Laravel 框架中如何无缝集成微信支付和支付宝支付?

    laravel 框架中微信支付和支付宝支付的封装 如何将微信支付和支付宝支付无缝集成到 laravel 框架中? 建议解决方案 考虑使用 easywechat 的 laravel 版本。easywechat 是一个成熟、维护良好的库,由腾讯官方人员开发,专为处理微信相关功能而设计。其 laravel…

    2025年12月24日
    300
  • 如何在 Laravel 框架中轻松集成微信支付和支付宝支付?

    如何用 laravel 框架集成微信支付和支付宝支付 问题:如何在 laravel 框架中集成微信支付和支付宝支付? 回答: 建议使用 easywechat 的 laravel 版,easywechat 是一个由腾讯工程师开发的高质量微信开放平台 sdk,已被广泛地应用于许多 laravel 项目中…

    2025年12月24日
    000
  • 使用Laravel框架如何整合微信支付和支付宝支付?

    使用 Laravel 框架整合微信支付和支付宝支付 在使用 Laravel 框架开发项目时,整合支付网关是常见的需求。对于微信支付和支付宝支付,推荐采用以下方法: 使用第三方库:EasyWeChat 的 Laravel 版本 建议直接使用现有的 EasyWeChat 的 Laravel 版本。该库由…

    2025年12月24日
    000
  • 如何将微信支付和支付宝支付无缝集成到 Laravel 框架中?

    如何简洁集成微信和支付宝支付到 Laravel 问题: 如何将微信支付和支付宝支付无缝集成到 Laravel 框架中? 答案: 强烈推荐使用流行的 Laravel 包 EasyWeChat,它由腾讯开发者维护。多年来,它一直保持更新,提供了一个稳定可靠的解决方案。 集成步骤: 安装 Laravel …

    2025年12月24日
    100
  • 网络进化!

    Web 应用程序从静态网站到动态网页的演变是由对更具交互性、用户友好性和功能丰富的 Web 体验的需求推动的。以下是这种范式转变的概述: 1. 静态网站(1990 年代) 定义:静态网站由用 HTML 编写的固定内容组成。每个页面都是预先构建并存储在服务器上,并且向每个用户传递相同的内容。技术:HT…

    2025年12月24日
    000
  • 为什么多年的经验让我选择全栈而不是平均栈

    在全栈和平均栈开发方面工作了 6 年多,我可以告诉您,虽然这两种方法都是流行且有效的方法,但它们满足不同的需求,并且有自己的优点和缺点。这两个堆栈都可以帮助您创建 Web 应用程序,但它们的实现方式却截然不同。如果您在两者之间难以选择,我希望我在两者之间的经验能给您一些有用的见解。 在这篇文章中,我…

    2025年12月24日
    000
  • 深入理解CSS框架与JS之间的关系

    深入理解CSS框架与JS之间的关系 在现代web开发中,CSS框架和JavaScript (JS) 是两个常用的工具。CSS框架通过提供一系列样式和布局选项,可以帮助我们快速构建美观的网页。而JS则提供了一套功能强大的脚本语言,可以为网页添加交互和动态效果。本文将深入探讨CSS框架和JS之间的关系,…

    2025年12月24日
    000
  • HTML+CSS+JS实现雪花飘扬(代码分享)

    使用html+css+js如何实现下雪特效?下面本篇文章给大家分享一个html+css+js实现雪花飘扬的示例,希望对大家有所帮助。 很多南方的小伙伴可能没怎么见过或者从来没见过下雪,今天我给大家带来一个小Demo,模拟了下雪场景,首先让我们看一下运行效果 可以点击看看在线运行:http://hai…

    2025年12月24日 好文分享
    500
  • 10款好看且实用的文字动画特效,让你的页面更吸引人!

    图片和文字是网页不可缺少的组成部分,图片运用得当可以让网页变得生动,但普通的文字不行。那么就可以给文字添加一些样式,实现一下好看的文字效果,让页面变得更交互,更吸引人。下面创想鸟就来给大家分享10款文字动画特效,好看且实用,快来收藏吧! 1、网页玻璃文字动画特效 模板简介:使用css3制作网页渐变底…

    2025年12月24日 好文分享
    000
  • tp5如何引入css文件

    tp5引入css文件的方法:1、将css文件放在public目录下的static文件里即可;2、在页面引入中写上“”语句即可。 本教程操作环境:windows7系统、CSS3&&HTML5版、Dell G3电脑。 其实很简单,只需要将css,js,image文件放在这个目录下即可 页…

    2025年12月24日
    000
  • 网页设计css样式代码大全,快来收藏吧!

    减少很多不必要的代码,html+css可以很方便的进行网页的排版布局。小伙伴们收藏好哦~ 一.文本设置    1、font-size: 字号参数  2、font-style: 字体格式 3、font-weight: 字体粗细 4、颜色属性 立即学习“前端免费学习笔记(深入)”; color: 参数 …

    2025年12月24日
    000
  • css中id选择器和class选择器有何不同

    之前的文章《什么是CSS语法?详细介绍使用方法及规则》中带了解CSS语法使用方法及规则。下面本篇文章来带大家了解一下CSS中的id选择器与class选择器,介绍一下它们的区别,快来一起学习吧!! id选择器和class选择器介绍 CSS中对html元素的样式进行控制是通过CSS选择器来完成的,最常用…

    2025年12月24日
    000
  • 聊聊CSS 与 JS 是如何阻塞 DOM 解析和渲染的

    本篇文章给大家介绍一下css和js阻塞 dom 解析和渲染的原理。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 hello~各位亲爱的看官老爷们大家好。估计大家都听过,尽量将CSS放头部,JS放底部,这样可以提高页面的性能。然而,为什么呢?大家有考虑过么?很长一段时间,我都是知其…

    2025年12月24日
    200
  • js如何修改css样式

    js修改css样式的方法:1、使用【obj.className】来修改样式表的类名;2、使用【obj.style.cssTest】来修改嵌入式的css;3、使用【obj.className】来修改样式表的类名;4、使用更改外联的css。 本教程操作环境:windows7系统、css3版,DELL G…

    2025年12月24日
    000
  • 如何使用纯CSS、JS实现图片轮播效果

    本篇文章给大家详细介绍一下使用纯css、js实现图片轮播效果的方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。 .carousel {width: 648px;height: 400px;margin: 0 auto;text-align: center;position: a…

    2025年12月24日
    000
  • js如何修改css

    js修改css的方法:1、使用【obj.style.cssTest】来修改嵌入式的css;2、使用【bj.className】来修改样式表的类名;3、使用更改外联的css文件,从而改变元素的css。 本教程操作环境:windows7系统、css3版,DELL G3电脑。 js修改css的方法: 方法…

    2025年12月24日
    000
  • js如何改变css样式

    js改变css样式的方法:1、使用cssText方法;2、使用【setProperty()】方法;3、使用css属性对应的style属性。 本教程操作环境:windows7系统、css3版,DELL G3电脑。 js改变css样式的方法: 第一种:用cssText div.style.cssText…

    2025年12月24日
    000
  • nginx的css不起作用怎么办

    nginx的css不起作用是因为误删文件导致的,其解决办法就是打开相应的文件并添加代码“include /etc/nginx/mime.types;”,然后重启Nginx守护即可。 本文操作环境:windows7系统、css3版,DELL G3电脑。 nginx的css不起作用是什么原因? 最近部署…

    2025年12月24日 好文分享
    000

发表回复

登录后才能评论
关注微信