Integer 包装类源码实现与分析

回溯

关注

阅读 151

2022-08-06

这是《水煮 JDK 源码》系列 的第8篇文章,计划撰写100篇关于JDK源码相关的文章

在 Java 语言中,默认定义的整数类型为 int 类型,默认定义的浮点类型为 double 类型,如果定义的数值在基本类型所允许的范围内,那么在定义时就不需要显式申明,直接定义即可,如下所示:

// 下面的定义都是合法的
int intNum = 2;
// byte 的范围为 -128 到 127,所以 2 在范围内
byte byteNum = 2;
// short 的范围为 -32768 到 32767,所以 2 在范围内
short shortNum = 2;
// long 的范围为 -2的63次方 到 2的63次方-1
long longNum = 2;
// float 的范围为 1.4e-45f 到 3.4028235e+38f
float floatNum = 2;
// double 的范围为 4.9e-324 到 1.7976931348623157e+308
double doubleNum = 2;

// 下面的定义是不合法的,数值都超过了类型所允许的最大值
// 这种情况需要选择合适的数据类型
byte byteNum2 = 200;
byte shortNum2 = 40000;

// 该定义也是不合法的,默认的浮点类型为 double
float floatNum2 = 2.0;

虽然基本数据类型 byteshortintlongfloatdouble 经常使用,但是我们平时可能并不会过多关注其相应的包装类 ByteShortIntLongFloatDouble,这里以 Integer 类为例,分析其具体源码的实现。

Integer 类位于 java.lang 下,其继承于 Number 类,实现了 Comparable 比较接口,其定义如下:

public final class Integer extends Number implements Comparable<Integer>

关于父类 Number 类,已经在之前的文章中分析过了,可以查看我的另外一篇文章:

  • Number 类及各子类所占字节数源码分析

对于 Comparable 比较接口,其定义也是很简单的,只有一个 compareTo() 方法,如下:

package java.lang;
import java.util.*;

public interface Comparable<T> {
    public int compareTo(T o);
}

整个 Integer 类的源码还是比较多的,加上注释的话,接近 1600 行代码了,在分析源码的时候,按照以下的分类进行:

  • 成员变量
  • 构造函数
  • 内部缓存类
  • 方法

1、成员变量

Integer 类中定义了一个 value 成员变量,用于表示 Integer 对应的 int 值,如下:

// Integer 对象对应的 int 值
private final int value;

2、构造函数

Integer 类提供了2个构造函数,其定义如下:

public Integer(int value) {
    this.value = value;
}

public Integer(String s) throws NumberFormatException {
    // 调用 parseInt 方法,默认基数为 10,将字符串解析成 int
    this.value = parseInt(s, 10);
}

从上面可以看出,既可以从 int 类型构建 Integer,也可以从 String 类型构建,在实际的开发过程中,使用构造函数来创建 Integer 其实很少用到,因为 Integer 类和其他的普通类不一样,它对应于基础数据类型 int,而自从 JDK 1.5 提供的自动装箱和拆箱操作,可以使用一个数值来创建 Integer 对象;如果是对于 String 类型,想要转换为 Integer 类型,实际中更多直接使用 Integer.parseInt() 方法进行,如下:

// 自动装箱
Integer num = 1;
// Integer.parseInt() 方法首先返回的是 int 类型,然后会涉及自动装箱
Integer strNum = Integer.parseInt("1");

Java 的自动装箱和拆箱操作是编译器自动完成的,装箱的时候调用的是包装类的 valueOf() 方法,上面代码定义的 numstrNum 对象其实是相同的,至于为什么会相同,在下文分析 valueOf() 方法时会作出具体的解释。

3、内部缓存类

Integer 类的内部,定义了缓存类 IntegerCache,该类是一个静态类,其定义如下:

private static class IntegerCache {
    // 最小值 -128
    static final int low = -128;
    // 最大值
    static final int high;
    // 缓存数组
    static final Integer cache[];

