咳咳,各位观众老爷们,晚上好!我是今晚的主讲人,江湖人称“SQL小霸王”(其实是自己封的)。今天给大家带来的是MySQL编程进阶系列之——SQL优化技巧:如何编写可读性强且执行高效的SQL语句。
咱们的目标是:写出像诗一样优雅,跑得像火箭一样快的SQL!
第一部分:SQL优化的大方向:让MySQL知道你要什么
SQL优化,说白了就是让MySQL的查询优化器更好地理解你的意图,然后选择最佳的执行计划。MySQL查询优化器也不是神仙,你写的SQL语句含糊不清,它也只能猜,猜错了自然效率就低了。所以,咱们要做的就是:
- 明确目标: 你想查什么?
- 提供线索: 如何高效地查到?
1.1 避免SELECT *,只取需要的列
这应该是老生常谈了,但还是有很多人犯这个错误。SELECT *
会读取所有列的数据,即使你只需要其中的几列。
-
坏例子:
SELECT * FROM users WHERE id = 1;
-
好例子:
SELECT id, username, email FROM users WHERE id = 1;
好处:
- 减少IO: 只需要读取需要的列,减少磁盘IO。
- 减少网络带宽: 减少数据传输量。
- 减少内存消耗: 减少服务器端和客户端的内存消耗。
- 索引覆盖: 如果查询只涉及到索引列,可以利用索引覆盖,避免回表查询(后面会讲到)。
1.2 善用WHERE条件,缩小查询范围
WHERE条件是SQL查询的核心,用好WHERE条件可以大大缩小查询范围,减少扫描的行数。
-
坏例子:
SELECT * FROM orders WHERE order_date BETWEEN '2023-01-01' AND '2023-12-31';
如果
order_date
没有索引,或者索引效果不好,这条语句会扫描全表。 -
好例子: (假设
order_date
有索引)SELECT * FROM orders WHERE order_date >= '2023-01-01' AND order_date <= '2023-12-31';
或者,如果可以确定具体的订单状态,加上状态过滤:
SELECT * FROM orders WHERE order_date >= '2023-01-01' AND order_date <= '2023-12-31' AND status = '已完成';
原则:
- 越精确越好: WHERE条件越精确,查询范围越小。
- 索引优先: 尽量使用索引列作为WHERE条件。
- 避免函数计算: 尽量避免在WHERE条件中使用函数计算,例如
WHERE YEAR(order_date) = 2023
,会导致索引失效。
1.3 避免使用!= 或 <> 或 NOT IN 或 NOT EXISTS
这些操作符通常会导致全表扫描,除非有非常好的理由,否则应该尽量避免使用。
-
坏例子:
SELECT * FROM users WHERE status != 'active';
-
好例子:
SELECT * FROM users WHERE status = 'inactive' OR status IS NULL;
如果
status