八种基本数据类型的默认值
整型 byte short int long
浮点型 float double
逻辑型 boolean
字符型 char
Java 基本类型的包装类的大部分都实现了常量池技术,即 Byte,Short,Integer,Long,Character,Boolean;前面 4 种包装类默认创建了数值[-128,127] 的相应类型的缓存数据,Character创建了数值在[0,127]范围的缓存数据,Boolean 直接返回True Or False。如果超出对应范围仍然会去创建新的对象。
为啥把缓存设置为[-128,127]区间? (参见issue/461)性能和资源之间的权衡。
1 2 3 public static Boolean valueOf (boolean b) { return (b ? TRUE : FALSE); }
1 2 3 4 5 6 7 8 9 private static class CharacterCache { private CharacterCache () {} static final Character cache[] = new Character[127 + 1 ]; static { for (int i = 0 ; i < cache.length; i++) cache[i] = new Character((char )i); } }
两种浮点数类型的包装类 Float,Double 并没有实现常量池技术。
1 2 3 4 5 6 7 8 9 Integer i1 = 33 ; Integer i2 = 33 ; System.out.println(i1 == i2); Integer i11 = 333 ; Integer i22 = 333 ; System.out.println(i11 == i22); Double i3 = 1.2 ; Double i4 = 1.2 ; System.out.println(i3 == i4);
Integer 缓存源代码:
1 2 3 4 5 6 7 8 public static Integer valueOf (int i) { if (i >= IntegerCache.low && i <= IntegerCache.high) return IntegerCache.cache[i + (-IntegerCache.low)]; return new Integer(i); }
应用场景:
Integer i1=40;Java 在编译的时候会直接将代码封装成 Integer i1=Integer.valueOf(40);,从而使用常量池中的对象。
Integer i1 = new Integer(40);这种情况下会创建新的对象。
1 2 3 Integer i1 = 40 ; Integer i2 = new Integer(40 ); System.out.println(i1==i2);
Integer 比较更丰富的一个例子:
1 2 3 4 5 6 7 8 9 10 11 12 13 Integer i1 = 40 ; Integer i2 = 40 ; Integer i3 = 0 ; Integer i4 = new Integer(40 ); Integer i5 = new Integer(40 ); Integer i6 = new Integer(0 ); System.out.println("i1=i2 " + (i1 == i2)); System.out.println("i1=i2+i3 " + (i1 == i2 + i3)); System.out.println("i1=i4 " + (i1 == i4)); System.out.println("i4=i5 " + (i4 == i5)); System.out.println("i4=i5+i6 " + (i4 == i5 + i6)); System.out.println("40=i5+i6 " + (40 == i5 + i6));
结果:
1 2 3 4 5 6 i1=i2 true i1=i2+i3 true i1=i4 false i4=i5 false i4=i5+i6 true 40 =i5+i6 true
解释:
语句 i4 == i5 + i6,因为+这个操作符不适用于 Integer 对象,首先 i5 和 i6 进行自动拆箱操作,进行数值相加,即 i4 == 40。然后 Integer 对象无法与数值进行直接比较,所以 i4 自动拆箱转为 int 值 40,最终这条语句转为 40 == 40 进行数值比较。
参考 8 种基本类型的包装类和常量池