1、问题

1 1POST domain/v1.0/module/method?order=condition&orderType=desc&offset=1800000&limit=500

domain、module 和 method 都是化名,代表接口的域、模块和实例方法名,后面的offset和limit代表 分页操作的偏移量每页的数量, 也就是说该用户是在 翻第(1800000/500+1=3601)页。初步捞了一下日志,发现 有8000多次这样调用。

这太神奇了,而且页面上的分页单页数量也不是500,而是 25条每页,这个绝对不是人为的在功能页面上进行一页一页的翻页操作,而是数据被刷了(说明下,生产环境数据有1亿+)。详细对比日志发现,很多分页的时间是重叠的,对方应该是多线程调用。

2、分析

对于MySQL查询语句来说,整体效率还是可以的,该有的联表查询优化都有,该简略的查询内容也有,关键条件字段和排序字段该有的索引也都在,问题在于他一页一页的分页去查询,查到越后面的页数,扫描到的数据越多,也就越慢。

我们在查看前几页的时候,发现速度非常快,比如 limit 200,25,瞬间就出来了。但是越往后,速度就越慢,特别是百万条之后,卡到不行,那这个是什么原理呢。先看一下我们翻页翻到后面时,查询的sql是怎样的:

  1. select * from t_name where c_name1='xxx'order by c_name2 limit 2000000,25;

这种查询的慢,其实是因为limit后面的偏移量太大导致的。 比如像上面的 limit 2000000,25 ,这个等同于数据库要扫描出 2000025条数据,然后再丢弃前面的 20000000条数据,返回剩下25条数据给用户,这种取法明显不合理。

三、MySQL高级分享-分页 - 图1

查询性能优化中,

分页操作通常会使用limit加上偏移量的办法实现,同时再加上合适的order by子句。但这会出现一个常见问题:当偏移量非常大的时候,它会导致MySQL扫描大量不需要的行然后再抛弃掉。

三、数据模拟

那好,了解了问题的原理,那就要试着解决它了。涉及数据敏感性,我们这边模拟一下这种情况,构造一些数据来做测试。

1、创建两个表:员工表和部门表

  1. /*部门表,存在则进行删除 */
  2. DROP TABLE ifEXISTS dep;
  3. CREATE TABLE dep (
  4. id INT UNSIGNED PRIMARY KEY auto_increment,
  5. depno MEDIUMINT UNSIGNED NOT NULL default0,
  6. depname VARCHAR ( 20 ) NOT NULL DEFAULT "",
  7. memo VARCHAR ( 200 ) NOT NULL DEFAULT ""
  8. );
  9. /*员工表,存在则进行删除*/
  10. DROP tableifEXISTS emp;
  11. CREATE TABLE emp (
  12. idintunsigned PRIMARY KEY auto_increment,
  13. empno MEDIUMINT UNSIGNED notnulldefault0,
  14. empname VARCHAR ( 20 ) notnulldefault "",
  15. job VARCHAR ( 9 ) notnulldefault "",
  16. mgr MEDIUMINT UNSIGNED notnulldefault0,
  17. hiredate datetime notnull,
  18. sal DECIMAL ( 7, 2 ) notnull,
  19. comn DECIMAL ( 7, 2 ) notnull,
  20. depno MEDIUMINT UNSIGNED notnulldefault0
  21. );

2、创建两个函数:生成随机字符串和随机编号

  1. /* 产生随机字符串的函数*/
  2. DELIMITER $
  3. DROP FUNCTION ifEXISTS rand_string;
  4. CREATE FUNCTION rand_string ( n INT ) RETURNS VARCHAR ( 255 ) BEGIN
  5. DECLARE
  6. chars_str VARCHAR ( 100 ) DEFAULT 'abcdefghijklmlopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';
  7. DECLARE
  8. return_str VARCHAR ( 255 ) DEFAULT '';
  9. DECLARE
  10. i INT DEFAULT0;
  11. WHILE
  12. i < n DOSET return_str = CONCAT( return_str, SUBSTRING( chars_str, FLOOR( 1+ RAND( ) * 52 ), 1 ) );
  13. SET i = i + 1;
  14. END WHILE;
  15. RETURN return_str;
  16. END $DELIMITER;
  17. /*产生随机部门编号的函数*/
  18. DELIMITER $
  19. DROP FUNCTION ifEXISTS rand_num;
  20. CREATE FUNCTION rand_num ( ) RETURNS INT ( 5 ) BEGIN
  21. DECLARE
  22. i INT DEFAULT0;
  23. SET i = FLOOR( 100+ RAND( ) * 10 );
  24. RETURN i;
  25. END $DELIMITER;

