Mysql优化实践(分页优化)

作者: 凤之恋 | 来源:发表于2016-08-23 22:04 被阅读2111次

当你和别人都能实现一个某个功能,这时候区分你们能力的不是谁干活多少,而是谁能写出效率更高的代码。比如显示一个订单列表它不仅仅是写一条SELECT SQL那么简单,我们还需要很清楚的知道这条SQL他大概扫描了多少行数据,返回了多少行数据,是否需要创建索引,创建什么样的索引,索引是否生效,等等。
这里以订单列表显示和订单导出为例来谈谈Mysql分页优化。

发现问题

下边是一个订单表的简单表结构。里边有大概270万条数据,其中渠道ID为35的有132万调数据。

CREATE TABLE IF NOT EXISTS `order_info` (
  `order_id` int(11) NOT NULL AUTO_INCREMENT COMMENT '订单ID',
  `order_sn` varchar(60) NOT NULL COMMENT '订单号',
  `user_id` int(11) NOT NULL COMMENT '用户ID',
  `channels_id` int(11) NOT NULL COMMENT '渠道ID',
  ……一些其他字段
  `order_time` datetime NOT NULL COMMENT '下单时间',  
  PRIMARY KEY (`order_id`),
  KEY `channels_id` (`channels_id`),
  KEY `order_sn` (`order_sn`),
  KEY `user_id` (`user_id`),
  KEY `order_time` (`order_time`)
) ENGINE=InnoDB  DEFAULT CHARSET=utf8;

一个订单列表页面一般很多人是这么写的。显示一个总数或者总页数,然后是上一页 1 2 3 4 5 下一页


而我们一般会这样写sql语句去实现上边的功能:

select count(1) as num from order_info where channels_id=35;    0.24 sec
select * from order_info where channels_id=35 order by order_id desc limit 0,20;    0.01 sec
select * from order_info where channels_id=35 order by order_id desc limit 1320000,20;  12.55 sec  即便是第二次查询也用了4.27 sec(mysql自身也会有查询缓存机制)

这里获取数据总数用了相当长的时间。随着你数据量的增多需要的时间也会更长。在获取第一页的数据的时候也没用多长时间,但是越往后需要的时间也就越长。
在多人操作尤其是大并发量的情况下,大量的数据被扫描造成系统IO和CPU资源消耗完,进而导致整个数据库不可服务。 而cpu 消耗过大通常情况下都是由于慢sql 造成的,这里的慢sql 包括全表扫描,扫描数据量过大,内存排序,磁盘排序,锁争用等待等; 表现现象为:sql 执行状态为:sending data,Copying to tmp table,Copying to tmp table on disk,Sorting result,locked;

如何优化

普通的limit M,N 的写法越往后查询越慢。因为mysql总是会去扫描M+N条数据来得到你想要的数据。

我们来看一下京东的分页




上边是京东的搜索和分页。京东的订单很明显根据时间维度做了分库或者分表,也可能根据用户维度又做了分库分表。京东没有显示总数,但是显示了页码 1 2 3 4 5

获取数据总数的优化

尽量不要去获取数据总数。如果业务确实需要获取当前搜索条件下的数据总数也建议使用ajax让用户点击按钮触发后获取总数,或者根据时间维度做数据的分表。大多数用户在点击订单列表的时候关心的不是订单总数,也不是很久之前的订单,而是最近一段时间下的订单。

获取数据的优化

下边我们利用索引只获取主键ID。用了0.40 sec,比上边的sql少了很多。

select order_id from order_info where channels_id=35 order by order_id desc limit 1320000,20;   0.40 sec

所以我们可以有这样的优化写法:

select * from order_info,(select order_id from order_info where channels_id=35 order by order_id desc limit 1320000,20) order_info_tmp where order_info.order_id = order_info_tmp.order_id; 0.47 sec

select * from order_info,(select order_id from order_info where channels_id=35 order by order_id desc limit 0,20) order_info_tmp where order_info.order_id = order_info_tmp.order_id; 0.00 sec

先查询翻页中需要的N条数据的主键id,然后根据主键id去查询你所需要的N条数据,此过程中查询N条数据的主键ID在索引中完成。

这里我们尽量只显示上一页或者下一页。那么如何去判断下一页是否有数据呢(没有数据的时候把下一页的按钮置灰)?参考laravel的简单分页设计。比如每页显示20条数据,而我显示当前页面的时候去获取21条数据,根据是否存在第21条数据来判断是否需要显示下一页。
如果需要显示页码1 2 3 4 5呢?其实也可以获取当前范围的几个页面的数据来判断,尽量减少扫描范围。


