-
首先,我们来聊聊Java中的字符串(String)。在Java里,字符串是不可变的,也就是说,一旦我们创建了一个字符串,它的值就不能更改了。这涉及到几个核心概念:
- 字符串常量池:这是Java中一个特别的地方,用来缓存我们写的那些字符串字面量。这样做的好处是,如果有相同的字符串出现,Java就不需要重复创建对象,从而节省了内存。
- 堆:这是Java虚拟机用来动态分配内存的地方,所有的对象都是在这里创建的。
- 字节码:这是我们Java源代码编译后的中间代码,JVM实际上执行的就是这些字节码。
接下来,让我们看看字符串对象是如何创建的:
当我们用
new String("abc")
这样的语法创建字符串时,JVM会先检查字符串常量池里是否已经有了 “abc” 的引用。如果没有,它会在堆上创建一个新的String对象,并且把这个引用放到字符串常量池里。如果已经有了,JVM就只会在堆上创建一个指向那个引用的新对象。对应的字节码:
new
指令:在堆上分配一个未初始化的String对象。dup
指令:复制这个引用,为接下来的操作做准备。Ldc
指令:加载字符串常量池中的引用。invokespecial
指令:调用String对象的构造方法来完成对象的初始化。
举个例子,如果字符串常量池里没有 “abc”,那么 String s1 = new String("abc");
会产生两个对象:一个在堆上,一个在字符串常量池中。
但如果 “abc” 已经存在了,那么 String s2 = new String("abc");
就只会在堆上创建一个对象。