String s="a"+"b"+"c";
首先看一下这道题创建几个对象?答案是一个对象
其实如果查看Java源码和反编译后的字节码文件,就可以用直观地看出,只创建了一个String对象。
呢为什么源文件中的字符串拼操作到了字节码文件就消失了呢?
这就要提到编译期的常量折叠技术了:
这是因为在编译期间应用了编译器优化中的一种常量折叠技术,会将编译期常量的加减乘除运算过程在编译过程中折叠。编译器通过语法分析,将常量表达式计算求求值,用计算出来的值替换表达式,从而在运行期间节省处理器资源。
特点:
- 被声明为final
- 基本类型或字符串类型
- 声明时就已经初始化
- 使用常量表达式进行初始化
以上条件都满足,才可能是一个编译期常量
这里着重说一下第三、四条:
final String s1="hello"+"world"; final String s2=UUID.randomUUID().toString()+"Hydra";
编译器在编译期就能得到s1的值是”hello word“,不需要等到程序运行期间,所以s1属于编译期常量;对于s2来说,虽然也已经被声明为final,并且声明时就已经初始化,但是它使用的不是常量表达式,所以它是运行期常量。
编译期和运行期还有一个特点就是:是否需要对类进行初始化.
public class Test{
public static void main(String[] args){
System.out.println(A.a);
}
}
class A{
static {
System.out.println("Test class");
}
public static int a=1;
}
输出的结果为:
1
从这里可以看出被final关键字修饰后变量a成为了编译期常量,所以它将不会初始化类,这也说明了在声明编译期常量final关键字是必要的,而static是非必要的。



