Java 1.7 中 String 内部表示的变化

发布于 2021-02-12 20:06:32 字数 5866 浏览 1272 评论 0

Java1.7 之前 String 的内部表示

Java1.7之前,String内部有4个非静态字段:

  • char[] value; //存放字符串,多个字符串可以共享
  • int offset; //value中第一个有效字符的索引
  • int count; //value中有效字符的数量
  • int hash; //缓存的 String hash 码

在很多情况下,一个 String 的 offset 总是为0,而 count 为 value.length。唯一例外的情况是,通过调用 String.substring 或内部使用该方法的API创建的字符串(比如,Pattern.split)。

String.substring 创建的字符串会共享原字符串内部的 char[] value,这样做的好处:

  1. 通过共享字符数据节省内存。
  2. String.substring 的时间复杂度为O(1)。

于此同时,这种特性可能是造成内存泄漏的罪魁祸首:如果你从一个巨大的字符串截取一个很小的子串并丢弃原字符串,你仍持有原字符串中巨大的char[] value的活引用。避免这种情况的唯一方式是在截取的字符串上调用 new String(String) 构造器创建一个新的String-该过程会拷贝底层char[]中需要的部分,这样就把子串和原始的长串分离了。

从Java 1.7.0_06,包括目前的Java 8版本,String类内部都移除了offset和count字段。这意味着你不能再共享底层的char[] value了!上面描述的内存泄漏也不会出现,你也不用通过new String(String)构造器重新创建新字符串了!遗憾的是,现在String.substring的时间复杂度变成线性的而不再是常量了!

hash 算法逻辑的变化

文章剩下的部分只适用于 Java 7u6+,Java 8 中该部分代码已移除

在本次更新中,String类还有一个变化:一个新的hash算法。Oracle建议使用新的hash算法,它能提供更好的hash码分布,这会改善许多基于hash的集合类的性能:HashMap,Hashtable,HashSet,LinkedHashMap,LinkedHashSet,WeakHashMap和ConcurrentHashMap。和文章第一部分讨论的改变不同,这个改变是试验性的,默认是关闭的。

如你所料,这些改变只针对String键。如果想要启用这个特性,你需要设置jdk.map.althashing.threshold系统参数为非负值(默认为-1)。这个值代表一个集合大小的阀值,超过这个值,就会使用新的hash算法。需要注意的是,只有rehashing(没有足够的可用空间)的过程新的hash算法才会起作用!所以,如果集合上次rehash时size=160,jdk.map.althashing.threshold=200,那当集合增长到大约320时才会使用新的hash算法。

现在的String类有个hash32()方法,它执行的结果缓存到int hash32字段中。hash32()方法最大的特点是,它在同一个String上执行的结果在不同的JVM运行过程中产生的值可能不一样(实际上,多数情况下都会不同,因为它使用一个System.currentTimeMills()和两个System.nanoTime()调用来初始化种子seed)。因此,每次运行程序时,同样的集合遍历顺序将会不同。

实际上,我对hash32()有点惊讶。既然原有的hashCode方法能够工作,为什么还需要这个方法呢?我决定运行hashCode方法性能优化指南文章中的测试程序来检测下hash32方法会产生多少重复的hash码。

String.hash32()方法不是public的,所以我需要查看HashMap的实现,找出如何调用该方法。答案是sun.misc.Hashing.stringHash32(String)。

使用相同的测试方法测试1百万不同的键,String.hash32产生304个重复的hash值,但String.hashCode却没有重复的。我认为,我们需要等Oracle做进一步的优化才可以使用。

Java 7u6 到 Java 7u40 版(不包括)的 JDK 中,新的hash算法可能会严重影响多线程代码

本段适用于 Java 7 build 6(包括)到 build 40(不包括),这些代码在Java8中已移除。查看下个段落可以了解到关于 Java 7u40+ 的信息

Oracle在以下类的hash算法实现中留下个bug:HashMap,Hashtable,HashSet,LinkedHashMap,LinkedHashSet和WeakHashMap。只有ConcurrentHashMap不受影响。原因在于所有的非并发类都有下面的字段:

