Mysql中分页查询两个方法比较

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
云数据库 RDS MySQL,高可用系列 2核4GB
简介:

mysql中分页查询有两种方式, 一种是使用COUNT(*)的方式,具体代码如下

1
2
3
SELECT COUNT(*) FROM foo WHERE b = 1;
 
SELECT a FROM foo WHERE b = 1 LIMIT 100,10;
1
  

另外一种是使用SQL_CALC_FOUND_ROWS

1
2
SELECT SQL_CALC_FOUND_ROWS a FROM foo WHERE b = 1 LIMIT 100, 10;
SELECT FOUND_ROWS();

 

第二种方式调用SQL_CALC_FOUND_ROWS之后会将WHERE语句查询的行数放在FOUND_ROWS()之中,第二次只需要查询FOUND_ROWS()就可以查出有多少行了。

 

讨论这两种方法的优缺点:

首先原子性讲,第二种肯定比第一种好。第二种能保证查询语句的原子性,第一种当两个请求之间有额外的操作修改了表的时候,结果就自然是不准确的了。而第二种则不会。但是非常可惜,一般页面需要进行分页显示的时候,往往并不要求分页的结果非常准确。即分页返回的total总数大1或者小1都是无所谓的。所以其实原子性不是我们分页关注的重点。

 

下面看效率。这个非常重要,分页操作在每个网站上的使用都是非常大的,查询量自然也很大。由于无论哪种,分页操作必然会有两次sql查询,于是就有很多很多关于两种查询性能的比较:

SQL_CALC_FOUND_ROWS真的很慢么?

http://hi.baidu.com/thinkinginlamp/item/b122fdaea5ba23f614329b14

To SQL_CALC_FOUND_ROWS or not to SQL_CALC_FOUND_ROWS?

http://www.mysqlperformanceblog.com/2007/08/28/to-sql_calc_found_rows-or-not-to-sql_calc_found_rows/

老王这篇文章里面有提到一个covering index的概念,简单来说就是怎样才能只让查询根据索引返回结果,而不进行表查询

具体看他的另外一篇文章:

MySQL之Covering Index

http://hi.baidu.com/thinkinginlamp/item/1b9aaf09014acce0f45ba6d3

 

实验

结合这几篇文章,做的实验:

表:

1
2
3
4
5
6
7
CREATE TABLE IF NOT EXISTS `foo` (
`a`  int (10) unsigned NOT NULL AUTO_INCREMENT,
`b`  int (10) unsigned NOT NULL,
`c` varchar(100) NOT NULL,
PRIMARY KEY (`a`),
KEY `bar` (`b`,`a`)
) ENGINE=MyISAM;

 

 

注意下这里是使用b,a做了一个索引,所以查询select * 的时候是不会用到covering index的,select a才会使用到covering index

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
<?php
 
$host  = '192.168.100.166' ;
$dbName  = 'test' ;
$user  = 'root' ;
$password  = '' ;
 
$db  = mysql_connect( $host , $user , $password ) or  die ( 'DB connect failed' );
mysql_select_db( $dbName , $db );
 
 
echo  '=========================================='  . "\r\n" ;
 
$start  = microtime(true);
for  ( $i  =0; $i <1000; $i ++) {
     mysql_query( "SELECT SQL_NO_CACHE COUNT(*) FROM foo WHERE b = 1" );
     mysql_query( "SELECT SQL_NO_CACHE a FROM foo WHERE b = 1 LIMIT 100,10" );
}
$end  = microtime(true);
echo  $end  - $start  . "\r\n" ;
 
echo  '=========================================='  . "\r\n" ;
 
$start  = microtime(true);
for  ( $i  =0; $i <1000; $i ++) {
     mysql_query( "SELECT SQL_NO_CACHE SQL_CALC_FOUND_ROWS a FROM foo WHERE b = 1 LIMIT 100, 10" );
     mysql_query( "SELECT FOUND_ROWS()" );
}
$end  = microtime(true);
echo  $end  - $start  . "\r\n" ;
 
echo  '=========================================='  . "\r\n" ;
 
$start  = microtime(true);
for  ( $i  =0; $i <1000; $i ++) {
     mysql_query( "SELECT SQL_NO_CACHE COUNT(*) FROM foo WHERE b = 1" );
     mysql_query( "SELECT SQL_NO_CACHE * FROM foo WHERE b = 1 LIMIT 100,10" );
}
$end  = microtime(true);
echo  $end  - $start  . "\r\n" ;
 
echo  '=========================================='  . "\r\n" ;
 
$start  = microtime(true);
for  ( $i  =0; $i <1000; $i ++) {
     mysql_query( "SELECT SQL_NO_CACHE SQL_CALC_FOUND_ROWS * FROM foo WHERE b = 1 LIMIT 100, 10" );
     mysql_query( "SELECT FOUND_ROWS()" );
}
$end  = microtime(true);
echo  $end  - $start  . "\r\n" ;

返回的结果:

clip_image001

和老王里面文章说的是一样的。第四次查询SQL_CALC_FOUND_ROWS由于不仅是没有使用到covering index,也需要进行全表查询,而第三次查询COUNT(*),且select * 有使用到index,并没进行全表查询,所以有这么大的差别。

 

总结

PS: 另外提醒下,这里是使用MyISAM会出现三和四的查询差别这么大,但是如果是使用InnoDB的话,就不会有这么大差别了。

所以我得出的结论是如果数据库是InnoDB的话,我还是倾向于使用SQL_CALC_FOUND_ROWS

 

结论:SQL_CALC_FOUND_ROWS和COUNT(*)的性能在都使用covering index的情况下前者高,在没使用covering index情况下后者性能高。所以使用的时候要注意这个。

相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
相关文章
|
关系型数据库 MySQL
MySql limit 数据分页
MySql limit 数据分页
59 0
|
7月前
|
关系型数据库 MySQL
实现MySQL分页查询的三种方式~
实现MySQL分页查询的三种方式~
|
7月前
|
前端开发 关系型数据库 MySQL
MySQL分页查询方法及优化
MySQL分页查询方法及优化
169 0
|
关系型数据库 MySQL
MySQL 分页查询
MySQL 分页查询
46 0
|
SQL 关系型数据库 MySQL
MySQL条件查询,WHERE的用法
MySQL条件查询,WHERE的用法
212 0
|
安全 关系型数据库 MySQL
mysql条件查询
mysql条件查询
40 0
|
关系型数据库 MySQL
MySQL数据查询limit
MySQL数据查询limit
102 0
|
SQL 缓存 前端开发
使用MySQL实现分页查询
使用MySQL实现分页查询
439 0
|
SQL Oracle 关系型数据库
【MySQL学习笔记】系列六:MySQL中的排序与分页
【MySQL学习笔记】系列六:MySQL中的排序与分页
|
Java 关系型数据库 MySQL
JDBC与MySQL实现分页查询技术
JDBC与MySQL实现分页查询技术
638 0
JDBC与MySQL实现分页查询技术