3、编写存储过程,模拟500W的员工数据

  1. /*建立存储过程:往emp表中插入数据*/
  2. DELIMITER $
  3. DROP PROCEDUREifEXISTS insert_emp;
  4. CREATE PROCEDURE insert_emp ( IN START INT ( 10 ), IN max_num INT ( 10 ) ) BEGINDECLARE i INT DEFAULT0;
  5. /*set autocommit =0 把autocommit设置成0,把默认提交关闭*/
  6. SET autocommit = 0;
  7. REPEATSET i = i + 1;
  8. INSERT INTO emp ( empno, empname, job, mgr, hiredate, sal, comn, depno )
  9. VALUES
  10. (
  11. ( START + i ),
  12. rand_string ( 6 ),
  13. 'SALEMAN',
  14. 0001,
  15. now( ),
  16. 2000,
  17. 400,
  18. rand_num ( )
  19. );
  20. UNTIL i = max_numEND
  21. REPEAT
  22. ;
  23. COMMIT;
  24. END $DELIMITER;
  25. /*插入500W条数据*/
  26. CALL insert_emp ( 0, 5000000 );

4、编写存储过程,模拟120的部门数据

  1. /*建立存储过程:往dep表中插入数据*/
  2. DELIMITER $
  3. DROP PROCEDUREifEXISTS insert_dept;
  4. CREATE PROCEDURE insert_dept ( IN START INT ( 10 ), IN max_num INT ( 10 ) ) BEGINDECLARE i INT DEFAULT0;
  5. SET autocommit = 0;
  6. REPEATSET i = i + 1;
  7. INSERT INTO dep ( depno, depname, memo )
  8. VALUES
  9. ( ( START + i ), rand_string ( 10 ), rand_string ( 8 ) );
  10. UNTIL i = max_numEND
  11. REPEAT
  12. ;
  13. COMMIT;
  14. END $DELIMITER;
  15. /*插入120条数据*/
  16. CALL insert_dept ( 1, 120 );

5、建立关键字段的索引

这边是跑完数据之后再建索引,会导致建索引耗时长,但是跑数据就会快一些。

  /*建立关键字段的索引:[排序]()、条件*/
  CREATE INDEX idx_emp_id ON emp(id);
  CREATE INDEX idx_emp_depno ON emp(depno);
  CREATE INDEX idx_dep_depno ON dep(depno);

四、测试

1、测试数据

  /*偏移量为100,取25*/
  SELECT a.empno,a.empname,a.job,a.sal,b.depno,b.depname 
  from emp a left join dep b on a.depno = b.depno 
  order by a.id desc limit 100,25;

  /*偏移量为4800000,取25*/
  SELECT a.empno,a.empname,a.job,a.sal,b.depno,b.depname 
  from emp a left join dep b on a.depno = b.depno 
  order by a.id desc limit 4800000,25;

2、执行结果

     [SQL]SELECT a.empno,a.empname,a.job,a.sal,b.depno,b.depnamefrom emp a left join dep b on a.depno = b.depno order by a.id desc limit 100,25;受影响的行:0时间:0.001s

     [SQL]SELECT a.empno,a.empname,a.job,a.sal,b.depno,b.depnamefrom emp a left join dep b on a.depno = b.depno order by a.id desc limit 4800000,25;受影响的行:0时间:12.275s

因为扫描的数据多,所以这个明显不是一个量级上的耗时。

五、解决方案