    static {
        // 最大值默认为 127,也可以通过属性字段配置,具体属性名为 java.lang.Integer.IntegerCache.high
        int h = 127;
        // 获取 java.lang.Integer.IntegerCache.high 属性值
        String integerCacheHighPropValue =
            sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
        // 如果属性值不为空
        if (integerCacheHighPropValue != null) {
            try {
                // 使用 parseInt() 方法将属性值转换为 int 类型
                int i = parseInt(integerCacheHighPropValue);
                // 取属性值与127两者之间的最大值,也就是说 i 的最小值其实是127
                i = Math.max(i, 127);
                // 数组的最大值为 Integer.MAX_VALUE
                // 取 i 和 Integer.MAX_VALUE - (-low) -1 两者之间最小值
                h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
            } catch( NumberFormatException nfe) {
                // If the property cannot be parsed into an int, ignore it.
            }
        }
        high = h;

        // 创建大小为 (high - low) + 1 的整型数组
        cache = new Integer[(high - low) + 1];
        int j = low;
        // 创建并缓存 Integer 对象
        for(int k = 0; k < cache.length; k++)
            cache[k] = new Integer(j++);

        // range [-128, 127] must be interned (JLS7 5.1.7)
        // 断言缓存最大值 high 大于等于 127
        assert IntegerCache.high >= 127;
    }

    private IntegerCache() {}
}

IntegerCache 的定义可以看出,3个成员变量都是静态常量,只有一个静态代码块,没有任何其他的方法,关于静态代码块的分析已经在代码中备注了,从静态代码块中可以得出以下的几点:

  • IntegerCache 的缓存最小值 low 为 -128,与其他的类 ByteShortLong 是相同的;
  • IntegerCache 的缓存最大值 high 可以通过属性 java.lang.Integer.IntegerCache.high 进行设置,而其他的类 ByteShortLong 都是无法设置的;
  • IntegerCache 的缓存最大值 high 的最小值为127,如果通过属性设置的值比127小,则赋值为 127,也就是此时的属性值是无效的;

关于 ByteShortIntegerLong 的内部缓存类详细的对比与分析,可以参考我的上一篇源码分析文章:

  • [Byte、Short、Integer、Long内部缓存类的对比与分析]()

4、方法

Integer 类提供的方法较多,大部分都是静态方法,主要分为以下的几类:

  • toString() 方法:主要是将 Integer 转换为字符串,这类方法中还有 toUnsignedString()toHexString() 等等;
  • parseInt() 方法:这个方法和 toString() 方法相反,主要是将 String 类型转换为 int 类型;
  • valueOf() 方法:主要是通过 intString 类型创建 Integer 对象;
  • getInteger() 方法:主要用于获取指定名称的系统属性的整数值;
  • compare() 方法:用于比较两个整型数值是否相等;
  • 位操作方法
  • 其他方法

4.1 toString() 方法

public String toString() {
    return toString(value);
}

public static String toString(int i) {
    // 如果为 int 类型最小值
    if (i == Integer.MIN_VALUE)
        return "-2147483648";
    // 获取 int 数值字符长度
    int size = (i < 0) ? stringSize(-i) + 1 : stringSize(i);
    char[] buf = new char[size];
    getChars(i, size, buf);
    return new String(buf, true);
}

public static String toString(int i, int radix) {
    // MIN_RADIX 为 2,MAX_RADIX 为36,如果不在范围内,默认为 10
    if (radix < Character.MIN_RADIX || radix > Character.MAX_RADIX)
        radix = 10;

    /* Use the faster version */
    if (radix == 10) {
        return toString(i);
    }

    char buf[] = new char[33];
    boolean negative = (i < 0);
    int charPos = 32;

    if (!negative) {
        i = -i;
    }

    while (i <= -radix) {
        buf[charPos--] = digits[-(i % radix)];
        i = i / radix;
    }
    buf[charPos] = digits[-i];

    if (negative) {
        buf[--charPos] = '-';
    }

    return new String(buf, charPos, (33 - charPos));
}

/** 转换为指定进制的无符号字符串 */
public static String toUnsignedString(int i, int radix) {
    return Long.toUnsignedString(toUnsignedLong(i), radix);
}

/** 转换为十六进制 */
public static String toHexString(int i) {
    return toUnsignedString0(i, 4);
}

