首先先看如下一小段代码:
final String book = "考研成绩:"+"380";
final String book2 = "考研成绩:"+380;
final String book3 = "考研成绩:"+String.valueOf(380);
String book4 = "考研成绩:"+ String.valueOf(380);
String book5 = "考研成绩:380";
String book6 = "考研成绩:" + (300 + 80);
System.out.println(book == "考研成绩:380");//true
System.out.println(book2 == "考研成绩:380");//true
System.out.println(book3 == "考研成绩:380");//false
System.out.println(book4 == "考研成绩:380");//false
System.out.println(book5 == book4);//false
System.out.println(book5 == "考研成绩:380");//true
System.out.println(book6 == "考研成绩:380");//true
我们都清楚,两个引用类型变量中“==”成立说明指向了同一个对象。
代码体现出如果被赋值的表达式只是基本的算术运算或者字符串连接运算,没有访问普通变量,调用方法,即使字符串连接运算中包含隐式类型(数值转换为字符串)转换,编译器依然可以在编译时中就确定他们的值。对于book,book2,book5实际都指向常量池的同一个对象。而对于book3与book4需要调用String类的方法,因此编译器无法在编译时确定他们的值。所以并不相等。
需要注意的是,如果用final修饰的实例变量拼接在定义时指定初始值,编译器会在编译阶段就确定结果。但如果两个普通变量连接,编译器就无法在编译阶段确定结果。具体体现在和字符串池中缓冲的字符串进行比较的结果中。
String s1 = "第一第二";
String s2 = "第一" + "第二";
System.out.println(s1 == s2);//true,简单的字符串拼接运算
String s3 = "第一";
String s4 = "第二";
String s5 = s3 + s4;
System.out.println(s5 == s1);//false,此时相当于访问了变量,编译阶段无法确定结果
final String s6 = "第一";
final String s7 = "第二";
String s8 = s6 + s7;
System.out.println(s1 == s8);//true,final修饰,编译时就可以确定s8结果
需要特别指出的是,final实例变量要实现编译时确定值,需要在定义时直接指定初始值。