在日常开发中,String 的使用是非常多的,那么再深入的关于 String 的内存分配,就属于比较少涉及的了。

今天,小编就给大家说说这个,String 的内存分配。

一,String的创建和基本特性



创建字符串对象,主要有两种方式:


//方式1
String str = "abc";
//方式2
String str = new String("abc");

方式1表示:在字符串常量池(String Pool)中先寻找值为 “abc” 的内存地址,如果找不到,则开辟内存,赋值该内存地址为 “abc”,然后将该内存地址传递给变量 str,str 直接指向字符串常量池中值为 “abc” 的内存地址。

方式2表示:先在堆(Heap)空间中开辟空间,new String对象,然后再去字符串常量池中寻找值为 “abc” 的地址,如果找到,则把该地址的引用传给 new String对象;如果找不到,则开辟内存,赋值该内存地址为 “abc”,然后把该地址的引用传给 new String对象。最后,把堆空间中 new String 对象的地址,传递给 str,str 指向堆空间中的对象。

所以,估计没明白,那就看看下面这个图吧。

java new string 内存 java string 内存分配_数据结构



应该没看明白,我直接上代码感受一下。

public class StringTest {

    public static void main(String[] args) {
        String str1 = "abc";
        String str2 = "abc";
        String str3 = new String("abc");
        String str4 = new String("abc");

        System.out.println(str1 == str2);
        System.out.println(str1 == str3);
        System.out.println(str1.equals(str3));

        System.out.println(str3 == str4);
    }
}



输出:

D:\jdk\jdk1.8.0_171\bin\java.exe
true
false
true
false



就一句话:

字符串常量池(String Pool)中是不会存储相同内容的字符串。




二,String的内存分配



Jdk6以及以前,字符串常量池存放在永久代。
Jdk7及以后,字符串常量池存放在堆(Heap)中。

java new string 内存 java string 内存分配_jvm_02



java new string 内存 java string 内存分配_jvm_03



字符串常量池在 jdk7及以后(小编用的是jdk8)存放在堆空间验证:



1,代码
public class StringTest {

    public static void main(String[] args) {
        //使用set保持常量池引用,避免full gc回收常量池行为
        Set<String> set = new HashSet<>();
        short i = 0;
        while (true){
            set.add(String.valueOf(i++).intern());
        }
    }
}



2,设置参数值:

java new string 内存 java string 内存分配_jvm_04


3,执行结果
D:\jdk\jdk1.8.0_171\bin\java.exe
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space



发现堆空间内存溢出,可证明 jdk8版本的字符串常量池位于堆空间中。