0

0

详解Java中的Math.abs对于最小负数的特殊处理_潜在的Bug预警

P粉602998670

P粉602998670

发布时间:2026-03-07 11:48:10

|

804人浏览过

|

来源于php中文网

原创

math.abs(integer.min_value) 返回负数;因-2147483648的绝对值2147483648超出int范围(-2147483648~2147483647),补码溢出后仍为-2147483648。

详解java中的math.abs对于最小负数的特殊处理_潜在的bug预警

Math.abs(int) 遇到 Integer.MIN_VALUE 会返回负数

这是 Java 中少有「输入合法、调用正确、结果反直觉」的陷阱。当你对 Integer.MIN_VALUE(即 -2147483648)调用 Math.abs(),它不返回正数,而是原封不动返回 -2147483648

原因很简单:32 位补码中,Integer.MIN_VALUE 的绝对值是 2147483648,但超出了 int 的最大值 Integer.MAX_VALUE2147483647),溢出后回绕成负数。JDK 文档明确写了这个行为,但很多人没细读。

  • 常见错误现象:Math.abs(x) 竟然为 <code>true,导致后续逻辑(比如判断非负、取模、数组索引)崩掉
  • 典型使用场景:做哈希桶索引(Math.abs(x % capacity))、归一化坐标、校验输入范围
  • 别指望加个 (long) 强转就能躲开——Math.abs((long) Integer.MIN_VALUE) 是安全的,但如果你忘了转,还是踩坑

Math.abs(long) 同样不免疫 Long.MIN_VALUE

同理,Math.abs(long)Long.MIN_VALUE-9223372036854775808L)也返回负值。64 位补码下,它的绝对值无法用 long 表示。

这个问题比 int 版更隐蔽:因为 long 值通常来自计算或外部输入,开发者很少手动写 Long.MIN_VALUE,但一旦上游传入(比如解析 JSON 或数据库字段为 NULL 时默认填最小值),就可能触发。

光子AI
光子AI

AI电商服饰商拍平台

下载

立即学习Java免费学习笔记(深入)”;

  • 参数差异:Math.abs(int)Math.abs(long) 都未做溢出检查,语义上就是“按位取反加一”,不是“数学意义的绝对值”
  • 性能影响:无额外开销,但错误结果带来的修复成本远高于预防成本
  • 兼容性没问题——这行为从 JDK 1.0 就存在,改不了,也不该改

安全替代方案:用条件判断代替无脑调用

最直接的办法,就是别依赖 Math.abs() 的“绝对安全”假象。尤其在涉及索引、循环边界、比较逻辑的地方,显式处理极值。

  • int:用 x == Integer.MIN_VALUE ? 0 : Math.abs(x)(若业务允许映射为 0);或升级为 Math.abs((long) x) 再转回 int(注意二次溢出风险)
  • long:必须用 x == Long.MIN_VALUE ? 0L : Math.abs(x),或者直接用 BigInteger.valueOf(x).abs().intValue()(仅限低频路径)
  • 别用 x ——对 <code>Integer.MIN_VALUE-x 同样溢出,结果还是负数

容易被忽略的间接调用点

你以为只在自己写的 Math.abs() 里踩坑?错。很多工具方法底层偷偷用了它。

例如 Guava 的 Ints.constrainToRange(x, 0, 100)、Apache Commons Lang 的 NumberUtils.max(a, b)(当含负数时可能触发 abs)、甚至某些 JSON 库反序列化数字时做的规范化处理——都可能暗藏这个逻辑分支。

  • 排查方式:全局搜 Math\.abs,再顺藤摸瓜看哪些地方接收了用户可控输入
  • 测试要点:单元测试必须覆盖 Integer.MIN_VALUELong.MIN_VALUE,不能只测 -101
  • 最麻烦的是:这种 Bug 往往在线上低概率出现,日志里只显示“索引越界”或“非法参数”,根本看不出源头是 abs

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

453

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

546

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

331

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

82

2025.09.10

guava包作用
guava包作用

guava是一个java库,增强了java标准库,提供更有效率和易于使用的集合、实用程序、缓存和并发工具。想了解更多guava的相关内容,可以阅读本专题下面的文章。

271

2024.05.29

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

252

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1049

2024.03.01

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

970

2023.08.02

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

23

2026.03.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.2万人学习

C# 教程
C# 教程

共94课时 | 10.8万人学习

Java 教程
Java 教程

共578课时 | 78.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号