/** 转换为八进制 */
public static String toOctalString(int i) {
    return toUnsignedString0(i, 3);
}

/** 转换为二进制 */
public static String toBinaryString(int i) {
    return toUnsignedString0(i, 1);
}

/** 转换为无符号字符串,int 类型转换为无符号可能越界,因此使用 long 类型 */
public static String toUnsignedString(int i) {
    return Long.toString(toUnsignedLong(i));
}

private static String toUnsignedString0(int val, int shift) {
    // assert shift > 0 && shift <=5 : "Illegal shift value";
    int mag = Integer.SIZE - Integer.numberOfLeadingZeros(val);
    int chars = Math.max(((mag + (shift - 1)) / shift), 1);
    char[] buf = new char[chars];

    formatUnsignedInt(val, shift, buf, 0, chars);

    // Use special constructor which takes over "buf".
    return new String(buf, true);
}

toString() 方法又细分了很多方法,比如转换为二进制、八进制、十六进制、无符号字符串等方法,利用 toString() 方法可以将 int 整型转换为二进制、八进制、十六进制,如下:

public static void main(String[] args) {
    // 由于最小的 MIN_RADIX 为 2,而传入的 1 小于 2,所以默认为 10进制
    System.out.println(Integer.toString(10, 1));
    // 十进制
    System.out.println(Integer.toString(10, 10));
    // 二进制
    System.out.println(Integer.toBinaryString(10));
    // 八进制
    System.out.println(Integer.toOctalString(10));
    // 十六进制
    System.out.println(Integer.toHexString(10));
}

输出结果如下:

10
10
1010
12
a

4.2 parseInt() 方法

public static int parseInt(String s) throws NumberFormatException {
    // 默认转换为 10 进制数值,这个方法在实际中应用最多
    return parseInt(s,10);
}

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("null");
    }

    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");
    }

    int result = 0;
    boolean negative = false;
    int i = 0, len = s.length();
    int limit = -Integer.MAX_VALUE;
    int multmin;
    int digit;

    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);

            if (len == 1) // Cannot have lone "+" or "-"
                throw NumberFormatException.forInputString(s);
            i++;
        }
        multmin = limit / radix;
        while (i < len) {
            // Accumulating negatively avoids surprises near MAX_VALUE
            digit = Character.digit(s.charAt(i++),radix);
            if (digit < 0) {
                throw NumberFormatException.forInputString(s);
            }
            if (result < multmin) {
                throw NumberFormatException.forInputString(s);
            }
            result *= radix;
            if (result < limit + digit) {
                throw NumberFormatException.forInputString(s);
            }
            result -= digit;
        }
    } else {
        throw NumberFormatException.forInputString(s);
    }
    return negative ? result : -result;
}

public static int parseUnsignedInt(String s) throws NumberFormatException {
    // 转换为无符号的十进制数值
    return parseUnsignedInt(s, 10);
}

public static int parseUnsignedInt(String s, int radix)
    throws NumberFormatException {
    if (s == null)  {
        throw new NumberFormatException("null");
    }

    int len = s.length();
    if (len > 0) {
        char firstChar = s.charAt(0);
        if (firstChar == '-') {
            throw new
                NumberFormatException(String.format("Illegal leading minus sign " +
                                                    "on unsigned string %s.", s));
        } else {
            if (len <= 5 || // Integer.MAX_VALUE in Character.MAX_RADIX is 6 digits
                (radix == 10 && len <= 9) ) { // Integer.MAX_VALUE in base 10 is 10 digits
                return parseInt(s, radix);
            } else {
                long ell = Long.parseLong(s, radix);
                if ((ell & 0xffff_ffff_0000_0000L) == 0) {
                    return (int) ell;
                } else {
                    throw new
                        NumberFormatException(String.format("String value %s exceeds " +
                                                            "range of unsigned int.", s));
                }
            }
        }
    } else {
        throw NumberFormatException.forInputString(s);
    }
}

4.3 valueOf() 方法

valueOf() 方法主要是通过 intString 类型创建 Integer 对象,其源码如下:

