当你和别人都能实现一个某个功能,这时候区分你们能力的不是谁干活多少,而是谁能写出效率更高的代码。比如显示一个订单列表它不仅仅是写一条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优化实践(分页优化)的更多相关文章

  1. MySQL 百万级分页优化

    MySQL 百万级分页优化 http://www.jb51.net/article/31868.htm 一般刚开始学SQL的时候,会这样写 : , ; 但在数据达到百万级的时候,这样写会慢死 : , ...

  2. SQL通用优化方案(where优化、索引优化、分页优化、事务优化、临时表优化)

    SQL通用优化方案:1. 使用参数化查询:防止SQL注入,预编译SQL命令提高效率2. 去掉不必要的查询和搜索字段:其实在项目的实际应用中,很多查询条件是可有可无的,能从源头上避免的多余功能尽量砍掉, ...

  3. mysql百万级分页优化

    普通分页 数据分页在网页中十分多见,分页一般都是limit start,offset,然后根据页码page计算start , 这种分页在几十万的时候分页效率就会比较低了,MySQL需要从头开始一直往后 ...

  4. Mysql大范围分页优化案例

    在BBS线上业务抓到如下分页SQL: meizu_bbs meizu_bbs Query Sending data , meizu_bbs meizu_bbs Query Sending data , ...

  5. MySQL 百万级分页优化(Mysql千万级快速分页)(转)

    http://www.jb51.net/article/31868.htm 以下分享一点我的经验 一般刚开始学SQL的时候,会这样写 复制代码 代码如下: SELECT * FROM table OR ...

  6. MySQL 百万级分页优化(Mysql千万级快速分页)

    以下分享一点我的经验 一般刚开始学SQL的时候,会这样写 : SELECT * FROM table ORDER BY id LIMIT 1000, 10; 但在数据达到百万级的时候,这样写会慢死 : ...

  7. mysql 大数据分页优化

    一.mysql大数据量使用limit分页,随着页码的增大,查询效率越低下. 1.   直接用limit start, count分页语句, 也是我程序中用的方法: select * from prod ...

  8. 在MySQL中如何使用覆盖索引优化limit分页查询

    背景 今年3月份时候,线上发生一次大事故.公司主要后端服务器发生宕机,所有接口超时.宕机半小时后,又自动恢复正常.但是过了2小时,又再次发生宕机. 通过接口日志,发现MySQL数据库无法响应服务器.在 ...

  9. MySQL分页优化中的“INNER JOIN方式优化分页算法”到底在什么情况下会生效?

    本文出处:http://www.cnblogs.com/wy123/p/7003157.html 最近无意间看到一个MySQL分页优化的测试案例,并没有非常具体地说明测试场景的情况下,给出了一种经典的 ...

随机推荐

  1. Silverlight & Blend动画设计系列九:动画(Animation)与视图状态管理(Visual State Manager)

    Silverlight中的动画(Animation)与视图状态管理(Visual State Manager) 结合使用是非常常见的,动画用于管理对象在某段事件段内执行的动画动作,视图状态管理则用于控 ...

  2. MVC 下拉框联动效果(单选)

    下拉框联动效果,我们以部门--职位为例,选择部门时,关联到该部门的职位.下拉框的写法就不多说了,详细请参照前文. 视图: 其中,dept是部门的属性,deptlist是部门下拉框的属性,job是职位的 ...

  3. jQuery实现单击和鼠标感应事件。

    1.实现单击事件动态交替http://www.cnblogs.com/ahthw/p/4232837.html讲到了toggleClass(),对于单击事件而言,jQuery同样提供了动态交替的tog ...

  4. Jquery判断checkbox选中状态

    jQuery v3.3.1 <input type="checkbox" id="ch"> 判断 $('#ch').is(':checked'); ...

  5. 五、spring之DI循环依赖

    什么是循环依赖 循环依赖就是循环引用,就是两个或多个Bean相互之间的持有对方,比如CircleA引用CircleB,CircleB引用CircleC,CircleC引用CircleA,则它们最终反映 ...

  6. MySQL在DOS界面对database和table增删改查

    昨天新接触MySQL,学习了一些内容,今天过来复习一下.(吐槽一下:安装个MySQL耗费老子半天时间!!) 学习了一下,大概知道了对数据库基本的增删改查,增add,删drop,改alter,查show ...

  7. java 中国网建实现发送短信验证码

    现在中国网建上注册一个自己的账户, 然后里面有代码案例,也有相应的下载jar包的地址 package com.direct.note; import java.io.IOException; impo ...

  8. sass(@at-root与&配合使用、without和with)

    @at-root与&配合使用(找父级) scss.style css.style 应用于@keyframe scss.style css.style @at-root (without: .. ...

  9. VMWARE错误-"VirtualInfrastructure.Utils.ClientsXml"的类型初始值设定项引发异常

    异常信息:"VirtualInfrastructure.Utils.ClientsXml"的类型初始值设定项引发异常. 解决方案:以管理员的身份运行客户程序

  10. MongoDB 创建集合

    createCollection() 方法 MongoDB db.createCollection(name, options) 是用来创建集合. 语法: 基本的 createCollection() ...