1.java基础类型对应的包装类型是带有缓存功能的

   1)Integer/Short/Long 会对-128到127的值进行缓存;

   2)Character 会对0到127的值进行缓存

   Float、Double没有此功能

  为什么说Integer类带有缓存功能,见java源代码Integer.java:

  其内部包含一个静态类IntegerCache,会实例化一个-128到127的Integer数组;

    private static class IntegerCache {
private IntegerCache(){}

static final Integer cache[] = new Integer[-(-128) + 127 + 1];

static {
for(int i = 0; i < cache.length; i++)
cache[i] = new Integer(i - 128);
}

在调用Integer.valueOf函数时就会使用到这个缓存功能,如果值在-128到127,则直接从缓存的Integer数组中取值

public static Integer valueOf(int i) {
final int offset = 128;
if (i >= -128 && i <= 127) { // must cache
return IntegerCache.cache[i + offset];
}
return new Integer(i);
}

做个实验:

Integer i = 1;
Integer j = 1;
System.out.println("i==j:"+(i==j));

Integer i1 = new Integer(1);
Integer j1 = new Integer(1);
System.out.println("i1==j1:"+(i1==j1));

结果:

i==j:true

i1==j1:false

原因:

1.Integer i=1; 编译之后实际变成了:Integer i = Integer.valueOf(1);(可以通过对class文件进行反编译进行确认),这样就会从缓存中获取数据;所以i和j是同一个对象;

2.对于采用new Integer方式,每次都会生成一个新的对象,所以i1和j1不相等。

2.String复制语句编译之后会发生变化

1) String s = "a"+"b";

编译后变为(通过反编译class类可以确认):

String s = "ab";

2)

String s1 = "a";

String s2 = s1+"b";

编译后变为:

String s1 = "a";

String s2 = new StringBuilder(String.valueOf(s1)).append("b").toString();

这个需要通过字节码查看工具确认:

 0 ldc #16 <a>
2 astore_1
3 new #18 <java/lang/StringBuilder>
6 dup
7 aload_1
8 invokestatic #20 <java/lang/String.valueOf>
11 invokespecial #26 <java/lang/StringBuilder.<init>>
14 ldc #29 <b>
16 invokevirtual #31 <java/lang/StringBuilder.append>
19 invokevirtual #35 <java/lang/StringBuilder.toString>
22 astore_2
23 return