WHERE 子句的基本写法和常见错误
WHERE 是 MySQL 中执行行级过滤的核心,它必须紧跟在
SELECT、
UPDATE或
DELETE语句之后(且在
FROM之后),不能单独使用。最常见的错误是把字符串值漏掉引号,比如写成
WHERE name = zhangsan—— 这会让 MySQL 把
zhangsan当作列名或未定义变量,直接报错
Unknown column 'zhangsan' in 'where clause'。
正确写法必须加引号:
WHERE name = 'zhangsan'(单引号是标准,双引号在 SQL_MODE 允许下也可用,但不推荐)。 数字类型可不加引号:
WHERE id = 123日期建议用标准格式字符串:
WHERE created_at >= '2024-01-01'NULL 判断不能用
=,必须用
IS NULL或
IS NOT NULL
多个条件组合:AND、OR、NOT 和括号优先级
实际查询中很少只用一个条件。
AND和
OR控制逻辑关系,但它们有默认优先级:
AND优先于
OR。不加括号容易出意料结果,比如
WHERE status = 'active' OR type = 'vip' AND score > 80实际等价于
WHERE status = 'active' OR (type = 'vip' AND score > 80),而不是你可能想表达的「活跃用户且(VIP 或高分)」。
实操建议:
只要混用AND和
OR,一律显式加括号,比如:
WHERE status = 'active' AND (type = 'vip' OR score > 80)
NOT尽量作用于最小逻辑单元,如
NOT (a = 1 AND b = 2)比
NOT a = 1 AND NOT b = 2更易读且不易错 避免过度嵌套,复杂条件可考虑拆到应用层拼接,或用视图/CTE 简化
IN、BETWEEN、LIKE 这些“快捷条件”的坑
IN看似简单,但遇到
NULL值会静默失效:
WHERE id IN (1, 2, NULL)实际等价于
WHERE id IN (1, 2),因为
id = NULL永远返回
UNKNOWN,不参与匹配。
BETWEEN是闭区间,
WHERE age BETWEEN 18 AND 25包含 18 和 25,这点和某些编程语言的 range 习惯不同。
LIKE的通配符要小心索引失效:
name LIKE 'abc%'可走索引(前缀匹配)
name LIKE '%abc'或
name LIKE '%abc%'通常无法使用普通 B+Tree 索引,除非用了全文索引或倒排索引(如 MySQL 8.0+ 的 ngram) 如果字段是大小写敏感 collation,
LIKE 'ABC%'不会匹配小写数据;不确定时用
LOWER(name) LIKE LOWER('abc%'),但会进一步抑制索引
WHERE 中函数调用对性能的影响
在 WHERE 条件里对字段使用函数(如
WHERE YEAR(created_at) = 2024或
WHERE UPPER(name) = 'ZHANG')会导致该字段无法使用索引,MySQL 必须全表扫描计算每行函数值。
更高效的做法是让条件适配索引结构:
时间范围改用区间:WHERE created_at >= '2024-01-01' AND created_at大小写统一改在写入时处理,或建函数索引(MySQL 8.0.13+ 支持):
CREATE INDEX idx_name_upper ON users ((UPPER(name)))模糊搜索需求高,考虑引入 Elasticsearch 或用 MySQL 的
FULLTEXT索引 +
MATCH ... AGAINST
WHERE 看似简单,但每个字符都可能决定是毫秒还是秒级响应——尤其是当表超过百万行后,函数、隐式类型转换、NULL 处理这些细节会立刻暴露出来。
