1、基础类型有byte short int long char boolean float double八种。
其中byte short int long char 的包装类型是存放在常量池(用来维护该类型所用到常量的有序集合)中的,浮点类型未实现常量池。
常量池存放的是>=-128和<=127中的数据,当数据不在这个范围内时,则不在常量池中维护
Integer a=new Integer(128);
Integer b=new Integer(128);
a==b?false,由于a,b超出127限制,不存在与常量池,所以引用不同。
public static integer valueOf(int t){
final int offset=128;
if(t>=-128&&t<=127){
return IntegerCache.cache[i+offset];
}
return new Integer(i)
}
由valueOf方法可以看出Integer类型在-128~127范围是取自常量池中。
2、String a=new String("bb");
可能创建了一个对象,也可能创建了两个对象。
在类编译期间,会先去常量池中查找是否有“bb”的对象,如果有则在栈中复制一份;如果没有,先在常量池中创建一份,再复制一份到堆中。之后把引用交给栈中的s。
String a = "a1";
String b = "a" + 1;
System.out.println((a == b)); //result = true
String a = "atrue";
String b = "a" + "true";
System.out.println((a == b)); //result = true
String a = "a3.4";
String b = "a" + 3.4;
System.out.println((a == b)); //result = true
分析:JVM对于字符串常量的"+"号连接,将程序编译期,JVM就将常量字符串的"+"连接优化为连接后的值,拿"a" + 1来说,经编译器优化后在class中就已经是a1。在编译期其字符串常量的值就确定下来,故上面程序最终的结果都为true。
String a = "ab";
String bb = "b";
String b = "a" + bb;
System.out.println((a == b)); //result = false
分析:JVM对于字符串引用,由于在字符串的"+"连接中,有字符串引用存在,而引用的值在程序编译期是无法确定的,即"a" + bb无法被编译器优化,只有在程序运行期来动态分配并将连接后的新地址赋给b。所以上面程序的结果也就为false。
String a = "ab";
final String bb = "b";
String b = "a" + bb;
System.out.println((a == b)); //result = true
分析:和[3]中唯一不同的是bb字符串加了final修饰,对于final修饰的变量,它在编译时被解析为常量值的一个本地拷贝存储到自己的常量池中或嵌入到它的字节码流中。所以此时的"a" + bb和"a" + "b"效果是一样的。故上面程序的结果为true。
可以看出当字符串+常量时,可以在编译期优化链接的值。当字符串+引用时,编译期无法确定,不能被优化