登录
首页 >  文章 >  java教程

Java中除零外,大数运算也可能抛出ArithmeticException吗?

时间:2026-03-21 10:19:32 359浏览 收藏

Java中的ArithmeticException远不止除零这么简单——它还广泛出现在整数精确运算溢出(如Math.addExact)、BigDecimal除法精度不足、BigInteger负模或非法除零等各类边界场景中,不同数值类型对“算术错误”的定义差异巨大,稍不留意就会在生产环境突然崩溃;真正危险的不是异常本身,而是开发者惯性假设“不会溢出”或“除数肯定安全”,而忽略了每种类型严格的校验逻辑和文档规定的触发条件。

Java的ArithmeticException除了除以零还有什么情况_大数运算精确度异常

ArithmeticException 不只是除零,大数溢出也会抛

Java 的 ArithmeticException 确实最常见于整数除零(/%),但另一个高频触发点是整数算术溢出——前提是用了 Math.addExact()Math.multiplyExact() 这类“精确运算”方法。它们不依赖 JVM 默认的静默溢出行为,而是主动检查并抛异常。

  • 普通 int a = Integer.MAX_VALUE + 1; 不会抛异常,结果是 Integer.MIN_VALUE(回绕)
  • Math.addExact(Integer.MAX_VALUE, 1) 会立刻抛 ArithmeticException: integer overflow
  • 同理,Math.multiplyExact(100000, 100000) 在 int 范围内就可能溢出,也抛这个异常
  • long 类型同理,对应方法是 Math.addExact(long, long)

BigDecimal 用错 scale 或 roundingMode 会抛 ArithmeticException

很多人以为 BigDecimal 是“绝对安全”的大数类型,其实它在除法(divide())时极易因精度问题抛 ArithmeticException,错误信息通常是 Non-terminating decimal expansion; no exact representable decimal result

  • 直接调用 bigDec1.divide(bigDec2)(无参数)时,如果除不尽且无法用有限小数表示(如 1/3),就会抛异常
  • 必须显式指定 scaleRoundingMode:例如 bigDec1.divide(bigDec2, 10, RoundingMode.HALF_UP)
  • 漏掉 RoundingMode 参数(只传 scale)仍会抛异常,因为重载方法要求二者同时存在
  • divideToIntegralValue() 可避免该异常,但它只返回商的整数部分,丢弃余数

BigInteger 的 divide() 除零和负模运算陷阱

BigInteger 虽然不会整数溢出,但它的 divide()remainder() 方法对边界条件很敏感,稍不注意就抛 ArithmeticException

  • new BigInteger("0").divide(new BigInteger("1")) 没问题;但 new BigInteger("1").divide(BigInteger.ZERO) 抛异常 —— 除零判断是基于值,不是引用
  • BigInteger.remainder() 要求除数非零,否则抛 ArithmeticException: BigInteger divide by zero
  • 更隐蔽的是:BigInteger.mod() 要求除数为正数,若传入负数(如 mod(new BigInteger("-5"))),会抛 ArithmeticException: BigInteger: modulus not positive
  • 想做带符号取模,得用 remainder(),但要注意它不保证结果非负(remainder 符号跟随被除数)

自定义数值类或封装工具时容易忽略的校验点

如果你封装了类似 SafeIntPreciseDecimal 工具类,很容易只防除零,却漏掉其他触发路径。

  • 调用 Math.*Exact() 前没做输入范围预判,导致生产环境突然崩溃
  • 包装 BigDecimaldivide 方法时,把 RoundingMode 写死成 UNNECESSARY,而没意识到这仅适用于能整除的场景
  • BigInteger.mod() 做哈希或索引计算时,误把用户可控的参数直接传入,结果对方传负数就炸了
  • 日志里只打印 e.toString(),看不出是除零还是溢出还是模非法,建议至少打 e.getMessage() 并检查异常类型分支

真正麻烦的不是异常本身,而是不同数值类型(int/long/BigInteger/BigDecimal)对“算术错误”的定义完全不同,同一个业务逻辑换种类型实现,异常触发条件就变了。别假设“不会溢出”或“除数肯定正”,每处数值操作都得按实际类型查文档确认边界行为。

今天关于《Java中除零外,大数运算也可能抛出ArithmeticException吗?》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>