数据库
首页 > 数据库> > 如何使用join改进此MySQL查询?

如何使用join改进此MySQL查询?

作者:互联网

我有一个简单的查询,它花费了14秒钟以上.

select 
     e.title, e.date, v.name, v.city, v.region, v.country

from seminar e force index for join (venueid) 
     left join venues v on e.venueid = v.id 

where v.country = 'US'
     and v.city = 'New York' 
     and v.region = 'NY'
     and e.date > curdate() 
     and e.someid != 0

注意:count(e.id)代表用于调试目的的缩写.实际上,我们从这两个表中获取信息.

说明如下:

+----+-------------+-------+-------------+--------------------------------------------------------------------------------------+--------------------------+---------+-----------------+------+--------------------------------------------------------+
| id | select_type | table | type        | possible_keys                                                                        | key                      | key_len | ref             | rows | Extra                                                  |
+----+-------------+-------+-------------+--------------------------------------------------------------------------------------+--------------------------+---------+-----------------+------+--------------------------------------------------------+
|  1 | SIMPLE      | v     | index_merge | PRIMARY,city,country,region | city,region | 378,378 | NULL            |    2 | Using intersect(city,region); Using where |
|  1 | SIMPLE      | e     | ref         | venueid                     |  venueid    | 5       | v.id            |   11 | Using where                                            |
+----+-------------+-------+-------------+--------------------------------------------------------------------------------------+--------------------------+---------+-----------------+------+--------------------------------------------------------+

我在e.id,e.date,e.someid以及v.id,v.country,v.city和v.region上都有索引.

我知道数据库设置是一团糟,但这就是我现在必须处理的问题.

为什么SQL会花这么长时间,直到最后会有一个大约.数150?在赛事中,大约有100万个参赛作品,在场地中,则有10万个作品.

这两个表都是MyISAM.有什么想法可以改善这一点吗?

创建这样的索引后

create index location on venues (city, region, country)

需要20秒,解释是这样的:

+----+-------------+-------+------+--------------------------------------+--------------+---------+-------------------+------+------------------------------------+
| id | select_type | table | type | possible_keys                        | key          | key_len | ref               | rows | Extra                              |
+----+-------------+-------+------+--------------------------------------+--------------+---------+-------------------+------+------------------------------------+
|  1 | SIMPLE      | v     | ref  | PRIMARY,city,country,region,location | location     | 765     | const,const,const |  410 | Using index condition; Using where |
|  1 | SIMPLE      | e     | ref  | EventVenueID                         | venueid      | 5       | v.id              |   11 | Using where                        |
+----+-------------+-------+------+--------------------------------------+--------------+---------+-------------------+------+------------------------------------+

解决方法:

您已经离开了联接场所,但是在联接场所行的where子句中有条件,因此将仅返回联接行.但是,这是一个附带问题-请继续阅读,以了解为什么根本不需要联接.

接下来,如果城市是温哥华,则无需测试国家或州.

最后,如果您要查找“温哥华未来有多少事件”,则不需要加入,因为场地ID是一个常数!

尝试这个:

select count(*) as event_count
from events
where venueid = (select id from venues where city = 'vancouver')
and startdate > curdate() 
and te_id != 0

Mysql无需使用提示即可在会场ID上使用索引.如果没有,执行以下命令:

analyze events

这将更新索引列中数据分布的统计信息.请注意,如果您的许多活动都在温哥华,那么不使用索引会更有效(因为无论如何大多数行都必须被访问).

标签:query-performance,mysql
来源: https://codeday.me/bug/20191011/1896154.html