mysql如何使用where进行过滤_mysql查询条件说明

2次阅读

where子句必须紧跟在select/update/delete后且在from之后,字符串需加引号、NULL用is null判断、混用and/or须加括号、in遇null失效、between为闭区间、like通配符影响索引、字段函数调用导致全表扫描。

mysql如何使用where进行过滤_mysql查询条件说明

WHERE 子句的基本写法和常见错误

WHERE 是 mysql 中执行行级过滤的核心,它必须紧跟在 SELECTUPDATEDELETE 语句之后(且在 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 NULLIS NOT NULL

多个条件组合:AND、OR、NOT 和括号优先级

实际查询中很少只用一个条件。ANDOR 控制逻辑关系,但它们有默认优先级:AND 优先于 OR。不加括号容易出意料结果,比如 WHERE status = 'active' OR type = 'vip' AND score > 80 实际等价于 WHERE status = 'active' OR (type = 'vip' AND score > 80),而不是你可能想表达的「活跃用户且(VIP 或高分)」。

实操建议:

  • 只要混用 ANDOR,一律显式加括号,比如: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) = 2024WHERE 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 处理这些细节会立刻暴露出来。

text=ZqhQzanResources