Integer.toString(int i)方法 - Java 8

1 篇文章 0 订阅

今天看到一道笔试题:String类中,使用哪个方法可以将int值100转换成字符串?写出具体方法:
挺简单的,学过Java基础应该都知道,String.valueOf(100)就能完成。今天就想着看看里面是啥。

参考(建议直接看链接3):

  1. Integer源码分析 - zxcvbnmzsedr - 简书
  2. java源码中Integer.class中有个getChars方法,里面有个52429是怎么确定的? - 知乎
  3. Java 源码学习系列(三)——Integer 写的真心不错,网站已收藏。

写了个代码测试一下:

public class Test {
    public static void main(String[] args) {
        System.out.println(String.valueOf(100));
    }
}

进入String.valueOf(int i)方法:

    public static String valueOf(int i) {
        return Integer.toString(i);
    }

String有点懒,啥都不干,直接调用Integer.toString(int i)方法。所以重点是Integer.toString(int i)方法,跟String.valueOf(int i)方法半毛钱关系没有。

进入Integer.toString(int i)方法:

    public static String toString(int i) {
        // 如果i是-2147483648,stringSize(-i)取反的时候会溢出,正数最大为2147483647,所以单独判断
        if (i == Integer.MIN_VALUE)
            return "-2147483648";
        // 获取i的长度。如果i是负数,负号也算一位,所以要加1
        int size = (i < 0) ? stringSize(-i) + 1 : stringSize(i);
        // 创建相应长度的字符数组,存储i
        char[] buf = new char[size];
        // 将i转换为字符数组,存储在buf中
        getChars(i, size, buf);
        // 调用String类的package构造方法,转为字符串,Integer和String都是java.lang包下的类
        return new String(buf, true);
    }

里面有:stringSize(int x),getChars(int i, int index, char[] buf)和String(char[] value, boolean share)三个方法。

Integer类的stringSize(int x)方法如下:

    final static int [] sizeTable = { 9, 99, 999, 9999, 99999, 999999, 9999999,
                                      99999999, 999999999, Integer.MAX_VALUE };
                                      
    // package静态方法
    // JDK规定x必须是非负数,取值范围是[0, 2147483647]
    static int stringSize(int x) {
        // 获取x的长度
        for (int i=0; ; i++)
            if (x <= sizeTable[i])
                return i+1;
    }

上面的方法没什么好说的,下面看看getChars方法(😂源码中上面这段代码和下面这段代码是邻居):

    // 将i转成字符数组存储在buf中
    static void getChars(int i, int index, char[] buf) {
        int q, r;
        int charPos = index;
        char sign = 0;// 符号

        // 如果是负数,保存负号,并取反
        if (i < 0) {
            sign = '-';
            i = -i;
        }

        // 每次迭代生成两位数,减少循环的次数
        while (i >= 65536) {
            q = i / 100;// 商
            // 等价于:r = i - (q * 100);
            // 100 = 2^6 + 2^5 + 2^2
            r = i - ((q << 6) + (q << 5) + (q << 2));
            i = q;// 更新i
            buf [--charPos] = DigitOnes[r];// 将个位赋值给数组
            buf [--charPos] = DigitTens[r];// 将十位赋值给数组
        }

        // 下降到快速模式为较小的数字,每次迭代生成一位数
        for (;;) {
            // 参考上面的while循环可知,这里应该是q = i / 10,具体分析见http://www.hollischuang.com/archives/1058
            q = (i * 52429) >>> (16+3);
            // 等价于:r = i-(q*10)
            // 10 = 2^3 + 2^1
            r = i - ((q << 3) + (q << 1));
            // 通过映射表(digits数组),将i对应的字符赋给buf数组
            buf [--charPos] = digits [r];
            i = q;// 更新i
            if (i == 0) break;
        }
        if (sign != 0) {
            buf [--charPos] = sign;// 负号
        }
    }

分析我不想写了。

关键点:
移位操作比乘法操作快
乘法操作比除法操作快

精确度,溢出等因素

String构造方法

    // String类的package构造方法,直接赋值比使用public的String构造方法(复制数组)快。
    String(char[] value, boolean share) {
        // assert share : "unshared not supported";
        this.value = value;
    }
  • 8
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值