文档章节

PHP+MySQL缓冲查询和无缓冲查询

eechen
 eechen
发布于 2015/12/30 10:46
字数 927
阅读 1221
收藏 3
http://php.net/manual/zh/mysqlinfo.concepts.buffering.php
http://php.net/manual/zh/mysqli.query.php
PHP MySQL查询(mysqli,pdo_mysql)默认使用缓冲模式.
也就是说查询结果将一次性从MySQL传输到PHP进程内存中,
这时可以统计结果集的行数,以及移动结果集指针.
缓冲模式下,如果结果集很大,那么PHP进程也会占用大量的内存,
直到结果集被unset或者free.

store_result用于缓冲模式,所有结果一次性存储到PHP进程中:
mysqli::query MYSQLI_STORE_RESULT
mysqli::store_result
mysqli_stmt::store_result
如果PHP的MySQL数据库驱动底层用的是libmysqlclient,那么memory_limit不能统计到结果集占用的内存,
除非结果集已经赋值给PHP变量,如果底层使用mysqlnd作为驱动时则可以统计到(PHP从5.4开始底层默认使用mysqlnd).
无缓冲模式下执行的查询将会返回一个resource资源引用,位于MySQL的查询结果等待PHP获取.
无缓冲模式下,PHP进程占用的内存很少,但会增大MySQL服务器的负载.
在PHP取回所有结果前,在当前数据库连接下不能发送其他的查询请求.

use_result表示无缓冲查询:
mysqli::query MYSQLI_USE_RESULT
mysqli::use_result

总结:
当结果集不大时,或者需要在读取所有行前获取结果集行数时,使用缓冲查询(默认).
当结果集很大时,使用无缓冲查询,避免PHP进程占用大量的内存.
$rs = $mysqli->query("SELECT * FROM City", MYSQLI_USE_RESULT);
$pdo->setAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY, false);
$rs = $pdo->query("SELECT * FROM City");

默认情况下,mysqli_stmt的SELECT查询结果将留在MySQL服务器上,等待fetch方法把记录逐条取回到PHP程序中,这样做会降低性能,但能节省内存.
如果需要对所有记录进行处理,可以调用mysqli_stmt::store_result,把所有结果一次性全部传回到PHP程序中,
这样做更高效,能减轻MySQL服务器的负担,虽然内存占用会多一些.
如果获取SELECT语句查找到了多少条记录,可以用 mysqli_stmt::$num_rows 获取.
这个属性只有在提前执行过 mysqli_stmt::store_result 方法,将全部查询结果传回到PHP程序中的情况下才可以使用.
对比 mysqli_result::$num_rows 则不没有这个限制.
用 mysqli_stmt::free_result 关闭 mysqli_stmt::store_result:
$stmt->store_result();
echo $stmt->num_rows;
$stmt->free_result();

http://php.net/manual/zh/mysqli-stmt.fetch.php
mysqli_stmt::store_result能让mysqli_stmt::fetch更高效,但也需要用mysqli_stmt::free_result显式关闭.
可以用mysqli_stmt::get_result拿到结果集对象$result,然后mysqli_result::fetch_all拿到查询数组$results:
$result = $stmt->get_result();
$results = $result->fetch_all(MYSQLI_ASSOC);

http://php.net/mysqli
mysqli::query              执行SQL,成功返回 mysqli_result(SELECT,SHOW,DESCRIBE操作)对象或TRUE(其他操作), 失败返回FALSE.用 mysqli::close 关闭.
mysqli::prepare            预处理SQL,成功返回 statement 对象, 失败返回 FALSE.
mysqli_stmt::execute       执行SQL.用 mysqli_stmt::close 关闭.
mysqli_stmt::store_result  取回全部查询结果(SELECT,SHOW,DESCRIBE,EXPLAIN)到PHP,可选.用 mysqli_stmt::free_result 关闭.
mysqli_stmt::bind_result   把prepare和execute产生的结构绑定结果到变量,然后在 mysqli_stmt::fetch 中把这些变量输出或赋值.
mysqli_stmt::fetch         每次返回结果集的一条,赋值给 mysqli_stmt::bind_result 绑定的变量.
mysqli_stmt::get_result    获得结果对象,然后调用 mysqli_result::fetch_all 就能返回结果集数组.mysqlnd下可用.
mysqli_result::fetch_all   返回一个结果集数组(MYSQLI_NUM(默认),MYSQLI_ASSOC,MYSQLI_BOTH),用 mysqli_result::close 关闭.mysqlnd下可用.
mysqli_result::fetch_array 每次返回结果集的一条,包含一个一维的数字数组和关联数组.
mysqli_result::fetch_assoc 每次返回结果集的一条,即一个一维的关联数组.
mysqli_result::fetch_row   每次返回结果集的一条,即一个一维的数字数组.

