如何使用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