程德耀 發表於 2026-5-3 17:21:53

PHP建立MySQL与MySQLi持久化连接(长连接)区别

<div id="navCategory"><h5 class="catalogue">目录</h5><ul class="first_class_ul"><li><a href="#_label0">持久化连接的优势</a></li><li><a href="#_label1">mysql 和 mysqli 扩展的区别如下:</a></li><li><a href="#_label2">持久化长连接的风险</a></li><li><a href="#_label3">为什么我的长连接不生效?</a></li></ul></div><p>在PHP开发中,与数据库的交互是非常常见的操作。MySQL是一种流行的关系型数据库,而PHP为其提供了两种不同的API,即MySQL和MySQLi。在使用这两个API时,我们可以选择使用持久连接或非持久连接。MySQL的持久连接需使用`mysql_pconnect`,而MySQLi持久连接默认启用。尽管MySQLi有性能和安全优势,但使用持久连接时应注意服务器连接限制和资源占用问题。</p>
<p>php-mysql的持久连接只在同一个进程里可&ldquo;复用&rdquo;,不同进程之间玩不了。fpm的子进程可以常驻,所以只要子进程还在,那么上一次请求创建的持久连接,下一次请求还可以复用。cli每次都新起一个进程,没法把连接复用,那么短连接和持久连接其实是一样的。</p>
<p>如果在 PHP 5.3 的版本以前想要创建MySQL的持久化连接(长连接),需要显式调用 pconnect 创建:</p>
<div class="jb51code"><pre class="brush:php;">$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……
}
</pre></div>
<p>从 PHP 5.3 开始, mysqli 扩展开始支持持久化连接,持久化连接已经在 PDO MYSQL 和 ext/mysql 中提供支持。</p>
<p class="maodian"><a name="_label0"></a></p><h2>持久化连接的优势</h2>
<p>持久化连接的目的在于重用客户端到服务器之间的连接, 而不是每次在需要的时候都重新建立一个连接。<br /><strong>由于持久化连接可以将已经建立的连接缓存起来,以备后续的使用, 所以省去了建立新的连接的开销, 因此可以带来性能上的提升</strong>。</p>
<p>不像 mysql 扩展,mysqli 没有提供一个特殊的方法用于打开持久化连接。 如果 mysqli 打开一个持久化连接,需要在创建连接时,在host前面增加<code>p:</code>两个字符。</p>
<p class="maodian"><a name="_label1"></a></p><h2>mysql 和 mysqli 扩展的区别如下:</h2>
<blockquote><p>持久链接建立方式,mysqli是在host前面增加&ldquo;p:&rdquo;两个字符;mysql使用mysql_pconnect函数;<br />mysqli 建立的持久链接,必须在mysqli_close之后,才能被下一个请求复用;mysql的长连接,可以立即被复用;<br />pdo 建立的持久链接,不必关闭,就能复用;<br />mysqli 建立持久链接时,会自动清理上一个会话变量、回滚事务、解锁表、释放锁等操作;而 mysql 扩展则不会(<strong>这点非常重要</strong>);<br />mysqli 判断是否为同一持久链接标识是 IP,PORT、USER、PASS、DBNAME、SOCKET;mysql 是 IP、PORT、USER、PASS、CLIENT_FLAGS;</p></blockquote>
<p class="maodian"><a name="_label2"></a></p><h2>持久化长连接的风险</h2>
<blockquote><p>使用持久化连接也会存在一些风险, 因为在缓存中的连接可能处于一种不可预测的状态。<br />例如,如果客户端未能正常关闭连接, 可能在这个连接上残留了对库表的锁, 那么当这个连接被其他请求重用的时候,这个连接还是处于 有锁的状态。<br />所以,如果要很好的使用持久化连接,那么要求代码在和数据库进行交互的时候, 确保做好清理工作,保证被缓存的连接是一个干净的,没有残留的状态。</p></blockquote>
<p>mysqli 扩展的持久化连接提供了内建的清理处理代码。 mysqli 所做的清理工作包括:</p>
<blockquote><p>回滚处于活动状态的事务<br />关闭并且删除临时表<br />对表解锁<br />重置会话变量<br />关闭预编译 SQL 语句(在PHP中经常发生)<br />关闭处理程序<br />释放通过 GET_LOCK() 获得的锁</p></blockquote>
<p>这确保了将连接返回到连接池的时候, 它处于一种&ldquo;干净&rdquo;的状态,可以被其他客户端进程所使用。</p>
<blockquote><p>mysqli 扩展 通过自动调用 C-API 函数 mysql_change_user() 来完成这个清理工作。</p></blockquote>
<p>自动清理的特性有优点也有缺点:<br />优点是程序员不再需要担心附加的清理代码, 因为它们会自动调用。<br />然而缺点就是 性能 可能会 慢一点, 因为每次从连接池返回一个连接都需要执行这些清理代码。</p>
<p>这个自动清理的代码可以通过在编译 php 时定义<br /><code>MYSQLI_NO_CHANGE_USER_ON_PCONNECT</code> 来关闭。</p>
<blockquote><p>注意: mysqli 扩展在使用 MySQL Native Driver 或 Mysql Client Library(libmysql)时都支持持久化连接。</p></blockquote>
<p class="maodian"><a name="_label3"></a></p><h2>为什么我的长连接不生效?</h2>
<p>村长多说两句,相信很多小伙伴遇到过这个问题:</p>
<blockquote><p>明明创建了 pconnect,show process_list 查看数据库链接,却发现长连接没有被复用,而是重新创建了一个,这是为啥呢?这得从PHP的运作模式说起。</p></blockquote>
<p>一般 php 有2种运行模式, 一是作为 cgi 运行, 二是作为 apache 的模块运行:</p>
<blockquote><p>作为 cgi 的时候 connect 跟 pconnect 没什么不同, 因为每次 cgi 进行运行结束后都会被销毁清理掉资源;<br />php 作为 apache 模块方式运行时, 可以使用到数据库持续连接, 但可能会存在潜在的问题;</p></blockquote>
<p><strong>说白了,如果你是 cgi 运行方式,pconnection 永远也不会生效</strong>。</p>
<p>长连接最大的缺点就是万一一个用户锁死,当前进程就永久锁死了;假如你在apache里的设置是进程永不销毁的话就&hellip;<br />总之,尽量使用 mysql_connect,因为运行结束后会自动断开;再说了现在的机器性能都不差,不至于缺少那点儿创建销毁带来的内存开销。</p>
<p><img alt="" src="https://img.jbzj.com/file_images/article/202502/2025022214363938.png" /></p>
頁: [1]
查看完整版本: PHP建立MySQL与MySQLi持久化连接(长连接)区别