Java小陷阱

  基本数据类型与字符串的连接

  

  在Java中,+不仅可作为加法运算符使用,还可作为字符串连接运算符使用。

  当把任何基本数据类型的值与字符串值进行连接运算时,基本类型的值将自动类型转换为字符串类型。

  

public class PrimitiveAndString
{
    public static void main(String[] args)
    {
        //下面的语句输出 7Hello!
        System.out.println(3 + 4 + "Hello!");

        //下面的语句输出 Hello!34
        System.out.println("Hello!" + 3 + 4);

        //下面的语句输出 Hello!a7
        System.out.println("Hello!" + 'a' + 7);

        //下面的语句输出 104Hello!
        System.out.println('a' + 7 + "Hello!");
    }
}

  上面程序中第一个”3 + 4 + "Hello!"“的表达式,这个表达式先执行”3 + 4“运算,这是执行两个整数之间的加法,得到7,然后进行”7 + "Hello!"“的运算,此时会把7当成字符串进行处理,从而得到7Hello!。

  第二个,对于”"Hello!" + 3 + 4“表达式,先进行”"Hello!" + 3“运算,得到一个Hello!3字符串,再和4进行连接运算,4也被转换成字符串进行处理,最后得到Hello!34。

  第三个表达式”"Hello!" + 'a' + 7“同第二个类似。

  对于最后一个表达式”'a' + 7 + "Hello!"“,先进行”'a' + 7“加法运算,其中'a'自动提升到int类型,编程a对应的ASCⅡ值:97,从”97+7“将得到104,然后进行”104 + "Hello!"“运算,104会自动转换成字符串,将变成两个字符串的连接运算,从而得到104Hello!。

  Integer自动装箱的缓存机制

  (2016年腾讯实习生招聘笔试题,扩展)下面这段java代码的输出结果是?(不考虑java 1.5之前的老版本,因为老版本不支持自动装箱)

public class IntegerTest {

    public static void main(String[] args) {
        
        Integer i1 = 127;   // autoboxing
        Integer i2 = 127;   // autoboxing
        System.out.println(i1.equals(i2));   // true
        System.out.println(i1 == i2);        // true

        Integer i3 = 128;   // autoboxing
        Integer i4 = 128;   // autoboxing
        System.out.println(i3.equals(i4));   // true
        System.out.println(i3 == i4);        // false

        Integer i5 = new Integer(127);
        Integer i6 = new Integer(127);
        System.out.println(i5.equals(i6));  // true
        System.out.println(i5 == i6);       // false

        Integer i7 = 127;   // autoboxing
        Integer i8 = new Integer(127);
        System.out.println(i7.equals(i8));  // true
        System.out.println(i7 == i8);       // false

        int i = 127;
        System.out.println(i7.equals(i));   // true
        System.out.println(i8.equals(i));   // true
        System.out.println(i7 == i);        // true
        System.out.println(i8 == i);        // true
    }
}

  分析:本题的考察点在于Integer类对于[-128 , 127]之间的整数自动装箱缓存的机制,查看Java系统中java.lang.Integer类的源代码,其中有一个叫做IntegerCache的静态内部类如下:

    /**
     * Cache to support the object identity semantics of autoboxing for values between
     * -128 and 127 (inclusive) as required by JLS.
     *
     * The cache is initialized on first usage.  The size of the cache
     * may be controlled by the {@code -XX:AutoBoxCacheMax=<size>} option.
     * During VM initialization, java.lang.Integer.IntegerCache.high property
     * may be set and saved in the private system properties in the
     * sun.misc.VM class.
     */

    private static class IntegerCache {
        static final int low = -128;
        static final int high;
        static final Integer cache[];

        static {
            // high value may be configured by property
            int h = 127;
            String integerCacheHighPropValue =
                sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
            if (integerCacheHighPropValue != null) {
                try {
                    int i = parseInt(integerCacheHighPropValue);
                    i = Math.max(i, 127);
                    // Maximum array size is Integer.MAX_VALUE
                    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;

            cache = new Integer[(high - low) + 1];
            int j = low;
            for(int k = 0; k < cache.length; k++)
                cache[k] = new Integer(j++);

            // range [-128, 127] must be interned (JLS7 5.1.7)
            assert IntegerCache.high >= 127;
        }

        private IntegerCache() {}
    }

  从上面的代码可以看出,系统已经把[-128 , 127]之间的256个整数自动装箱成Integer实例,并放入了cache数组中缓存起来

  当把[-128 , 127]之间的同一个整数自动装箱成Integer实例时,永远都是引用cache数组的同一个元素,(i1 == i2)结果为true;

  当把一个不在[-128 , 127]范围内的整数自动装箱成Integer实例时,系统总是重新new一个Integer实例,开辟新的内存空间,因此(i3 == i4)、(i5 == i6)、以及(i7 == i8)的结果均为false;

  Integer和int进行比较时,Integer会自动拆箱成int类型变成值比较,因此(i7 == i)和(i8 == i)的结果均为true;

  Integer类重写了从Object类继承而来的equals方法进行值比较,所以上述equals的结果均为true