String常量池对象过多导致系统性能下降

本文探讨了使用FastJSON框架时遇到的性能问题,特别是由于SymbolTable中的大量String intern对象导致的日志记录延迟现象。文章深入分析了StringTable的工作原理,并提出了解决方案,包括调整JVM参数-XX:StringTableSize以优化内存分配。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 

      最近工作中对系统进行调优,发现有一个后台任务执行后会导致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 常量池详解 在 Java 中,String 是一个特殊的类,它不仅被广泛使用,还通过常量池机制进行了深度优化。为了提升性能和减少内存开销,JVM 引入了 **字符串常量池StringTable)** 来管理字符串字面量。 #### 字符串常量池的定义作用 字符串常量池是一种特殊的内存区域,用于存储字符串字面量(literal)和通过 `intern()` 方法添加的字符串对象。当使用双引号定义字符串时,JVM 会优先检查常量池中是否已存在该字符串,若存在则直接返回其引用,否则新建一个并放入池中[^2]。 例如: ```java String a = "hello"; String b = "hello"; ``` 变量 `a` 和 `b` 指向的是同一个字符串对象,因为它们都指向常量池中的同一个实例。 #### 字符串常量池的运行机制 在程序运行期间,字符串常量池会动态管理字符串实例。当使用 `new String("hello")` 创建字符串时,JVM 会先检查常量池中是否存在 `"hello"`,若不存在则将其加入常量池。随后,会在堆中创建一个新的 `String` 对象,并指向常量池中的字符串[^4]。 ```java String c = new String("hello"); ``` 上述代码中,`c` 是一个堆中的新对象,但它内部的字符数据仍然指向常量池中的 `"hello"`。 #### intern() 方法字符串池的扩展 Java 提供了 `String.intern()` 方法用于手动将字符串加入常量池。当调用该方法时,JVM 会查找常量池中是否存在相同 Unicode 编码的字符串,如果存在则返回其引用,否则将当前字符串加入常量池并返回其引用[^3]。 ```java String d = new String("world").intern(); String e = "world"; ``` 此时,`d` 和 `e` 指向的是同一个常量池中的字符串对象。 #### 常量池的分类关系 字符串常量池是 JVM 中多种常量池机制的一部分。在编译时,类的常量信息(如字符串字面量)会被存储在 **静态常量池** 中。类加载后,这些信息会被加载到 **运行时常量池** 中,而字符串常量池则是运行时常量池的一部分,专门用于管理字符串字面量和通过 `intern()` 方法加入的字符串[^1]。 例如,在编译后的 `.class` 文件中,以下代码: ```java String a = "a"; String b = "b"; String ab = "ab"; ``` 会被编译为常量池中的符号信息,在类加载后成为运行时常量池的一部分,并在运行时由 JVM 管理其在字符串常量池中的存在[^5]。 #### 总结 字符串常量池通过复用字符串对象,减少了内存的使用并提升了程序性能。它静态常量池、运行时常量池共同构成了 Java常量管理机制。开发者可以通过 `intern()` 方法显式控制字符串在常量池中的存在状态,从而优化内存使用。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值