String对象的创建形式:
Java中字符串对象创建有两种形式,一种为字面量形式,如String str = "droid",另一种就是使用new这种标准的构造对象的方法,如String str = new String("droid"),这两种实现其实存在着一些性能和内存占用的差别。这一切都是源于JVM为了减少字符串对象的重复创建,其维护了一个特殊的内存,这段内存被成为字符串常量池或者字符串字面量池。Java 程序中的所有字符串字面值(如 String str="abc" )都是此类的实例。java中的字符串是常量,它们的值在创建之后不能更改,其实就是说一旦这个字符串确定了,那么就会在jvm堆内存中生成该字符串对象,并且在字符串常量池中生成这个字符串的引用。字符串本身不能改变,但str变量中记录的地址值是可以改变的。
String对象的创建原理:
当字面量形式创建字符串对象时,JVM首先会对这个字面量进行检查,如果字符串常量池中存在相同内容的字符串对象的引用,则将这个引用返回,否则新的字符串对象被创建,然后将这个引用放入字符串常量池,并返回该引用。
当使用new构造对象时,如以String str = new String("adc")为例,String str 第一部分变量名,=为第二部分给str赋值使用的,new String()为第三部分创建对象,"abc"为第三部分对象内容。第三部分new String()在堆内存中创建了一个对象。从上面的源码中我们会看到一个带参的String的构造器;如果字符串常量池里面没有所传的参数对象,新的字符串对象被创建,然后将这个引用放入字符串常量池;但是如果字符串常量池中存在相同内容的字符串对象的引用,那么就会直接获取。
String对象的垃圾回收:
假设一个字符串"everor"已经进入了字符串常量池,但是当前系统中没有一个String对象引用常量池中的"everor"常量(即String str = new String("everor");),也没有其他地方引用了这个字面量(即String str = "everor";),如果这个时候发生了内存回收,而且有必要的话,这个"everor"常量就会被系统清理出字符串常量池。
通过代码加深理解:
1 public class TestStringConstant {
2 public static void main(String args[]) {
3 // 字符串常量,分配在常量池中,编译器会对其进行优化, Interned table
4 // 即当一个字符串已经存在时,不再重复创建一个相同的对象,而是直接将s2也指向"hello".
5 String s1 = "hello";
6 String s2 = "hello";
7 // new出来的对象,分配在heap中.s3与s4虽然它们指向的字符串内容是相同的,但是是两个不同的对象.
8 // 因此==进行比较时,其所存的引用是不同的,故不会相等
9 String s3 = new String("world");
10 String s4 = new String("world");
11
12 System.out.println(s1 == s2); // true
13 System.out.println(s3 == s4); // false
14 System.out.println(s3.equals(s4)); // true
15 // String中equals方法已经被重写过,比较的是内容是否相等.
16 }
17 }