mysql limit optimization
mysql limit optimization
Test data, mars_tianchi_user_actions (id, primary key index added)
Question: How to obtain numbers 10000001 to 10000005?
Use limit directly
mysql> select * from mars_tianchi_user_actions limit 10000000,5;
+----------+----------------------------------+--- -------------------------------+----------------+----- --------+----------+
| id | user_id | song_id | gmt_create | action_type | ds |
+----------+----------------------------------+--- -------------------------------+----------------+----- --------+----------+
| 10000001 | c4e89a01990bf87d745aa3b0cc6bd7aa | e5b475da38985734b6848130a31545a8 |
| 10000002 | f54c30493238f3f4dd79a4624fa20601 | f35da5a68294505c0e2f5dde6f8a2111 |
| 10000003 | f54c30493238f3f4dd79a4624fa20601 | ef776ef07e703d58c0af76c1031c0483 |
| 10000004 | d09af4bcf642b21fd13e11eb672e87f5 | 18b2cd447b72712a106bca7138187534 |
| 10000005 | d09af4bcf642b21fd13e11eb672e87f5 | 18b2cd447b72712a106bca7138187534 |
+----------+----------------------------------+--- -------------------------------+----------------+----- --------+----------+
5 rows in set (5.50 sec)
It took 5.50 seconds to view the explain
mysql> explain select * from mars_tianchi_user_actions limit 100000000,5;
+----+-------------+---------------------------+-- ----------+------+---------------+------+--------- +------+---------+
| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |
----+----------+
| 1 | SIMPLE | mars_tianchi_user_actions | NULL | ALL | NULL | NULL | NULL | NULL | 15763020 | 100.00 | NULL |
+----+-------------+---------------------------+-- ----------+------+---------------+------+--------- +------+---------+
1 row in set, 1 warning (0.00 sec)
Oh, it turns out to be a full table scan. How to use the index?
Connection query
mysql> select * from mars_tianchi_user_actions m , (select id from mars_tianchi_user_actions limit 10000000 ,5) t where m.id =t.id;
+----------+----------------------------------+--- -------------------------------+----------------+----- --------+----------+
| id | user_id | song_id | gmt_create | action_type | ds | id |
+----------+----------------------------------+--- -------------------------------+----------------+----- --------+----------+
| 10000001 | c4e89a01990bf87d745aa3b0cc6bd7aa | e5b475da38985734b6848130a31545a8 |
| 10000002 | f54c30493238f3f4dd79a4624fa20601 | f35da5a68294505c0e2f5dde6f8a2111 |
| 10000003 | f54c30493238f3f4dd79a4624fa20601 | ef776ef07e703d58c0af76c1031c0483 |
10000004 | d09af4bcf642b21fd13e11eb672e87f5 | 18b2cd447b72712a106bca7138187534 | 1427497200
10000005 d09af4bcf642b21fd13e11eb672e87f5 18b2cd447b72712a106bca7138187534
+----------+----------------------------------+--- -------------------------------+----------------+----- --------+----------+
5 rows in set (4.23 sec)
mysql> explain select * from mars_tianchi_user_actions m , (select id from mars_tianchi_user_actions limit 10000000 ,5) t where m.id =t.id;
+----+-------------+---------------------------+-- ----------+--------+---------------+---------+---- -----+----------+
| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |
+----+-------------+---------------------------+-- ----------+--------+---------------+---------+---- -----+----------+
| 1 | PRIMARY | <derived2> | NULL | ALL | NULL | NULL | NULL | NULL | 10000005 | 100.00 | NULL |
| 1 | PRIMARY | m | NULL | eq_ref | PRIMARY | PRIMARY | 8 | t.id | 1 | 100.00 | NULL |
| 2 | DERIVED | mars_tianchi_user_actions | NULL | index | NULL | PRIMARY | 8 | NULL | 15763020 | 100.00 | Using index |
+----+-------------+---------------------------+-- ----------+--------+---------------+---------+---- -----+----------+
3 rows in set, 1 warning (0.00 sec)
It took 4.23s, which is indeed much faster (thinking whether the derived table can be further optimized?)
Subquery
mysql> select * from mars_tianchi_user_actions m where m.id in (select id from mars_tianchi_user_actions limit 10000000,5);