/* A randomizing value associated with this instance that is applied to hash code of keys to make hash collisions harder to find. /

transient final int hashSeed = sun.misc.Hashing.randomHashSeed(this);

这意味着,每次创建map/set实例都会调用sun.misc.Hashing.randomHashSeed方法。相应地,randomHashSeed方法调用java.util.Random.nextInt方法。众所周知,Random类对多线程支持并不友好:它包含private final AtomicLong seed字段。在竞争小的情况下Atomics工作的很好,但竞争大的情况下性能极差。

结果,很多处理HTTP/JSON/XML请求的高负载web应用都会受到这个bug的影响,因为几乎所有知名的解析器都使用受影响的这几个集合类来表示"name-value"。所有这些解析器内部可能会创建嵌套的map,这又进一步增加每秒创建的map的数量。

如何解决该问题?

实时调用 randomHashSed

像 ConcurrentHashMap 那样:只有在系统属性 jdk.map.althashing.threshold 设置的时候才调用 randomHashSed 方法。不幸的是,只有 JDK 核心开发人员才能这么干。

/* A randomizing value associated with this instance that is
applied to hash code of keys to make hash collisions harder to find. /
private transient final int hashSeed = randomHashSeed(this);
private static int randomHashSeed(ConcurrentHashMap instance) {
  if (sun.misc.VM.isBooted() && Holder.ALTERNATIVE_HASHING) {
    return sun.misc.Hashing.randomHashSeed(instance);
  }
  return 0;
}

手动打补丁

修改 sun.misc.Hashing 类,不过不推荐,如果你想这么干,可以这样:java.util.Random类不是final的,你可以继承这个类,覆盖里面的nextInt()方法,返回thread-local的值(甚至可以是常量)。

更好的方式是,你可以使用Java 7的java.util.concurrent.ThreadLocalRandom,它是一个使用ThreadLocal的线程局部的Random子类(感谢BenjaminPossolo指出我在原来的文章中没有提到这个类)。除了是线程局部外,ThreadLocalRandom还是CPU缓存的(cache-aware):为了消除在一个缓存线结束时产生两个不同种子的机会,它在每个ThreadLocalRandom实例产生种子后添加64个字节填充(高速缓存大小)。

然后修改 sun.misc.Hashing.Holder.SEED_MAKER 这个字段,把它设置为你继承的Random类的实例。 虽然这个字段是private static final的,别担心,用反射来搞定:

public class Hashing {
private static class Holder {
static final java.util.Random SEED_MAKER;

Buddhist 方式

不要升级到 Java 7u6 和更高版本,检查 Java 7 的更新版本看是否修复了该 bug。

Java 7u40+ 中新的 hash 算法不再影响多线程代码的性能

Oracle在Java 7u40修复了上面提到的问题。他们采用上段的方式1解决的-只有在新的hash算法启用的情况下才调用sun.misc.Hashing.randomHashSeed(this)。Oracle只修复2个类:HashMap和Hashtable,但它也间接修复HashSet,LinkedHashMap和LinkedHashSet,因为这3个类是构建在HashMap之上的。唯一没有修复的类是WeakHashMap,但我无法想象积极创建该类实例的使用示例。

总结

  • 从Java 1.7.0_06 开始,String.substring总是为每个它产生的String实例创建一个新的底层char[] value。也就是说,和以前的常量复杂度相比,现在的时间复杂为线性。这个改变的好处是每个String占用的内存字节数减少了8字节,而且还避免了因为调用String.substring导致的内存泄漏(可以查看 String packing part 1: converting characters to bytes了解更多关于Java 对象内存模型的信息)。
  • 从Java 7u6版本开始(Java8已移除),String类有了第二个hash方法:hash32。这个方法目前不是public的,不使用反射的情况下,只能通过sun.misc.Hashing.stringHash32(String)调用。如果Jdk 7中基于hash的集合的大小超过jdk.map.althashing.threshold系统属性的值,该方法会被使用。这是试验性的功能,目前不建议你在自己的代码中使用。
  • Java 7u6(包括)和Java 7u40(不包括)之间的Java版本中,由于受新的hash算法实现的影响,所有标准的JDK非并发maps和sets都会产生性能问题。这个bug只会影响每秒创建大量map的多线程应用。

See also

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

JSmiles

生命进入颠沛而奔忙的本质状态,并将以不断告别和相遇的陈旧方式继续下去。

0 文章
0 评论
84960 人气
更多

推荐作者

文章 0 评论 0

云雾

文章 0 评论 0

夏尔

文章 0 评论 0

alipaysp_yxYxYl56FW

文章 0 评论 0

涙—继续流

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文