“abc”、new String(“abc“)、“a”+“b“=“ab“、常量池

“abc”

如果常量池存在“abc”常量

变量s1指向常量池的"abc"

如果常量池不存在"abc"常量

常量池先创建"abc",然后变量s1指向常量池的"abc"

“abc”、new String(“abc“)、“a”+“b“=“ab“、常量池_第1张图片

new String(“abc”)

如果常量池存在“abc”常量

将“abc”拷贝到Java堆,变量s1指向Java堆的"abc"

如果常量池不存在"abc"常量

常量池先创建"abc",然后将“abc”拷贝到Java堆,变量s1指向Java堆的"abc"

“abc”、new String(“abc“)、“a”+“b“=“ab“、常量池_第2张图片

思考题

问题

public static void main(String[] args) {
        String s1 = "hello";
        String s2 = new StringBuilder("he").append("llo").toString();
        String s3 = s2.intern();
        String s4 = "hello";

        System.out.println(s1 == s2);	// false
        System.out.println(s2 == s3);	// false
        System.out.println(s3 == s4);	// true
}

分析

s1 = "hello",将"hello"放入方法区,并且s1指向"hello"。s2获取到了“hello”,但是这并不是从方法区获取的"hello",
s3=s2.intern()是将s2的"hello"与方法区进行对比,如果存在"hello",那么就返回原来"hello"的方法区地址,
否则放入方法区并返回新的方法区地址。这里因为方法区已经有"hello",所以返回了以前的地址,所以s2 != s3。
s4 = "hello",指向了方法区的"hello"的地址。

再看一下下面的例子

        String s2 = new StringBuilder("he").append("llo").toString();
        String s3 = s2.intern();
        String s4 = "hello";
        
        System.out.println(s2 == s3);	// true
        System.out.println(s3 == s4);	// true

你可能感兴趣的:(12万字Java面经总结,Java,java,开发语言)