最近工作中对系统进行调优,发现有一个后台任务执行后会导致logger打日志会变得很慢。经过层层分析,发现是任务代码中用到的阿里巴巴的fastjson框架中的SymbolTable对象中有大量String intern对象。
参考:http://tech.meituan.com/in_depth_understanding_string_intern.html
因为GC优化比较难搞,所以我们先从String.intern()的开销开始做。String.intern()的作用就是在JVM运行时往常量池去写东西,如果intern的String已经存在,返回已存在的地址,如果不存在,往常量池写一份,再返回常量池中的地址,来减少String的实例数。
StringTable的结构类似于一个HashMap,默认大小1009,当JVM中存在大量的String并且intern时,这个大小会导致严重的哈希冲突,接下来的结果相信大家都知道了,某些Entry的链表可能非常长,导致HashMap的性能下降非常多。
这个在JDK7或者淘宝的JDK6最新版中提供了一个启动参数-XX:StringTableSize=N来设定StringTable的大小,通过扩张Table来减少Hash冲突。
更新一步,实际上序列化/反序列化的过程中会大量调用String.intern(),主要是ClassName、MethodSignature、FieldName等等这些类信息,都会用String.intern()来保证每次用同一个实例。导致的后果就是Table太小,性能的指数下降。
jdk1.7可以设置-XX:StringTableSize参数
http://xmlandmore.blogspot.com/2013/05/understanding-string-table-size-in.html
相关推荐
【Java面试题】对String常量池的理解
string常量池和intern_韩雅茹Java系列2021.pdf
下面小编就为大家分享一篇java String源码和String常量池的全面解析,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
主要介绍了Java String 字符串常量池解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
在JDK6.0中,StringTable的长度是固定的,长度就是1009,因此如果放入String Pool中的String非常多,就会造成hash冲突,导致链表过长,当调用String#intern()时会需要到链表上一个一个找,从而导致性能大幅度下降;...
关于String类中常量池、字符串池的理解
主要介绍了java String类常量池分析及"equals"和"==”区别详细介绍的相关资料,需要的朋友可以参考下
常量池专门用来用来存放常量的内存区域,常量池分为:静态常量池和运行时常量池; 静态常量池:*.class文件中的常量池,class文件中的常量池不仅仅包含字符串,数值字面量,还包含类、方法的信息,占用class...
String对象常用的属性的方法String对象常用的属性的方法
java面试题目 1.String S=new String("xyz”);...第二种:如果String常量池中没有创建“xyz”,则会创建两个对象,一个对象的值是“xyz”,一个对象new String(“xyz”)。 String类被final修饰不可被继承。
String int 字符串常量池 包装类型 函数参数 值传递引用传递 的 内存分配例子——源码 public static void fun_ref (Ref_test ref_out){ Ref_test ref_in=new Ref_test(); ref_in.s1="in"; //ref_out.s1=...
Java中颜色的String和Color对象之间的互相转换
Java中的String池
此程序是向大家展示js脚本中string对象的基本应用
享元模式运用共享技术有效地支持大量细粒度对象的复用。系统只使用少量的对象,而这些对象都很相似,状态变化很小,可以实现对象的多次复用。
上图红色的这3个箭头,对于通过new产生一个字符串(”宜春”)时,会先去常量池中查找是否已经有了”宜春”对象,如果没有则在常量池中创建一个此字符串对象,然后堆中再创建一个常量池中此”宜春”对象的拷贝对象。...
string 、对象 、json之间相互互转 ,包含jar包测试类,做了简单的,能使而已!
//在java中有一个常量池,当创建String 类型的引用变量给它赋值时,java会到它的常量池中找"hello world"是不是在常量池中已存在。如果已经存在则返回这个常量池中的"hello world"的地址(在java中叫引用)给变量a 。...
主要介绍了深入探索Java常量池,涉及静态常量池和运行时常量池的介绍,常量池的好处,8种基本数据类型的包装类和常量池等相关内容,具有一定参考价值,需要的朋友可以了解下。
主要介绍了通过String.intern()方法浅谈堆中常量池,在JDK7之前,字符串常量是存在永久带Perm 区的,JDK7开始在将常量池迁移到堆中,这个变化也导致了String的新特性,下面我们慢慢进行介绍。,需要的朋友可以参考下