上边的方法虽然快了不少,可是依然扫描了很多的数据行,在数据量大的情况下依然会很慢,尤其是在做数据导出的时候。
比较常见的导出数据的应用场景就是用户输入搜索条件然后按照搜索条件导出数据。数据的导出不像列表页的显示。我们完全可以利用主键来操作。

select * from order_info where channels_id=35 AND order_id <=54388 order by order_id desc limit 20; 0.00 sec

我们主要是利用了主键ID,这里你可以看到即便是非常往后的数据也是很快的速度就能获取到。这样写能很大程度上减少表扫描的行数,减少数据查询的时间。

//auth by duxiaokong 2016-08-23
$fp = fopen('php://output', 'a');
$num_limit = 1000;
$order_id = 0;
$order_list = [];
while (true) {
    //执行sql  select * from order_info where $where AND order_id > $order_id order by order_id ASC limit $num_limit; 得到$order_list订单列表
    //这里一定要注意 order_id > $order_id 和 order_id ASC的排序
    if (empty($order_list)) {
        break;
    }
    $line = 0;
    $row_str = '';
    foreach ($order_list as $key => $val) {
        $order_id = $val['order_id']; //这行代码一定要记得赋值不然会造成死循环
        $line++;
        // 获取导出数据
        $row = [
            $val['order_sn'],
            $val['order_time'],
            $val['user_name']
            // ……
        ];
        //$row 过滤 $row中的非法字符
        $row_str .= mb_convert_encoding(implode(',', $row), 'gbk', 'utf-8') . PHP_EOL;
        //每获取20次记录写入一次数据库,减少IO
        if ($line >= 20) {
            fwrite($fp, $row_str);
            $line = 0;
            $row_str = '';
        }
    }
    if (!empty($row_str)) {
        fwrite($fp, $row_str);
        $line = 0;
        $row_str = '';
    }
}
fclose($fp);

总结:如何优化?最主要的原则就是避免数据量大时扫描过多的记录。


相关文章

  • MYSQL分页limit速度太慢优化方法

    MySQL 百万级分页优化(Mysql千万级快速分页)(转) MYSQL分页limit速度太慢优化方法 MYSQL...

  • Mysql优化实践(分页优化)

    当你和别人都能实现一个某个功能,这时候区分你们能力的不是谁干活多少,而是谁能写出效率更高的代码。比如显示一个订单列...

  • 千万级MySQL数据库建立索引,提高性能的秘诀

    实践中如何优化MySQL 实践中,MySQL的优化主要涉及SQL语句及索引的优化、数据表结构的优化、系统配置的优化...

  • 64MySQL-分页查询&表连接&count统计&索引优化总结

    1 Mysql 分页查询sql 执行原理? 2,千万级数据mysql 分页查询如何优化 3,Mysql表连接底层实...

  • MySQL性能优化之关联查询优化

    上一篇 <<

  • MySQL 分页优化

    随着偏移量 m 的增大,MySQL 需要花费大量时间来扫描需要丢弃的数据 优化方法:延迟关联,通过使用覆盖索引查询...

  • 优化MySQL分页

    之前写的SQL查询语句根本没有考虑到性能方面的问题,一是自己没有经验,二是因为网站规模小数据量不大,自然感受不到查...

  • Mysql 分页优化

    互联网公司拥有海量数据时,页面加载慢会直接导致用户体验的下降,用户体验决定着产品的未来。所有一个高效的分页变得尤为...

  • MySQL分页优化

    不需要担心数据库性能优化问题的日子已经一去不复返了。 随着时代的进步,随着野心勃勃的企业想要变成下一个 Faceb...

  • mysql分页优化

    Mysql limit分页语句用法 如何优化limit 当一个查询语句偏移量offset很大的时候,如select...

网友评论

  • weigs:学习了
  • 秦昊的手指:您说的这些开发中都有遇到类似的,除了框架那个我用的是CI框架,还有我想问下您对分库分表怎么看,什么情况下分库分表是好的,或者需要做好什么再去分库分表
    凤之恋:@周小然小周 分库分表是要看你的数据量和业务情况的,如果你的业务量三年内达不到一定数据量那就没必要分库分表过度设计。分析一下你的业务,评估一下你的数据量。另外还要根据实际业务,比如你做的是一个SaaS版系统,那也可以考虑分库分表
  • 735a341226a2:多谢分享:stuck_out_tongue:

本文标题:Mysql优化实践(分页优化)

本文链接:https://www.haomeiwen.com/subject/jdunsttx.html