当前位置: 首页 > 知识库问答 >
问题:

java - 如何理解Integer.parseInt源码?

薛高澹
2023-12-26

今天在研究Java中Integer.parseInt的源码时,对于int multmin = limit / radix;这一句代码不太理解,请教一下为什么通过 result < multmin可以判断是否越界?

参考资料:https://www.jianshu.com/p/da80a793dd57

public static int parseInt(String s, int radix)                throws NumberFormatException    {        /*         * WARNING: This method may be invoked early during VM initialization         * before IntegerCache is initialized. Care must be taken to not use         * the valueOf method.         */        if (s == null) {            throw new NumberFormatException("Cannot parse null string");        }        if (radix < Character.MIN_RADIX) {            throw new NumberFormatException("radix " + radix +                                            " less than Character.MIN_RADIX");        }        if (radix > Character.MAX_RADIX) {            throw new NumberFormatException("radix " + radix +                                            " greater than Character.MAX_RADIX");        }        boolean negative = false;        int i = 0, len = s.length();        int limit = -Integer.MAX_VALUE;        if (len > 0) {            char firstChar = s.charAt(0);            if (firstChar < '0') { // Possible leading "+" or "-"                if (firstChar == '-') {                    negative = true;                    limit = Integer.MIN_VALUE;                } else if (firstChar != '+') {                    throw NumberFormatException.forInputString(s, radix);                }                if (len == 1) { // Cannot have lone "+" or "-"                    throw NumberFormatException.forInputString(s, radix);                }                i++;            }            int multmin = limit / radix;            int result = 0;            while (i < len) {                // Accumulating negatively avoids surprises near MAX_VALUE                int digit = Character.digit(s.charAt(i++), radix);                if (digit < 0 || result < multmin) {                    throw NumberFormatException.forInputString(s, radix);                }                result *= radix;                if (result < limit + digit) {                    throw NumberFormatException.forInputString(s, radix);                }                result -= digit;            }            return negative ? result : -result;        } else {            throw NumberFormatException.forInputString(s, radix);        }    }

共有2个答案

洪凯定
2023-12-26

它在循环中计算的数值的负绝对值,就是 -|value| ,而不是 |value| 。也就是说,如果输入是 "3" ,那个求出的 result 是 -3 。最后通过 negative ? result : -result; 把正数的符号反过来。limit 是一个很小的负数。因为整形能表示的负数比正数多,有一个负数的绝对值在整数中是不能表示的。但是 -|value| 总是可以表示的。

于是,由于 mulmin = limit / radix,如果 result < multin ,那么 result * radix (如果忽略溢出的话) < limit 。就是说 result * radix 会溢出。

于是,溢出判断原理上应该是 result - digit < limit 。但是实际程序中,如果溢出,那么 result - digit 就溢出了,这个判断得不到期望的结果。那么就换了另一个等价的计算过程中不会溢出的判断:result < limit + digit 。

章子航
2023-12-26

int multmin = limit / radix;
上面这步是最小限制数值除以进制radix,得到multmin
如果result < multmin,那么在往下执行时,代码为result *= radix;
即result = result * radix因为result已经小于multmin,也就是
result<(limit / radix) * radix,也就是result < limit了,就发生越界了。

 类似资料:
  • 我无法理解与Scope方法的功能(实际上,我真的不知道RDD操作范围类的含义) 特别是,(正文:=)的含义是什么 您可以通过以下链接找到源代码:https://github.com/apache/spark/blob/master/core/src/main/scala/org/apache/spark/rdd/RDDOperationScope.scala 任何人都可以帮我吗?谢谢,我对此感到困

  • 问题内容: 我已经在其中一个视频教程中看到了下面的代码。它可以很好地执行,但是当我尝试在系统中执行时,它可以很好地编译,但是我遇到了运行时错误, 有人可以指导我这段代码有什么问题以及如何纠正吗? 提前致谢! 问题答案: __当您尝试访问数组中不存在的索引处的元素时,将发生 ArrayIndexOutOfBoundsException 。 例如:假设int a [] = {2,4,5,10,3}是一

  •   上下文管理器用于资源管理。它们允许你在需要时分配和释放资源。上下文管理器最常用和最受认可的例子是 with 语句。 它主要用于打开和关闭文件。with 允许在单行出现问题时打开和关闭文件。 它的主要优点是它可以确保文件正确关闭。 with open ('./data.txt','w') as f:     f.write("Hello")

  • 问题内容: 我有一个项目,我们经常在其中将String转换为int。当出现问题时(例如,不是数字,而是字母,等等),此方法将引发异常。但是,如果我必须到处都处理代码中的异常,这很快就会变得非常难看。我想将其放在一个方法中,但是,我不知道如何返回一个干净的值以表明转换出错。 在C ++中,我可以创建一个方法,该方法接受一个指向int的指针,并让该方法本身返回true或false。但是,据我所知,这在

  • 本文向大家介绍深入理解Java之HashMap源码剖析,包括了深入理解Java之HashMap源码剖析的使用技巧和注意事项,需要的朋友参考一下 一、HashMap概述 HashMap基于哈希表的 Map 接口的实现。此实现提供所有可选的映射操作,并允许使用 null 值和 null 键。(除了不同步和允许使用 null 之外,HashMap 类与 Hashtable 大致相同。)此类不保证映射的顺

  • 我明白为什么第三个和第四个输出打印真和假。这是因为返回一个对象,包装类缓存值在-128到127范围内的对象。如果向传递了该范围内的任何值,则它应该重用缓存中的对象。否则,它将创建一个新对象。 现在,为什么第二个输出打印出false?我以为返回一个原语,而不是像那样返回一个对象。