当前位置: 首页 > news >正文

PHP建立MySQL持久化连接(长连接)及mysql与mysqli扩展的区别

如果在 PHP 5.3 的版本以前想要创建MySQL的持久化连接(长连接),需要显式调用 pconnect 创建:

$con = mysql_pconnect($server['host'], $server['username'], $server['password']);
if (!($con === false)) {
if (mysql_select_db($server['database'], $con) === false) {echo('Could not select database: ' . mysql_error());continue;
}
// do something here……
}

从 PHP 5.3 开始, mysqli 扩展开始支持持久化连接,持久化连接已经在 PDO MYSQL 和 ext/mysql 中提供支持。

持久化连接的优势

持久化连接的目的在于重用客户端到服务器之间的连接, 而不是每次在需要的时候都重新建立一个连接。
由于持久化连接可以将已经建立的连接缓存起来,以备后续的使用, 所以省去了建立新的连接的开销, 因此可以带来性能上的提升

不像 mysql 扩展,mysqli 没有提供一个特殊的方法用于打开持久化连接。 如果 mysqli 打开一个持久化连接,需要在创建连接时,在host前面增加p:两个字符。

mysql 和 mysqli 扩展的区别如下:

持久链接建立方式,mysqli是在host前面增加“p:”两个字符;mysql使用mysql_pconnect函数;
mysqli 建立的持久链接,必须在mysqli_close之后,才能被下一个请求复用;mysql的长连接,可以立即被复用;
pdo 建立的持久链接,不必关闭,就能复用;
mysqli 建立持久链接时,会自动清理上一个会话变量、回滚事务、解锁表、释放锁等操作;而 mysql 扩展则不会(这点非常重要);
mysqli 判断是否为同一持久链接标识是 IP,PORT、USER、PASS、DBNAME、SOCKET;mysql 是 IP、PORT、USER、PASS、CLIENT_FLAGS;

持久化长连接的风险

使用持久化连接也会存在一些风险, 因为在缓存中的连接可能处于一种不可预测的状态。
例如,如果客户端未能正常关闭连接, 可能在这个连接上残留了对库表的锁, 那么当这个连接被其他请求重用的时候,这个连接还是处于 有锁的状态。
所以,如果要很好的使用持久化连接,那么要求代码在和数据库进行交互的时候, 确保做好清理工作,保证被缓存的连接是一个干净的,没有残留的状态。

mysqli 扩展的持久化连接提供了内建的清理处理代码。 mysqli 所做的清理工作包括:

回滚处于活动状态的事务
关闭并且删除临时表
对表解锁
重置会话变量
关闭预编译 SQL 语句(在PHP中经常发生)
关闭处理程序
释放通过 GET_LOCK() 获得的锁

这确保了将连接返回到连接池的时候, 它处于一种“干净”的状态,可以被其他客户端进程所使用。

mysqli 扩展 通过自动调用 C-API 函数 mysql_change_user() 来完成这个清理工作。

自动清理的特性有优点也有缺点:
优点是程序员不再需要担心附加的清理代码, 因为它们会自动调用。
然而缺点就是 性能 可能会 慢一点, 因为每次从连接池返回一个连接都需要执行这些清理代码。

这个自动清理的代码可以通过在编译 php 时定义
MYSQLI_NO_CHANGE_USER_ON_PCONNECT 来关闭。

注意: mysqli 扩展在使用 MySQL Native Driver 或 Mysql Client Library(libmysql)时都支持持久化连接。

为什么我的长连接不生效?

村长多说两句,相信很多小伙伴遇到过这个问题:

明明创建了 pconnect,show process_list 查看数据库链接,却发现长连接没有被复用,而是重新创建了一个,这是为啥呢?这得从PHP的运作模式说起。

一般 php 有2种运行模式, 一是作为 cgi 运行, 二是作为 apache 的模块运行:

作为 cgi 的时候 connect 跟 pconnect 没什么不同, 因为每次 cgi 进行运行结束后都会被销毁清理掉资源;
php 作为 apache 模块方式运行时, 可以使用到数据库持续连接, 但可能会存在潜在的问题;

说白了,如果你是 cgi 运行方式,pconnection 永远也不会生效

长连接最大的缺点就是万一一个用户锁死,当前进程就永久锁死了;假如你在apache里的设置是进程永不销毁的话就…
总之,尽量使用 mysql_connect,因为运行结束后会自动断开;再说了现在的机器性能都不差,不至于缺少那点儿创建销毁带来的内存开销。

最后编辑于:2025-02-05 20:59:54


喜欢的朋友记得点赞、收藏、关注哦!!!

http://www.lryc.cn/news/540932.html

相关文章:

  • python爬虫系列课程2:如何下载Xpath Helper
  • 【Python项目】基于Python的Web漏洞挖掘系统
  • 多环境日志管理:使用Logback与Logstash集成实现高效日志处理
  • idea连接gitee(使用idea远程兼容gitee)
  • STM32 看门狗
  • 飞书API
  • 深入解析 Hydra 库:灵活强大的 Python 配置管理框架
  • 【开源免费】基于Vue和SpringBoot的失物招领平台(附论文)
  • 科普:你的笔记本电脑中有三个IP:127.0.0.1、无线网 IP 和局域网 IP;两个域名:localhost和host.docker.internal
  • 测试WSS服务器
  • unity学习49:寻路网格链接 offMeshLinks, 以及传送门效果
  • Web 开发中的 5 大跨域标签解析:如何安全地进行跨域请求与加载外部资源
  • UMLS数据下载及访问
  • 23种设计模式 - 空对象模式
  • Redis三剑客解决方案
  • 大学本科教务系统设计方案,涵盖需求分析、架构设计、核心模块和技术实现要点
  • Docker Mysql 数据迁移
  • ubuntu22.04离线安装K8S
  • 微信小程序中将图片截图为正方形(自动居中)
  • 传统的自动化行业的触摸屏和上位机,PLC是否会被取代?
  • 【论文精读】VLM-AD:通过视觉-语言模型监督实现端到端自动驾驶
  • 2024年数字政府服务能力优秀创新案例汇编(附下载)
  • Ollama Docker 镜像部署
  • [深度学习][python]yolov12+bytetrack+pyqt5实现目标追踪
  • 【深度学习】矩阵的理解与应用
  • 我是如何从 0 到 1 找到 Web3 工作的?
  • 《Keras 3 :使用 Vision Transformers 进行物体检测》:此文为AI自动翻译
  • java(spring boot)实现向deepseek/GPT等模型的api发送请求/多轮对话(附源码)
  • module ‘cv2.dnn‘ has no attribute ‘DictValue‘解决办法
  • 将RocketMQ集成到了Spring Boot项目中,实现站内信功能