PDO::fetchAll 与 PDO::fetch 循环
只是一个快速的问题。
在循环中使用 PDO::fetchAll() 和 PDO::fetch() 之间是否存在性能差异(对于大型结果集)?
我正在获取用户定义类的对象,如果这有什么不同的话。
我最初未受过教育的假设是 fetchAll 可能更快,因为 PDO 可以在一个语句中执行多个操作,而 mysql_query 只能执行一个。但是,我对 PDO 的内部工作原理知之甚少,文档也没有说明这一点,以及 fetchAll() 是否只是转储到数组中的 PHP 端循环。
有什么帮助吗?
带有 200k 随机记录的小基准测试。正如预期的那样,fetchAll 方法更快,但需要更多内存。
Result :
fetchAll : 0.35965991020203s, 100249408b
fetch : 0.39197015762329s, 440b
使用的基准代码:
<?php
// First benchmark : speed
$dbh = new PDO('mysql:dbname=testage;dbhost=localhost', 'root', '');
$dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
$sql = 'SELECT * FROM test_table WHERE 1';
$stmt = $dbh->query($sql);
$data = array();
$start_all = microtime(true);
$data = $stmt->fetchAll();
$end_all = microtime(true);$stmt = $dbh->query($sql);
$data = array();
$start_one = microtime(true);
while($data = $stmt->fetch()){}
$end_one = microtime(true);// Second benchmark : memory usage
$stmt = $dbh->query($sql);
$data = array();
$memory_start_all = memory_get_usage();
$data = $stmt->fetchAll();
$memory_end_all = memory_get_usage();$stmt = $dbh->query($sql);
$data = array();
$memory_end_one = 0;
$memory_start_one = memory_get_usage();
while($data = $stmt->fetch()){$memory_end_one = max($memory_end_one, memory_get_usage());
}echo 'Result : <br/>
fetchAll : ' . ($end_all - $start_all) . 's, ' . ($memory_end_all - $memory_start_all) . 'b<br/>
fetch : ' . ($end_one - $start_one) . 's, ' . ($memory_end_one - $memory_start_one) . 'b<br/>';
我发现几乎总是正确的关于 PHP 的一件事是,您自己实现的函数几乎总是比 PHP 等效的要慢。这是因为当用 PHP 实现某些东西时,它没有 C 所具有的所有编译时优化(PHP 是用它编写的),并且 PHP 函数调用的开销很高。
@Arkh
// $data in this case is an array of rows;$data = $stmt->fetchAll();// $data in this case is just one row after each loop;while($data = $stmt->fetch()){}// Try using$i = 0;while($data[$i++] = $stmt->fetch()){}
内存差异应该可以忽略不计
出于非常简单的原因,所有测量“内存占用”的基准实际上都是不正确的。
默认情况下,PDO 确实会将所有内容加载到内存中,并且它不关心您是否使用 fetch 或 fetchAll。要真正获得无缓冲查询的好处,您应该指示 PDO 使用无缓冲查询:
$db->setAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY, false);
在这种情况下,您会看到脚本的内存占用有巨大差异
正如 Mihai Stancu 所说,尽管 fetchAll 胜过 fetch + while,但几乎没有内存差异。
Result :
fetchAll : 0.160676956177s, 118539304b
fetch : 0.121752023697s, 118544392b
我在正确运行时得到了上面的结果:
$i = 0;
while($data[$i++] = $stmt->fetch()){//
}
所以 fetchAll 消耗的内存更少,但是 fetch + while 更快!:)
但是,如果您将获取的数据存储在一个数组中,那么内存使用量肯定是相等的吗?
<?php
define('DB_HOST', 'localhost');
define('DB_USER', 'root');
define('DB_PASS', '');
// database to use
define('DB', 'test');
try
{$dbh = new \PDO('mysql:dbname='. DB .';host='. DB_HOST, DB_USER, DB_PASS); $dbh->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);$sql = 'SELECT * FROM users WHERE 1';$stmt = $dbh->query($sql);$data = array();$start_all = microtime(true);$data = $stmt->fetchAll();$end_all = microtime(true);$stmt = $dbh->query($sql);$data = array();$start_one = microtime(true);while($data = $stmt->fetch()){}$end_one = microtime(true);// Second benchmark : memory usage$stmt = $dbh->query($sql);$data = array();$memory_start_all = memory_get_usage();$data = $stmt->fetchAll();$memory_end_all = memory_get_usage();$stmt = $dbh->query($sql);$data = array();$memory_end_one = 0;$memory_start_one = memory_get_usage();while($data[] = $stmt->fetch()){$memory_end_one = max($memory_end_one, memory_get_usage());}echo 'Result : <br/>fetchAll : ' . ($end_all - $start_all) . 's, ' . ($memory_end_all - $memory_start_all) . 'b<br/>fetch : ' . ($end_one - $start_one) . 's, ' . ($memory_end_one - $memory_start_one) . 'b<br/>';
}
catch ( PDOException $e )
{echo $e->getMessage();
}
?>Result :
fetchAll : 2.6941299438477E-5s, 9824b
fetch : 1.5974044799805E-5s, 9824b
我知道这是一个老话题,但我遇到了同样的问题。在运行了我自己的简单“基准”并阅读了其他人在这里写的内容后,我得出的结论是,这不是一门精确的科学,虽然人们应该努力编写高质量、轻量级的代码,但一开始就浪费太多时间是没有意义的的项目。
我的建议是:通过运行代码(测试版?)一段时间来收集数据,然后开始优化。
在我的简单基准测试(仅测试执行时间)中,我得到的结果在 5% 和 50% 之间变化。我在同一个脚本中运行这两个选项,但是当我首先运行 fetch + 时,它比 fetchall 更快,反之亦然。(我知道我应该单独运行它们,然后几百次得到中位数和平均值,然后进行比较,但是 - 正如我在开始时所说的 - 我得出结论,就我而言,现在开始这样做还为时过早。)
PDO::fetchAll 与 PDO::fetch 循环相关推荐
- php fetchall与fetch,PDO :: fetchAll与PDO :: fetch在循环中
小编典典 几乎没有200k随机记录的基准.与预期的一样,fetchAll方法更快,但需要更多的内存. Result : fetchAll : 0.35965991020203s, 100249408b ...
- pdo mysql fetchall_pdo fetch,pdo fetchAll,pdo query,pdo prepare用法举例
通过pdo来从数据库中调取数据. 其中主要涉及到 pdo fetch,pdo fetchAll,pdo query,pdo prepare使用 . 可以通过以下几种方式,学习下php pdo的用法. ...
- fetchall函数php,PHP PDO fetchAll(),简单结果
关于PDO :: fetchAll()的问题太多,但我仍然找不到答案.我需要获取MySQL返回的结果.例如,如果我有要返回的列ID,名称,年龄,如下所示: array( "id"= ...
- php5.5 pdo mysql_PHP5中PDO的简单使用
PDO(PHP Data Object) 是PHP 5新出来的东西,在PHP 6都要出来的时候,PHP 6只默认使用PDO来处理数据库,将把所有的数据库扩展移到了PECL,那么默认就是没有了我们喜爱的 ...
- php中的PDO对象---对PDO对象的认识、什么是PDO对象
注:本报告总结完全由本人编著,部分概念内容由网络摘录 1 对PDO对象的认识 1.1 简介 PHP 数据对象 (PDO) 扩展为PHP访问数据库定义了一个轻量级的一致接口.实现 PDO 接口的每个数据 ...
- php pdo 封装类,php pdo封装类代码(支持事务)
/** * PDO数据库 * @copyright By GOOGLE */ class pdo_db { /** * PDO实例 * @var PDO */ protected $_db; /** ...
- php封装pdo实例以及pdo长连接的优缺点
文章目录 一.前言 二.为什么选择pdo 三.pdo的长连接 1.什么是pdo的长连接 2.长连接对nginx无效吗 3.php-fpm下的长连接测试 4.长连接对事务的影响 5.总结 四.pdo部分 ...
- php pdo 关闭,php pdo预处理
什么是预处理语句? 可以把它看作是想要运行的 SQL 的一种编译过的模板,它可以使用变量参数进行定制. 预处理语句可以带来两大好处: 查询仅需解析(或预处理)一次,但可以用相同或不同的参数执行多次.当 ...
- linux 下pdo fetchall()不起作用,使用fetchall(PDO :: FETCH_ASSOC)什么都没有返回
function get_prescriptionWith_4($ID){ $sql = "SELECT prescriptions FROM tblprescriptions WHERE ...
最新文章
- JQuery中html、append、appendTo、after、insertAfter系列方法
- Keil uvision 5 MDK 安装教程
- linux - svn命令操作使用随笔svn import
- 智慧交通day00-项目简介
- Ajax提交打开新窗口,浏览器拦截处理;以及跨域问题
- Web前端开发——BAT面试题汇总及答案01
- 后台权限管理系统设计(图文教程)
- 收藏--hibernate——继承关系以及三个subclass标签的区别
- 疫情期间谁干谁赚钱的三个小副业,普通人也能干,基本不用你投资
- 现代前端技术解析:前端三层结构与应用
- mysql获取当前日期的前一天_MySql取得日期(前一天、某一天)
- DevOps 工程师需要必备哪些技能?
- 微服务拆分时机与原则
- IE主页无法修改的办法
- Spring Boot修改启动神图
- nvm管理node版本的使用方法和常用指令
- horizon多网卡配置
- 英语面试常见问题集锦 .
- 压缩算法在单片机升级中的应用
- Windows10磁盘图标感叹号,Bitlocker提示等待激活