public static Integer valueOf(int i) {
    // 如果数值 i 在缓存的区间之内,则直接从缓存复用 Integer 对象
    // 自动装箱的时候会调用此方法
    if (i >= IntegerCache.low && i <= IntegerCache.high)
        return IntegerCache.cache[i + (-IntegerCache.low)];
    return new Integer(i);
}

public static Integer valueOf(String s) throws NumberFormatException {
    // 先通过 parserInt() 方法将字符串转为 int,默认为十进制
    return Integer.valueOf(parseInt(s, 10));
}

public static Integer valueOf(String s, int radix) throws NumberFormatException {
    // 先通过 parserInt() 方法将字符串转为指定进制的 int
    return Integer.valueOf(parseInt(s,radix));
}

intInteger 之间涉及到自动装箱和拆箱操作,自动装箱操作调用的是 valueOf() 方法,如果数值在缓存区间的最小值与最大值之间时,会复用缓存中的对象,默认缓存区间为 [-128, 127]

4.4 getInteger() 方法

getInteger() 方法主要用于获取指定名称的系统属性的整数值,相关源码如下:

public static Integer getInteger(String nm) {
    return getInteger(nm, null);
}

public static Integer getInteger(String nm, int val) {
    Integer result = getInteger(nm, null);
    // 如果属性值不存在,则返回默认值 val
    return (result == null) ? Integer.valueOf(val) : result;
}

public static Integer getInteger(String nm, Integer val) {
    String v = null;
    try {
        // 获取属性名为 nm 的系统属性
        v = System.getProperty(nm);
    } catch (IllegalArgumentException | NullPointerException e) {
    }
    if (v != null) {
        try {
            // 调用 decode 方法将属性值转换为 Integer 
            return Integer.decode(v);
        } catch (NumberFormatException e) {
        }
    }
    return val;
}

关于 System.getProperty() 获取系统属性的更多应用,可以参考我写的另一篇文章:

  • Java代码判断当前操作系统是Windows或Linux或MacOS

4.5 compare()`方法

compare() 方法用于比较两个整型数值是否相等,其源码如下:

public int compareTo(Integer anotherInteger) {
    // 这里比较的是数值
    return compare(this.value, anotherInteger.value);
}

public static int compare(int x, int y) {
    // 如果 x 小于 y,则返回 -1
    // 如果 x 等于 y,则返回 0
    // 如果 x 大于 y,则返回 1
    return (x < y) ? -1 : ((x == y) ? 0 : 1);
}

public static int compareUnsigned(int x, int y) {
    // 无符号数值比较
    return compare(x + MIN_VALUE, y + MIN_VALUE);
}

public boolean equals(Object obj) {
    if (obj instanceof Integer) {
        // 比较数值是否相等
        return value == ((Integer)obj).intValue();
    }
    return false;
}

对于 compare() 比较方法,返回的结果有 -1、0、1 三种,如果不需要返回 boolean 类型,可以使用这个方法,对于 Integer 包装类的比较,建议使用 equals() 方法。

4.6 位操作方法

Integer 类中提供了很多位操作的方法,主要有以下的这些:

  • highestOneBit():返回给定数值二进制最高位1的权值;
  • lowestOneBit():返回给定数值二进制最低位1的权值;
  • numberOfLeadingZeros():返回给定数值二进制中从最左边算起 0 的个数;
  • numberOfTrailingZeros():返回给定数值二进制中从最右边算起 0 的个数;
  • bitCount():返回给定数值的二进制中 1 的个数;
  • rotateLeft():将给定整数值的二进制补码数左旋转给定位数;
  • rotateRight():将给定整数值的二进制补码数右旋转给定位数;
  • reverse():将给定整数值的二进制补码数进行反转;
  • signum():返回给定 int 值的 sign 函数;
  • reverseBytes():将给定整数的二进制补码按照字节进行反转;

4.7 其他方法

  • sum():获取两个整数的和;
  • max():获取两个整数之间的最大值,自 JDK 1.8 版本新增的,具体调用的是 Math.max() 方法;
  • min():获取两个整数之间的最小值,自 JDK 1.8 版本新增的,具体调用的是 Math.min() 方法;

精彩评论(0)

0 0 举报