Java String 的 常量池 和 intern 方法 简析

简介: Java String 的 常量池 和 intern 方法 简析字符串常量池和 intern 方法先举个例子,我们创建 10000 个相同的 String,并且不使用常量池String[] list = new String[10000];for (int i = 0; i < 10000; i++) { list[i] = new String(new char[]{'a','b','c'});}结果是:每个 String 都用新开的对象,占用大量内存现在我们这样创建,增加一个 map ,key 和 value 内容一样,每次取的时候,先检查一下 map 里面有没

字符串常量池和 intern 方法

先举个例子,我们创建 10000 个相同的 String,并且不使用常量池

1
2
3
4
String[] list = new String[10000];
for (int i = 0; i < 10000; i++) {
    list[i] = new String(newchar[]{'a','b','c'});
}

结果是:每个 String 都用新开的对象,占用大量内存

现在我们这样创建,增加一个 map ,key 和 value 内容一样,每次取的时候,先检查一下 map 里面有没有,有就取 map 里面的,没有就放进去。即,创建 10000 个 String,使用自建常量池。

1
2
3
4
5
6
7
String[] list = new String[10000];
Map<String, String> pool = new HashMap<>();
for (int i = 0; i < 10000; i++) {
    String s = new String(newchar[]{'a','b','c'});
    pool.putIfAbsent(s, s); // 等价于 if (pool.get(s) == null) pool.put(s, s);
    list[i] = pool.get(s);
}

结果很明显,数组中每一个元素都指向堆中的同一个元素,其他新创建的 String 都会在下一次 GC 被清空。

其实,这个 map 就是字符串常量池。不过,JVM 把这个功能用 C++重新 实现了,存放在堆区。

那常量池这么好,要怎么使用 JVM 里面的常量池呢?

  1. 用双引号创建的 String ,自动使用常量池,比如 String a = "test";
  2. 使用 String 的 intern 方法,使用常量池,比如
1
2
String s = new String(new char[]{'a','b','c'});
String intern = s.intern(); // 类似于上面的pool.putIfAbsent(s, s) 和 pool.get(s)

关于 intern 方法,JDK 文档这样写:当调用 intern 方法时,如果常量池(内置在 JVM 中的)中已经包含相同的字符串,则返回池中的字符串。否则,将此 String 对象添加到池中,并返回对该 String 对象的引用。

我们再用 intern 写一个存 10000 个字符串的代码,使用 JVM 常量池,结果和例子中的第二个代码一致

1
2
3
4
String[] list = new String[10000];
for (int i = 0; i < 10000; i++) {
    list[i] = new String(newchar[]{'a','b','c'}).intern();
}

或者使用双引号创建字符串,自动使用 JVM 常量池,结果和例子中的上面的代码一致

1
2
3
4
String[] list = new String[10000];
for (int i = 0; i < 10000; i++) {
    list[i] = "abc";
}

intern 方法的小特性

其实 intern 的使用并不复杂,上面的例子已经讲得非常清晰。

不过,如果你要研究茴香豆的茴字有几种写法的话,那么这里有一个细节要注意:

  • 当堆区先创建了一个 String,并使用常量池,如果常量池中没有,常量池会直接把这个 刚刚在堆区创建的那个 String 作为 value

这个细节有点拗口,不过也非常好理解,我们搬回上面例子中的自建常量池:

1
2
3
4
5
6
7
String[] list = new String[10000];
Map<String, String> pool = new HashMap<>();
for (int i = 0; i < 10000; i++) {
    String s = new String(newchar[]{'a','b','c'});
    pool.putIfAbsent(s, s);
    list[i] = pool.get(s);
}

这个特性的意思就是说,常量池里面的 pool.get(s) 总是会返回第一个加进去的 s ,而不是别的。如果你走一下上面的代码,就会发现这个非常正常。

我们再复习一下使用常量池的两个方法:

  1. 用双引号创建的 String ,自动使用常量池
  2. 使用 String 的 intern 方法,使用常量池

那么,现在,我们祭出网上流传很广的题目:

1
2
3
4
5
6
7
8
9
10
11
publicstaticvoidmain(String[] args){
    String s = new String("1");
    s.intern();
    String s2 = "1";
    System.out.println(s == s2);
    String s3 = new String("1") + new String("1");
    s3.intern();
    String s4 = "11";
    System.out.println(s3 == s4);
}

你可以用我上面讲的内容试一下答案。