© 著作权归作者所有

共有 人打赏支持
eechen

eechen

粉丝 986
博文 107
码字总数 55962
作品 1
深圳
加载中

评论(1)

MySQL 优化之 query_cache_limit

querycachelimit querycachelimit指定单个查询能够使用的缓冲区大小,缺省为1M。 优化querycachesize 从4.0.1开始,MySQL提供了查询缓冲机制。使用查询缓冲,MySQL将SELECT语句和查询结果存放...

鉴客
2011/08/16
3.5K
0
mysql 优化(1)

一、 通过查询缓冲提高查询速度   一般我们使用SQL语句进行查询时,数据库服务器每次在收到客户端 发来SQL后,都会执行这条SQL语句。但当在一定间隔内(如1分钟内),接到完全一样的SQL语句...

真爱2015
2015/12/07
109
1
如何解决PHP里大量数据循环时内存耗尽的问题

在开发一个PHP程序时我们有时会遇到了下面的错误: 代码如下:PHP Fatal error: Allowed memory size of 268 435 456 bytes exhausted 错误信息显示允许的最大内存已经耗尽。遇到这样的错误起...

snowing1990
2016/03/07
19
0
PHP大量数据循环时内存耗尽问题的解决方案

最近在开发一个PHP程序时遇到了下面的错误: PHP Fatal error: Allowed memory size of 268 435 456 bytes exhausted 错误信息显示允许的最大内存已经耗尽。遇到这样的错误起初让我很诧异,但...

snowing1990
2016/04/15
12
0
你知道这样可以解决PHP里大量数据循环时内存耗尽的问题吗?

最近在开发一个PHP程序时遇到了下面的错误: PHP Fatal error: Allowed memory size of 268 435 456 bytes exhausted 错误信息显示允许的最大内存已经耗尽。遇到这样的错误起初让我很诧异,但...

咦我的天
2017/07/17
178
2

没有更多内容

加载失败,请刷新页面

加载更多

CentOS7.X部署Zabbix4.0

  /usr/share/doc/zabbix-server-mysql*/create.sql.gz | mysql -uzabbix -p 密码 数据库导入完成后需要个Zabbix配置数据库密码,修改/etc/zabbix/zabbix_server.conf文件:   DBPassword......

寰宇01
12分钟前
1
0
工厂模式

(从别人里copy的,方便自己查看) 意图:定义一个创建对象的接口,让其子类自己决定实例化哪一个工厂类,工厂模式使其创建过程延迟到子类进行。 主要解决:主要解决接口选择的问题。 何时使...

南桥北木
38分钟前
0
0
rabbitmq学习(一)

RabbitMQ是目前非常热门的一款消息中间件,具有高可靠、易拓展、高可用及丰富的功能 1.什么是消息中间件 消息是指在应用间传送的数据。包含文本字符串、JSON、内嵌对象 消息队列中间件(消息...

hensemlee
42分钟前
1
0
学习设计模式——原型模式

1. 认识原型模式 1. 定义:用原型实例指定创建对象的种类,并通过拷贝这些原型创建新的对象。 2. 结构: Prototype:声明一个克隆自身的接口,用来约束想要克隆自己的具体实现类,要求这些类...

江左煤郎
48分钟前
1
0
观察者模式

观察者模式的套路 有如下角色: 事件,比如修改,用户点击; 事件队列,触发事件之后,会把事件一个一个放入事件队列 监听器,采用某种方式(一般是轮询,或者io阻塞机制),来判断事件队列是否有新的未...

黄威
51分钟前
1
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部