YEAR()和MONTH()函数仅适用于DATE/DATETIME/TIMESTAMP类型字段,不可用于字符串或NULL;MySQL、SQL Server原生支持,PostgreSQL需用EXTRACT(YEAR FROM ...),SQLite须用strftime('%Y', ...),且WHERE中直接使用会导致索引失效。
YEAR() 和 MONTH() 函数能直接提取年份和月份,但必须作用于合法的日期类型字段;对字符串或 NULL 值调用会返回 NULL 或报错,不是所有数据库都支持这两个函数。
MySQL、PostgreSQL(需用 EXTRACT(YEAR FROM ...))、SQL Server 都支持 YEAR() 和 MONTH(),但 SQLite 默认不支持——它要用 strftime('%Y', ...) 和 strftime('%m', ...)。PostgreSQL 更推荐用标准 SQL 的 EXTRACT(),因为 YEAR() 是它的非标准别名函数,某些版本可能未启用。
YEAR(order_date)、MONTH(order_date)
EXTRACT(YEAR FROM order_date)(返回 numeric 类型)strftime('%Y', order_date),注意返回的是字符串YEAR(order_date),也支持 DATEPART(YEAR, order_date)
如果 order_date 实际是 VARCHAR 存储(如 '2023-10-05'),MySQL 可能隐式转换成功,但 PostgreSQL 会直接报错 function year(unknown) does not exist。这时候必须先显式转类型:
SELECT YEAR(CAST('2023-10-05' AS DATE)); -- MySQL/SQL Server 可行 SELECT EXTRACT(YEAR FROM '2023-10-05'::DATE); -- PostgreSQL SELECT strftime('%Y', '2023-10-05'); -- SQLite
DESCRIBE orders;(MySQL)或 d orders(psql)STR_TO_DATE() 或正则清洗写 WHERE YEAR(created_at) = 2023 看似简洁,但数据库无法用上 created_at 字段的索引,全表扫描风险高。等价且高效写法是:
WHERE created_at >= '2023-01-01' AND created_at < '2024-01-01'
created_at >= '2023-10-01' AND created_at
MONTH() 同理真正麻烦的不是函数怎么写,而是字段类型是否可信、查询是否走索引、以及跨数据库语法差异带来的隐形兼容问题——这些在本地测试时往往看不出来,上线后才暴露。