Java 中的这个绝对值有点不绝对啊!
现象
假如有如下代码定义了一个方法 test(),它入参可以任何一个 int 类型的整数,那么它输出结果可能是什么?
public class Test {
    public static void test(int a) {
        System.out.println("The result of absolute value compare to zero is:" + (Math.abs(a) >= 0));
    }
}
如果你的结论是 true,那么恭喜你,你掉入到绝对值不绝对的坑里面了。这个方法输出的结果有可能是 true,但是也有可能是 false。比如下面的调用代码将分别输出 true,true,true , false,如下图所示:
public static void main(String[] args) {
	test(1);
	test(-1);
	test(Integer.MAX_VALUE);
	test(Integer.MIN_VALUE);
}

在 Java 中,通过 Math.abs() 函数返回的值有的时候并不是这个数的绝对值。如下面的代码所示:
public static void main(String[] args) {
	System.out.println(Math.abs(Integer.MIN_VALUE));
}
上面的代码输出的结果并不是 Integer.MIN_VALUE 的绝对值,输出的结果是它自己,如下图所示:

从输出可以看到因为 Math.abs(Integer.MIN_VALUE) 的结果还是 Integer.MIN_VALUE,因此它是小于 0 的,这个也解释了上面判断大于等于 0 结果有可能输出的是 false。
为什么  Math.abs(Integer.MIN_VALUE) 的结果还是 Integer.MIN_VALUE 而不是它的绝对值呢?
原理
从 int 类型可以表示的数的范围解释是比较好理解的。以 int 类型为例,它能够表示的范围是 2^31 到 2^31 - 1。即 -2147483648 到 2147483647,可以看到最小的负数是 -2147483648 。它的绝对值实际上应该是 2147483648,但是这个值已经超过了 int 类型能够表示的最大的数 2147483647 了。如果返回 2147483648,它是不能在一个 int 类型的数中表示的。如果我们直接把 2147483648 这个数赋值给一个 int 类型的变量,编译器也会提示 Integer number too large,如下图所示:

因此这里 Math.abs() 函数返回的结果并不能是 214748364,因为 int 类型根本表示不了这个数。
那 Math.abs() 方法做了什么操作呢?查看 Math.abs() 方法的源码,实现逻辑如下:

在方法中就是判断了一下这个数是否小于 0,如果小于 0 的话,就返回对这个数取反后的值。那这个取反操作具体做了什么事情呢?为什么对 Integer.MIN_VALUE 进行了求反操作返回的还是它自己?
要回答这些问题,那就得知道计算机底层是表示一个整数的方式以及 int 类型表示的数的范围是 2^31 到 2^31 - 1 的原因。
Java语言规范中对此做了描述,规范中说到 Java 语言中使用 「two's-complement representation」 来表示整数,因为 「two's-complement representation」 的值不是对称的,所以对 int 或者 long 类型的最小值的取反的结果还是它们自身,在这个场景是有「溢出」发生的。而对一个整数的取反操作相当于把它的所有比特位取反,然后再加上 1。如下图所示:

规范这里的提到的 「two's-complement」 就是我们常常说的「补码」,学过计算机组成原理相关课程的应该对这个词语比较耳熟。
补码就是将二进制位的最高位作为符号位,它的权重是 -2^(w -1) (这里的 w 为比特位的个数) ,如果它设置为 1 表示负数,如果设置为 0,表示非负数。如下图所示:



根据补码的定义来看,补码能够表示的最大的数是 2^(w -1) - 1,而它能够表示的最小的的数是 -2^(w -1) (这里的 w 为比特位的个数)。那么最小数的绝对值是比最大数的绝对值还要大 1 的。从上面的图也可以看出(上图中的比特位数为 4),数轴最左侧的刻度是 -8,而数轴最右侧的刻度是 7。
对于补码的取反操作是把每个比特位都取反,然后加上 1。为什么补码的取反要这样操作?从数学的角度上讲一个数 x 加上它的取反 -x 的结果应该是 0。从计算机的角度我们可以知道 x 加上 x 的每个比特位取反的结果是每个比特位都是 1,按照补码的表示方式就是 10 进制的 -1,然后再加上 1 那就是 0,这样的结果就和数学上是相符合的了。比如假设总的比特位数是 8,1 的补码是 0000_0001,取反之后就是 1111_1110,相加的结果是 1111_1111,即 -1,然后加上 1 就是 0000_0000,即 0。如下图所示:

