Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 310|回复: 0

[默认分类] Mysql中分页查询两个方法比较

[复制链接]
  • TA的每日心情
    开心
    2021-12-13 21:45
  • 签到天数: 15 天

    [LV.4]偶尔看看III

    发表于 2018-7-5 15:54:53 | 显示全部楼层 |阅读模式
    mysql中分页查询有两种方式, 一种是使用COUNT(*)的方式,具体代码如下

    1. SELECT COUNT(*) FROM foo WHERE b = 1;
    2. SELECT a FROM foo WHERE b = 1 LIMIT 100,10;
    复制代码
    复制代码

    另外一种是使用SQL_CALC_FOUND_ROWS

    1. SELECT SQL_CALC_FOUND_ROWS a FROM foo WHERE b = 1 LIMIT 100, 10;
    2. 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. CREATE TABLE IF NOT EXISTS `foo` (
    2. `a` int(10) unsigned NOT NULL AUTO_INCREMENT,
    3. `b` int(10) unsigned NOT NULL,
    4. `c` varchar(100) NOT NULL,
    5. PRIMARY KEY (`a`),
    6. KEY `bar` (`b`,`a`)
    7. ) ENGINE=MyISAM;
    复制代码



    注意下这里是使用b,a做了一个索引,所以查询select * 的时候是不会用到covering index的,select a才会使用到covering index
    1. <?php
    2. $host = "192.168.100.166";
    3. $dbName = "test";
    4. $user = "root";
    5. $password = "";
    6. $db = mysql_connect($host, $user, $password) or die("DB connect failed");
    7. mysql_select_db($dbName, $db);
    8. echo "==========================================" . "\r\n";
    9. $start = microtime(true);
    10. for ($i =0; $i<1000; $i++) {
    11.     mysql_query("SELECT SQL_NO_CACHE COUNT(*) FROM foo WHERE b = 1");
    12.     mysql_query("SELECT SQL_NO_CACHE a FROM foo WHERE b = 1 LIMIT 100,10");
    13. }
    14. $end = microtime(true);
    15. echo $end - $start . "\r\n";
    16. echo "==========================================" . "\r\n";
    17. $start = microtime(true);
    18. for ($i =0; $i<1000; $i++) {
    19.     mysql_query("SELECT SQL_NO_CACHE SQL_CALC_FOUND_ROWS a FROM foo WHERE b = 1 LIMIT 100, 10");
    20.     mysql_query("SELECT FOUND_ROWS()");
    21. }
    22. $end = microtime(true);
    23. echo $end - $start . "\r\n";
    24. echo "==========================================" . "\r\n";
    25. $start = microtime(true);
    26. for ($i =0; $i<1000; $i++) {
    27.     mysql_query("SELECT SQL_NO_CACHE COUNT(*) FROM foo WHERE b = 1");
    28.     mysql_query("SELECT SQL_NO_CACHE * FROM foo WHERE b = 1 LIMIT 100,10");
    29. }
    30. $end = microtime(true);
    31. echo $end - $start . "\r\n";
    32. echo "==========================================" . "\r\n";
    33. $start = microtime(true);
    34. for ($i =0; $i<1000; $i++) {
    35.     mysql_query("SELECT SQL_NO_CACHE SQL_CALC_FOUND_ROWS * FROM foo WHERE b = 1 LIMIT 100, 10");
    36.     mysql_query("SELECT FOUND_ROWS()");
    37. }
    38. $end = microtime(true);
    39. echo $end - $start . "\r\n";
    复制代码
    返回的结果:

    和老王里面文章说的是一样的。第四次查询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情况下后者性能高。所以使用的时候要注意这个。
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-6-11 04:09 , Processed in 0.505518 second(s), 46 queries .

    Powered by Discuz! X3.4

    © 2001-2017 Comsenz Inc.

    快速回复 返回顶部 返回列表