1、使用索引覆盖+子查询优化

因为我们有主键id,并且在上面建了索引,所以可以先在索引树中找到开始位置的 id值,再根据找到的id值查询行数据。

    /*子查询获取偏移100条的位置的id,在这个位置上往后取25*/
    SELECT a.empno,a.empname,a.job,a.sal,b.depno,b.depname
    from emp a left join dep b on a.depno = b.depno
    where a.id >= (select id from emp order by id limit100,1)
    order by a.id limit 25;

    /*子查询获取偏移4800000条的位置的id,在这个位置上往后取25*/
    SELECT a.empno,a.empname,a.job,a.sal,b.depno,b.depname
    from emp a left join dep b on a.depno = b.depno
    where a.id >= (select id from emp order by id limit4800000,1)
    order by a.id limit 25;

执行结果

执行效率相比之前有大幅的提升:

   [SQL]
   SELECT a.empno,a.empname,a.job,a.sal,b.depno,b.depname
   from emp a left join dep b on a.depno = b.depno
   where a.id >= (select id from emp order by id limit 100,1)
   order by a.id limit 25;受影响的行:0时间:0.106s

   [SQL]SELECT a.empno,a.empname,a.job,a.sal,b.depno,b.depname
   from emp a left join dep b on a.depno = b.depno
   where a.id >= (select id from emp order by id limit 4800000,1)
   order by a.id limit 25;受影响的行:0时间:1.541s

2、起始位置重定义

记住上次查找结果的主键位置,避免使用偏移量 offset。

  /*记住了上次的分页的最后一条数据的id是100,这边就直接跳过100,从101开始扫描表*/
  SELECT a.id,a.empno,a.empname,a.job,a.sal,b.depno,b.depname 
  from emp a left join dep b on a.depno = b.depno
  where a.id >100
  order by a.id limit 25;

  /*记住了上次的分页的最后一条数据的id是4800000,这边就直接跳过4800000,从4800001开始扫描表*/
  SELECT a.id,a.empno,a.empname,a.job,a.sal,b.depno,b.depname
  from emp a left join dep b on a.depno = b.depno
  where a.id > 4800000
  order by a.id limit 25;

执行结果

 [SQL]
 SELECT a.id,a.empno,a.empname,a.job,a.sal,b.depno,b.depname
 from emp a left join dep b on a.depno = b.depno
 where a.id >100
 order by a.id limit 25;受影响的行:0时间:0.001s

 [SQL]
 SELECT a.id,a.empno,a.empname,a.job,a.sal,b.depno,b.depname
 from emp a left join dep b on a.depno = b.depno
 where a.id > 4800000
 order by a.id limit 25;受影响的行:0时间:0.000s

这个效率是最好的,无论怎么分页,耗时基本都是一致的,因为他执行完条件之后,都只扫描了25条数据。

但是有个问题,只适合一页一页的分页,这样才能记住前一个分页的最后Id。如果用户跳着分页就有问题了,比如刚刚刷完第25页,马上跳到35页,数据就会不对。

这种的适合场景是类似百度搜索或者腾讯新闻那种滚轮往下拉,不断拉取不断加载的情况。这种延迟加载会保证数据不会跳跃着获取。

3、降级策略

还有一种方案参考:配置limit的偏移量和获取数一个最大值,超过这个最大值,就返回空数据。

因为他觉得超过这个值你已经不是在分页了,而是在刷数据了,如果确认要找数据,应该输入合适条件来缩小范围,而不是一页一页分页。

request的时候 如果offset大于某个数值就先返回一个4xx的错误。

六、小结

使用第三个方案,对offset做一下限流,超过某个值,就返回空值。第二天使用第一种和第二种配合使用的方案对程序和数据库脚本进一步做了优化。

合理来说做任何功能都应该考虑极端情况,设计容量都应该涵盖极端边界测试。

另外,该有的限流、降级也应该考虑进去。比如工具多线程调用,在短时间频率内8000次调用,可以使用计数服务判断并反馈用户调用过于频繁,直接给予断掉。