补码的英文名字「 two's complement」 这个名字的由来是如果把一个数的补码和它取反的补码得到的二进制都看作是无符号数的话,它相加的结果就是 2^w (这里的 w 为比特位的个数)。如下图所示:

回过头来看 Java 中的 int 类型,它的最小值 Integer.MIN_VALUE 的补码表示形式就是 1000_0000_0000_0000_0000_0000_0000_0000,按照补码取反操作的规则,应该是把它的补码按位取反得到 0111_1111_1111_1111_1111_1111_1111_1111,然后加 1,得到的结果还是 1000_0000_0000_0000_0000_0000_0000_0000,即它自己。因此在 Math.abs() 函数中对 Integer.MIN_VALUE 取反后得到的值仍然是 Integer.MIN_VALUE。
解决方法
对于 Integer.MIN_VALUE 的绝对值溢出现象的解决方法有以下几种:
一种是使用 Math.absExact() 方法,该方法在获取绝对值之前会判断是否超过了表示范围,如果超过了表示范围会抛出一个异常,如下图所示:
public class Test {
    public static void main(String[] args) {
        System.out.println(Math.absExact(Integer.MIN_VALUE));
    }
}

实现原理其实就是在进入方法时判断了一下是否是 Integer.MIN_VALUE ,如果是就直接抛出异常了,如下图所示:

也可以转为 long 类型后再获取绝对值,因为 Integer.MIN_VALUE 的绝对值是可以用 long 类型来表示的,因此转为 long 类型来获取绝对值也是可以的,但是这个方法就解决不了 Long.MIN_VALUE 绝对值溢出现象。如下图所示:
public static void main(String[] args) {
    int a = Integer.MIN_VALUE;
    System.out.println(Math.abs((long) a) >= 0);
}

还有一种可以使用 Integer.MIN_VALUE 构造一个 BigInteger 对象,然后通过获取这个对象的绝对值来和 BigDecimal.ZERO 来比较,这种方式不仅可以解决 Integer.MIN_VALUE 的绝对值溢出问题,还可以解决 Long.MIN_VALUE 的绝对值溢出问题。如下面的代码:
public static void main(String[] args) {
    BigInteger minInt = BigInteger.valueOf(Integer.MIN_VALUE);
    System.out.println(minInt.abs().compareTo(BigInteger.ZERO) >= 0);  
    BigInteger minLong = BigInteger.valueOf(Long.MIN_VALUE);
    System.out.println(minLong.abs().compareTo(BigInteger.ZERO) >= 0);
}

