我所用到的表有两张,分别是 A表和B表。
A表
guess_id user_id
1 DD
2 CC
3 SS
B表
b_id guess_id b_name
1 1 WW
2 1 XX
3 2 RR
A表和B表的 字段的字符集等,都是相等的还有字段,排除是因为字符集或者长度等使用不了索引的关系。
guess_id varchar 64 utf8mb4 utf8mb4_bin
user_id varchar 64 utf8mb4 utf8mb4_bin
A表的 guess_id 和 B表的 guess_id 都加了索引。
名 栏位 索引类型
guess_id guess_id Normal
但是
EXPLAIN select
bgo.amount amount
from block_guess bg INNER JOIN block_guess_option bgo on bg.guess_id = bgo.guess_id
where bg.user_id = '2550' and bg.type != 'COMMON' and DATEDIFF(NOW(),bg.CREATE_TIME) = 1
order by bgo.amount desc
limit 5
按照上面的sql 执行之后
然后换做 这样的sql 只是把 inner join 换成了 left join
EXPLAIN select
bgo.amount amount
from block_guess bg LEFT JOIN block_guess_option bgo on bg.guess_id = bgo.guess_id
where bg.user_id = '2550' and bg.type != 'COMMON' and DATEDIFF(NOW(),bg.CREATE_TIME) = 1
order by bgo.amount desc
limit 5
执行之后如图:
这样就可以用到索引了。不用全盘扫描了。初步总结,是因为表的关系的一对多,用inner join 是逐一匹配,匹配的永远是一对一,而left join 则匹配的是一对多。我是这样理解的。如果有更好的解释可以在下面告诉我。谢谢!