只是一个快速的问题。

在循环中使用 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 循环相关推荐

  1. php fetchall与fetch,PDO :: fetchAll与PDO :: fetch在循环中

    小编典典 几乎没有200k随机记录的基准.与预期的一样,fetchAll方法更快,但需要更多的内存. Result : fetchAll : 0.35965991020203s, 100249408b ...

  2. pdo mysql fetchall_pdo fetch,pdo fetchAll,pdo query,pdo prepare用法举例

    通过pdo来从数据库中调取数据. 其中主要涉及到 pdo fetch,pdo fetchAll,pdo query,pdo prepare使用 . 可以通过以下几种方式,学习下php pdo的用法. ...

  3. fetchall函数php,PHP PDO fetchAll(),简单结果

    关于PDO :: fetchAll()的问题太多,但我仍然找不到答案.我需要获取MySQL返回的结果.例如,如果我有要返回的列ID,名称,年龄,如下所示: array( "id"= ...

  4. php5.5 pdo mysql_PHP5中PDO的简单使用

    PDO(PHP Data Object) 是PHP 5新出来的东西,在PHP 6都要出来的时候,PHP 6只默认使用PDO来处理数据库,将把所有的数据库扩展移到了PECL,那么默认就是没有了我们喜爱的 ...

  5. php中的PDO对象---对PDO对象的认识、什么是PDO对象

    注:本报告总结完全由本人编著,部分概念内容由网络摘录 1 对PDO对象的认识 1.1 简介 PHP 数据对象 (PDO) 扩展为PHP访问数据库定义了一个轻量级的一致接口.实现 PDO 接口的每个数据 ...

  6. php pdo 封装类,php pdo封装类代码(支持事务)

    /** * PDO数据库 * @copyright By GOOGLE */ class pdo_db { /** * PDO实例 * @var PDO */ protected $_db; /** ...

  7. php封装pdo实例以及pdo长连接的优缺点

    文章目录 一.前言 二.为什么选择pdo 三.pdo的长连接 1.什么是pdo的长连接 2.长连接对nginx无效吗 3.php-fpm下的长连接测试 4.长连接对事务的影响 5.总结 四.pdo部分 ...

  8. php pdo 关闭,php pdo预处理

    什么是预处理语句? 可以把它看作是想要运行的 SQL 的一种编译过的模板,它可以使用变量参数进行定制. 预处理语句可以带来两大好处: 查询仅需解析(或预处理)一次,但可以用相同或不同的参数执行多次.当 ...

  9. linux 下pdo fetchall()不起作用,使用fetchall(PDO :: FETCH_ASSOC)什么都没有返回

    function get_prescriptionWith_4($ID){ $sql = "SELECT prescriptions FROM tblprescriptions WHERE ...

最新文章

  1. JQuery中html、append、appendTo、after、insertAfter系列方法
  2. Keil uvision 5 MDK 安装教程
  3. linux - svn命令操作使用随笔svn import
  4. 智慧交通day00-项目简介
  5. Ajax提交打开新窗口,浏览器拦截处理;以及跨域问题
  6. Web前端开发——BAT面试题汇总及答案01
  7. 后台权限管理系统设计(图文教程)
  8. 收藏--hibernate——继承关系以及三个subclass标签的区别
  9. 疫情期间谁干谁赚钱的三个小副业,普通人也能干,基本不用你投资
  10. 现代前端技术解析:前端三层结构与应用
  11. mysql获取当前日期的前一天_MySql取得日期(前一天、某一天)
  12. DevOps 工程师需要必备哪些技能?
  13. 微服务拆分时机与原则
  14. IE主页无法修改的办法
  15. Spring Boot修改启动神图
  16. nvm管理node版本的使用方法和常用指令
  17. horizon多网卡配置
  18. 英语面试常见问题集锦 .
  19. 压缩算法在单片机升级中的应用
  20. Windows10磁盘图标感叹号,Bitlocker提示等待激活

热门文章

  1. QT屏蔽qDebug的方法
  2. 使用Python模拟登录知乎
  3. MySQL中的空值 Null 和空字符‘‘
  4. AAA信用等级证书,八证一牌不可或缺
  5. Android 静态分析smail
  6. 包装类型之间一律使用equals来判断值是否相等
  7. 小身板大能量 清微智能边缘盒子满足多场景智能应用
  8. 树莓派3B+使用0.96 oled 屏幕(附完整可运行代码)
  9. 信誉卡逾期以后该怎么办
  10. Android程序员正在消失,我也不想干了,要转行吗?