参考
Unary Minus Operator
Sign–magnitude
Ones' complement
Two's_complement
二进制—原码、反码、补码
Computer Organization and Design
Computer Systems: A Programmer's Perspective
Java 中的这个绝对值有点不绝对啊!的更多相关文章
- java中取整数绝对值_Java之——位运算求整数绝对值通过下面的位运算可以得到一个整数的绝对值
		public int abs( int a ) {return (a + (a >> 31)) ^ (a >> 31) ;//前半部分-1或+0,后半部分取反 } a为正数的情 ... 
- 在Java中,负数的绝对值竟然不一定是正数!!!
		绝对值是指一个数在数轴上所对应点到原点的距离,所以,在数学领域,正数的绝对值是这个数本身,负数的绝对值应该是他的相反数. 这几乎是每个人都知道的. 在Java中,想要获得有个数字的绝对值,可以使用ja ... 
- Java中的数是用补码表示的检验
		一.基本介绍(关于下列五个定义来自http://www.cnblogs.com/zhangziqiu/archive/2011/03/30/ComputerCode.html#!comments,谢原 ... 
- JAVA中分为基本数据类型及引用数据类型
		一.基本数据类型: byte:Java中最小的数据类型,在内存中占8位(bit),即1个字节,取值范围-128~127,默认值0 short:短整型,在内存中占16位,即2个字节,取值范围-32768 ... 
- Java中的大数处理类BigInteger和BigDecimar浅析
		这两个类位于java.math包内,要使用它们必须在类前面引用该包:import java.math.BigInteger;和import java.math.BigDecimal; BigInteg ... 
- java中的二进制
		(1)按位与运算 & 1 & 1 = 1, 0 & 1 = 0 51 & 5 即 0011 0011 & 0000 0101 =0000 0001 = 1 ... 
- 位运算及在java中的应用整理
		计算机编码: 原码 符号位为0表示正数,为1表示负数: 其余各位等同于真值的绝对值. 如:0000 0000 0000 0010 =2,1000 0000 0000 0010 =-2 反码 符号位的用 ... 
- Java中的二进制及基本的位运算
		Java中的二进制及基本的位运算 二进制是计算技术中广泛采用的一种数制.二进制数据是用0和1两个数码来表示的数.它的基数为2,进位规则是"逢二进一",借位规则是"借一当二 ... 
- 浅谈TreeMap以及在java中的使用
		treemap结构是红黑树 1.先介绍一下平衡二叉树 其特点是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树.也就是说该二叉树的任何一个子节点,其左右子树的高度 ... 
- Java基础学习(五)-- Java中常用的工具类、枚举、Java中的单例模式之详解
		Java中的常用类 1.Math : 位于java.lang包中 (1)Math.PI:返回一个最接近圆周率的 (2)Math.abs(-10):返回一个数的绝对值 (3)Math.cbrt(27): ... 
随机推荐
- 16. VUE怎么阻止冒泡
			给事件添加 stop 修饰符 ,比如 click.stop ; 补充: 阻止默认行为 prevent 修饰符 ,超链接的跳转,表单的默认提交 : once 修饰符 事件只触发一次 ps:事件修饰符可 ... 
- 云原生爱好者周刊:利用 DNS 计算圆周率
			开源项目推荐 dns.toys dns.toys 是一个比较有创意的 DNS 服务器,它利用 DNS 协议提供了很多非常有趣的功能和服务.例如查询时间.天气.圆周率.单位换算等等. Submarine ... 
- P2540 [NOIP2015 提高组] 斗地主 加强版
			简要题意 给你一副手牌,求最少的次数出完所有手牌.(按照它给出的规定出) 题目 分析 因为求最小次数直接贪心很明显是错的,但又直接写不出 \(dp\) 的式子,所以我们只能够爆搜所有情况,但这样明显会 ... 
- 使用-solidity-开发第一个-以太坊智能合约
			目录 目录 使用 solidity 开发第一个 以太坊智能合约 前言 项目源代码 最终效果 环境搭建 智能合约内容 Truffle 创建项目 Truffle 编码 Truffle 打包 Truffle ... 
- 2024御网线上Pwn方向题解
			ASM Checksec检查保护 基本上保护都关闭了 64位ida逆向 程序只有一段,并且返回地址就是输入的数据,看起来就是srop了,找一下可以用的gadget 通过异或清空rax值,然后通过异或e ... 
- 一文彻底弄懂Spring IOC 依赖注入
			Spring IOC(Inversion of Control,控制反转)依赖注入是 Spring 框架的核心特性之一,旨在实现对象之间的松耦合,提升代码的可维护性.可测试性和可扩展性.下面我们将从以 ... 
- 18.Kubernetes容器交付介绍
			Kubernetes容器交付介绍 如何在k8s集群中部署Java项目 容器交付流程 开发代码阶段 编写代码 编写Dockerfile[打镜像做准备] 持续交付/集成 代码编译打包 制作镜像 上传镜像仓 ... 
- 盘点Air780E的FTP应用,你了解吗?
			 一.FTP 概述 FTP(File Transfer Protocol,文件传输协议) 是 TCP/IP 协议组中的协议之一. FTP协议包括两个组成部分,其一为FTP服务器,其二为FTP客户端. ... 
- 【一步步开发AI运动小程序】十二、自定义一个运动分析器,实现计时计数01
			随着人工智能技术的不断发展,阿里体育等IT大厂,推出的"乐动力"."天天跳绳"AI运动APP,让云上运动会.线上运动会.健身打卡.AI体育指导等概念空前火热.那 ... 
- 从零开始学机器学习——入门NLP
			首先给大家介绍一个很好用的学习地址:https://cloudstudio.net/columns 今天我们将深入探讨自然语言处理(Natural Language Processing, NLP)这 ... 