答案是输出 false true,我再讲一次原理:

  1. 第二行,先执行括号中的 “1” ,在堆区创建了一个 String,我假设他的地址为 100 ,由于是双引号创建,自动使用常量池,设置常量池中 “1” 的 value 为 堆区 100 号
  2. 第二行,new String(xxx),在堆区又创建了一个 String,假设地址为 101,暂时没有使用常量池
  3. 第三行,很显然,s.intern() 会返回 100,但是并没有用变量接住 (谁知道网上出题的那个人脑子怎么想的呢)
  4. 第四行,双引号创建,自动使用常量池,返回的地址是 100
  5. 所以 s2 是 101,s 是 100,答案已经很明显了
  6. 第七行,两个”1”结合,堆区创建了一个 String s3,假设地址为 102
  7. 第八行,s3.intern() ,使用常量池,但是常量池里面没有 “11”,所以设置常量池的 11 的 value 为 地址 102
  8. 第九行,双引号创建自动使用常量池,所以 s4 地址为 102

好,再来第二题

1
2
3
4
5
6
7
8
9
10
11
publicstaticvoidmain(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);
}

答案是 false, false,我就不再细讲了,你可以自己推导一遍。如果还是不懂再重新看下上面的文章。

不适合用 intern 方法的情况

由于 JVM 里面的 C++写的 的 HashMap 设计并不像 JDK 的 HashMap 这么科学(超过链表负载链表转红黑树)。所以,如果你有几千万个不同的 String 要使用 intern 丢进常量池的话,那么,查找起来会非常慢。而且常量池也会变得非常大,所以,不建议丢太多不同的 String 进常量池

那硬是要丢可不可以呢,也是可以的,你可以扩大 JVM 的 -XX:StringTableSize 参数(jdk8 中默认为 60013),这个参数类似 JDK HashMap 的 initialCapacity。 但是这样子会占用更多的内存和 CPU。

相关文章
|
2月前
|
消息中间件 Java Kafka
在Java中实现分布式事务的常用框架和方法
总之,选择合适的分布式事务框架和方法需要综合考虑业务需求、性能、复杂度等因素。不同的框架和方法都有其特点和适用场景,需要根据具体情况进行评估和选择。同时,随着技术的不断发展,分布式事务的解决方案也在不断更新和完善,以更好地满足业务的需求。你还可以进一步深入研究和了解这些框架和方法,以便在实际应用中更好地实现分布式事务管理。
|
1月前
|
存储 JavaScript Java
Java 中的 String Pool 简介
本文介绍了 Java 中 String 对象及其存储机制 String Pool 的基本概念,包括字符串引用、构造方法中的内存分配、字符串文字与对象的区别、手工引用、垃圾清理、性能优化,以及 Java 9 中的压缩字符串特性。文章详细解析了 String 对象的初始化、内存使用及优化方法,帮助开发者更好地理解和使用 Java 中的字符串。
Java 中的 String Pool 简介
|
1月前
|
缓存 安全 Java
java 为什么 String 在 java 中是不可变的?
本文探讨了Java中String为何设计为不可变类型,从字符串池的高效利用、哈希码缓存、支持其他对象的安全使用、增强安全性以及线程安全等方面阐述了不可变性的优势。文中还通过具体代码示例解释了这些优点的实际应用。
java 为什么 String 在 java 中是不可变的?
|
1月前
|
存储 Java
Java 11 的String是如何优化存储的?
本文介绍了Java中字符串存储优化的原理和实现。通过判断字符串是否全为拉丁字符,使用`byte`代替`char`存储,以节省空间。具体实现涉及`compress`和`toBytes`方法,前者用于尝试压缩字符串,后者则按常规方式存储。代码示例展示了如何根据配置决定使用哪种存储方式。
|
2月前
|
安全 Java 开发者
Java中WAIT和NOTIFY方法必须在同步块中调用的原因
在Java多线程编程中,`wait()`和`notify()`方法是实现线程间协作的关键。这两个方法必须在同步块或同步方法中调用,这一要求背后有着深刻的原因。本文将深入探讨为什么`wait()`和`notify()`方法必须在同步块中调用,以及这一机制如何确保线程安全和避免死锁。
52 4
|
2月前
|
Java
深入探讨Java中的中断机制:INTERRUPTED和ISINTERRUPTED方法详解
在Java多线程编程中,中断机制是协调线程行为的重要手段。了解和正确使用中断机制对于编写高效、可靠的并发程序至关重要。本文将深入探讨Java中的`Thread.interrupted()`和`Thread.isInterrupted()`方法的区别及其应用场景。
72 4
|
2月前
|
Java 数据处理 数据安全/隐私保护
Java处理数据接口方法
Java处理数据接口方法
30 1
|
4月前
|
Java 索引
java基础(13)String类
本文介绍了Java中String类的多种操作方法,包括字符串拼接、获取长度、去除空格、替换、截取、分割、比较和查找字符等。
50 0
java基础(13)String类
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
69 2
|
3月前
|
Java
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
本文深入探讨了Java中方法参数的传递机制,包括值传递和引用传递的区别,以及String类对象的不可变性。通过详细讲解和示例代码,帮助读者理解参数传递的内部原理,并掌握在实际编程中正确处理参数传递的方法。关键词:Java, 方法参数传递, 值传递, 引用传递, String不可变性。
75 1
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性