设为首页 加入收藏

TOP

String 常量池和 String#intern()(二)
2017-12-30 06:06:50 】 浏览:469
Tags:String 常量 池和 String#intern
; String s3 = new String("1") + new String("1"); s3.intern(); String s4 = "11"; System.out.println(s3 == s4); }

打印结果:

# jdk6下
false false
# jdk7下
false true

具体为什么稍后再解释,然后将s3.intern();语句下调一行,放到String s4 = "11";后面。将s.intern();放到String s2 = "1";后面:

public static void main(String[] args) {
    String s = new String("1");
    String s2 = "1";
    s.intern();
    System.out.println(s == s2);
    String s3 = new String("1") + new String("1");
    String s4 = "11";
    s3.intern();
    System.out.println(s3 == s4);
}

打印结果:

# jdk6下
false false
# jdk7下
false false

jdk6的解释

注:图中绿色线条代表String对象的内容指向;黑色线条代表地址指向。

jdk6中,上述的所有打印都是false。

因为jdk6的常量池放在Perm区中,和正常的Heap(指Eden、Surviver、Old区)完全分开。具体来说:使用引号声明的字符串都是通过编译和类加载直接载入常量池,位于Perm区;new出来的String对象位于Heap(E、S、O)中。拿一个Perm区的对象地址和Heap中的对象地址进行比较,肯定是不相同的。

Perm区主要存储一些加载类的信息、静态变量、方法片段、常量池等。

jdk7的解释

在jdk6及之前的版本中,字符串常量池都是放在Perm区的。Perm区的默认大小只有4M,如果多放一些大字符串,很容易抛出OutOfMemoryError: PermGen space

因此,jdk7已经将字符串常量池从Perm区移到正常的Heap(E、S、O)中了。

Perm区即永久代。本身用永久代实现方法区就容易遇到内存溢出;而且方法区存放的内容也很难估计大小,没必要放在堆中管理。jdk8已经取消了永久代,在堆外新建了一个Metaspace实现方法区。

正是因为字符串常量池移到了Heap中,才产生了上述变化。

第一段代码

jdk7代码-1

先看s3和s4:

  • 首先,String s3 = new String("1") + new String("1");,生成了多个对象,s3最终指向堆中的”11”。注意,此时常量池中是没有字符串”11”的。
  • 然后,s3.intern();,将s3中的字符串”11”放入了常量池中,因为此时常量池中不存在字符串”11”,因此常规做法与跟jdk6相同,在常量池中生成一个String对象”11”——然而,jdk7中常量池不在Perm区中了,相应做了调整:常量池中不需要再存储一份对象了,而是直接存储堆中的引用,也就是s3的引用地址。
  • 接下来,String s4 = "11";,”11”通过双引号显示声明,因此会直接去常量池中查找,如果没有再创建。发现已经有这个字符串了,也就是刚才通过s3.intern();存储在常量池中的s3的引用地址。于是,直接返回s3的引用地址,s4赋值为s3的引用,s4指向堆中的”11”。
  • 最后,s3、s4指向的堆中的”11”,常量池中存储s3的引用,满足s3 == s4

再看s和s2:

  • 首先,String s = new String("1");,生成了2个对象,常量池中的”1”和堆中的”1”,s指向堆中的”1”。
  • 然后,s.intern();,上一句已经在常量池中创建了”1”,所以此处什么都不做。
  • 接下来,,String s2 = "1";,常量池中有”1”,因此,s2直接指向常量池中的”1”。
  • 最后,s指向的堆中的”1”,s2指向常量池中的”1”,常量池中存储字符串”1”,不满足s == s2

第二段代码

jdk7代码-2

先看s3和s4,将s3.intern();放在了String s4 = "11";后:

  • 先执行String s4 = "11";,此时,常量池中不存在”11”,因此,将”11”放入常量池,然后s4指向常量池中的”11”。
  • 再执行s3.intern();,上一句已经在常量池中创建了”11”,所以此处什么都不做。
  • 最后,s3仍指向的堆中的”11”,s4指向常量池中的”11”,常量池中存储字符串”11”,不再满足s3 == s4

再看s和s2,将s.intern();放到String s2 = "1";后:

  • 先执行String s2 = "1";,之前已通过String s = new String("1");在常量池中创建了”1”,因此,s2直接指向常量池中的”1”。
  • 再执行s.intern();,常量池中有”1”,所以此处什么都不做。
  • 最后,s指向的堆中的”1”,s2指向常量池中的”1”,常量池中存储字符串”1”,仍不满足s == s2

区别小结

jdk7与jdk6相比,对String常量池的位置、String#intern()的语义都做了修改:

  • 将String常量池从Perm区移到了Heap区。
  • 调用String#intern()方法时,堆中有该字符串而常量池中没有,则直接在常量池中保存堆中对象的引用,而不会在常量池中重新创建对象。

使用姿势

建议直接阅读参考资料。

额外的问题

String#intern()的基本用法如下:

String s1 = xxx1.toString().intern();
String s2 = xxx2.toString().intern();
assert s1 == s2;

然而,xxx1.toString()xxx2.toString()已经创建了两个匿名String对象,这之后再调用String#intern()。那么,这两个匿名对象去哪了?

估计猴子对创建对象的过程理解有问题,或许xxx1.toString()返回时还没有将对象保存到堆上?或许String#intern()上做了什么语法糖?

后面有时间再解决吧。。。


参考:

首页 上一页 1 2 下一页 尾页 2/2/2
】【打印繁体】【投稿】【收藏】 【推荐】【举报】【评论】 【关闭】 【返回顶部
上一篇大型网站系统与 Java 中间件实践 下一篇代码生成利器:IDEA 强大的 Live ..

最新文章

热门文章

Hot 文章

Python

C 语言

C++基础

大数据基础

linux编程基础

C/C++面试题目