UDN-企业互联网技术人气社区

板块导航

浏览  : 1320
回复  : 0

[讨论交流] 浅谈Java String内幕(2)

[复制链接]
呵呵燕的头像 楼主
发表于 2016-9-21 22:42:06 | 显示全部楼层 |阅读模式
  String.intern()原理

  String.intern()是一个Native方法,底层调用C++的 StringTable::intern 方法,源码注释:当调用 intern 方法时,如果常量池中已经该字符串,则返回池中的字符串;否则将此字符串添加到常量池中,并返回字符串的引用。

  1.   package com.ctrip.ttd.whywhy;

  2.   class Test {

  3.   public static void main(String args[]) {

  4.   String s1 = new StringBuilder().append("String").append("Test").toString();

  5.   System.out.println(s1.intern() == s1);

  6.   String s2 = new StringBuilder().append("ja").append("va").toString();

  7.   System.out.println(s2.intern() == s2);

  8.   }

  9.   }
复制代码


  在 JDK6 和 JDK7 中结果不一样:

  1、JDK6的执行结果:false false

  对于这个结果很好理解。在JDK6中,常量池在永久代分配内存,永久代和Java堆的内存是物理隔离的,执行intern方法时,如果常量池不存在该字符串,虚拟机会在常量池中复制该字符串,并返回引用,所以需要谨慎使用intern方法,避免常量池中字符串过多,导致性能变慢,甚至发生PermGen内存溢出。

2184951-ee42e5059da433e2.png

  2、JDK7的执行结果:true false

  对于这个结果就有点懵了。在JDK7中,常量池已经在Java堆上分配内存,执行intern方法时,如果常量池已经存在该字符串,则直接返回字符串引用,否则复制该字符串对象的引用到常量池中并返回,所以在JDK7中,可以重新考虑使用intern方法,减少String对象所占的内存空间。

2184951-0d48da18cea9606a.png


  对于变量s1,常量池中没有 "StringTest" 字符串,s1.intern() 和 s1都是指向Java对象上的String对象。

  对于变量s2,常量池中一开始就已经存在 "java" 字符串,所以 s2.intern() 返回常量池中 "java" 字符串的引用。

  String.intern()性能

  常量池底层使用StringTable数据结构保存字符串引用,实现和HashMap类似,根据字符串的hashcode定位到对应的数组,遍历链表查找字符串,当字符串比较多时,会降低查询效率。

  在JDK6中,由于常量池在PermGen中,受到内存大小的限制,不建议使用该方法。

  在JDK7、8中,可以通过-XX:StringTableSize参数StringTable大小,下面通过几个测试用例看看intern方法的性能。

  1.   public class StringTest {

  2.   public static void main(String[] args) {

  3.   System.out.println(cost(1000000));

  4.   }

  5.   public static long cost(int num) {

  6.   long start = System.currentTimeMillis();

  7.   for (int i = 0; i < num; i++) {

  8.   String.valueOf(i).intern();

  9.   }

  10.   return System.currentTimeMillis() - start;

  11.   }

  12.   }
复制代码


  执行一百万次intern()方法,不同StringTableSize的耗时情况如下:

  1、-XX:StringTableSize=1009, 平均耗时23000ms;

  2、-XX:StringTableSize=10009, 平均耗时2200ms;

  3、-XX:StringTableSize=100009, 平均耗时200ms;

  4、默认情况下,平均耗时400ms;

  在默认StringTableSize下,执行不同次intern()方法的耗时情况如下:

  1、一万次,平均耗时5ms;

  2、十万次,平均耗时25ms;

  3、五十万次,平均耗时130ms;

  4、一百万次,平均耗时400ms;

  5、五百万次,平均耗时5000ms;

  6、一千万次,平均耗时15000ms;

  从这些测试数据可以看出,尽管在Java 7以上对intern()做了细致的优化,但其耗时仍然很显著,如果无限制的使用intern()方法,将导致系统性能下降,不过可以将有限值的字符串放入常量池,提高内存利用率,所以intern()方法是一把双刃剑。

  END。

原文作者:占小狼   来源:开发者头条

相关帖子

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

关于我们
联系我们
  • 电话:010-86393388
  • 邮件:udn@yonyou.com
  • 地址:北京市海淀区北清路68号
移动客户端下载
关注我们
  • 微信公众号:yonyouudn
  • 扫描右侧二维码关注我们
  • 专注企业互联网的技术社区
版权所有:用友网络科技股份有限公司82041 京ICP备05007539号-11 京公网网备安1101080209224 Powered by Discuz!
快速回复 返回列表 返回顶部