从String.intern()深入理解JVM

代码分析:

官方说明翻译:1.返回字符串对象的规范表示

2.字符串s调用intern时,如果常量池中有对象t满足t.equals(s)==true,则返回t的地址;否则,将s的引用(即地址)加入常量池并返回s的引用

3.s.intern()==t.intern()的充分必要条件是s.equals(t)==true

注意:1.jdk1.6及以前版本的运行时常量池在方法区中,jdk1.7及以后版本的运行时常量池在堆中。
2.jdk1.6及以前版本会将字符串拷贝一份到运行时常量池,返回常量池中地址,jdk1.7及以后将堆上的地址引用复制到常量池,返回该引用地址。在此讨论jdk1.7及以后版本。

一:

String s1=new String("GY");
System.out.println(s1.intern()==s1);//false



如果常量池中有字符串对象"GY",则new String("GY")只创建一个堆中的对象;如果常量池中没有字符串对象"GY",则先在常量池中创建"GY",再在堆中生成一个String对象,共两个。

二:

String s1=new String("GY")+new String("HAPPY");//生成三个对象,两个在常量池,一个在JVM堆中
System.out.println(s1.intern()==s1);//true



String s1=new String("GY")+new String("HAPPY)并没有在运行时常量池中生成“GYHAPPY"对象,调用s1.intern()时,发现常量池中没有,则将对象的引用加入到池中,返回该引用(即s1地址)。

三:

诡异现象:

String s1=new String("1")+new String("1");
System.out.println(s1.intern()==s1);//false
String s1=new String("ja")+new String("va");
System.out.println(s1.intern()==s1);//false

这是因为JVM加载过程中已经在常量池中生成了"11","java"这些对象,因此intern函数返回的是常量池的地址。

底层实现:

1.JAVA调用C++实现的StringTable的intern方法,原理与JAVA中的HashMap差不多,都是用链表法解决Hash冲突。jdk1.6以前StringTable长度固定为1009,如果常量池中String很多,则链表很长,导致查找效率低。jdk1.7的StringTable长度可设置,-XX:StringTableSize=XXXXXX。
2.一旦常量池中的字符串达到的一定的规模后,性能会急剧下降。


你可能感兴趣的:(Java基础)