MySQL 中常用数值函数怎么写
MySQL 提供的数值函数大多直接作用于列或表达式,不需要额外声明类型,但必须注意参数是否为
NULL、是否支持浮点/整型混用。比如
ABS()对负数取绝对值,
ROUND(3.14159, 2)返回
3.14;而
ROUND(NULL)结果仍是
NULL,不是报错。
常见写法误区是把函数当变量用,例如写成
SELECT ABS col FROM t(漏括号),正确写法必须带括号:
SELECT ABS(col) FROM t。
ROUND、CEILING、FLOOR 的区别和精度陷阱
这三个函数都处理小数,但行为差异明显,容易在金额计算或分页逻辑中出错:
ROUND(x, d)四舍五入到
d位小数,
d可为负数(如
ROUND(1234.56, -2)→
1200)
CEILING(x)向上取整(不小于
x的最小整数),
CEILING(-1.2)→
-1
FLOOR(x)向下取整(不大于
x的最大整数),
FLOOR(-1.2)→
-2
特别注意:MySQL 的
ROUND()在 5 附近使用“银行家舍入”(偶数舍入),
ROUND(2.5)和
ROUND(3.5)都返回
4,不是传统四舍五入。若需严格四舍五入,得用
CAST(ROUND(x + 0.5 - SIGN(x)*0.5) AS SIGNED)这类绕过方式。
聚合场景下数值函数不能直接嵌套 COUNT
像
COUNT(ROUND(price))这种写法会报错,因为
COUNT()只接受列名、
*或表达式,但 MySQL 不允许在聚合函数内部再调用标量函数后再用于计数上下文(除非用子查询或派生表)。
正确做法分两步:
SELECT COUNT(*) FROM ( SELECT ROUND(price) AS rprice FROM orders WHERE price IS NOT NULL ) AS t;
或者改用条件统计:
COUNT(CASE WHEN price > 0 THEN 1 END)—— 这里
1是非空值,能被
COUNT统计。
MOD 和 % 运算符在负数时结果不一致
MySQL 中
MOD(a, b)和
a % b看似等价,但对负数处理不同:
MOD(-7, 3)→
2(遵循数学定义:
a - b * FLOOR(a/b))
-7 % 3→
-1(C 语言风格取余,符号跟随被除数)
如果做分组编号(如每 5 条一组),用
MOD(id, 5)更安全;若依赖符号一致性(比如判断奇偶),建议统一用
ABS(id) % 2避免负 ID 导致意外。
数值函数本身不慢,但嵌套过深或在
WHERE子句中对字段调用(如
WHERE ROUND(price) = 100)会导致索引失效。真正影响性能的,往往不是函数选错,而是没意识到它让条件无法走索引。
