从Java的字符串池、常量池理解String的intern()
前言
逛知乎遇到一个刚学Java就会接触的字符串比较问题:

通常,根据"==比较的是地址,equals比较的是值"介个定理就能得到结果。但是String有些特殊,通过new String(string)生成的两个同值的字符串地址就不相等,用其他方式来生成的两个同值字符串地址就相等。
代码如下:
// 第一种方式创建字符串,字面量赋值
String str1 = "abc";
String str2 = "abc";
// 第二种方式创建字符串
String str3 = new String("xyz");
String str4 = new String("xyz");
System.out.println(str1 == str2); //true
System.out.println(str3 == str4); //false
同样是创建字符串,两对等值的字符串进行为什么结果不一样,这就涉及到了常量池和堆。
第一种方式创建的字符串,会将"abc"这个字面量放到了常量池中,然后str1和str2都指向常量池中的"abc",所以两个变量地址相同;第二种方式创建的字符串,是先在常量池中放入"xyz",然后通过构造函数将常量池中的"xyz"拷贝一份到堆中生成新的String,和常量池中的"xyx"就没有了关系,所以两个变量指向的是堆中两个不同的变量,所以两个变量地址不同。
那intern()又是啥?和常量池之间又有什么联系?
常量池
常量池是存放字面量、符号引用或直接引用的地方。而常量池又分为class常量池和运行时常量池。
class常量池
class常量池是存放编译期类中的字面量和符号引用。上面的字符串"abc"就是字面量;符号引用就是类和接口的完全限定名,字段的名称和描述符,方法的名称和描述符。
如图:

图中的就是new String(String)这个方法在常量池中的名称和描述符,即符号引用。
运行时常量池
我们平时说的常量池指的就是运行时常量池。在类加载的解析阶段,会将class常量池载入内存中(JDK1.7之前位于方法区,现在位于Heap中),并且将符号引用解析成直接引用,即根据对方法/类的描述信息指向内存中对应的方法/类。运行时常量池具有动态性,可以在运行期添加新的变量进入常量池。
intern()
先看一下intern()这个方法的描述:
用二级英文水平翻译一波,大意就是一个string调用intern()的时候,如果池中有和这个字符串值相等的字符串对象,就会将字符串池中的字符串对象返回;如果没有,就将这个字符串添加进去,并返回这个字符串的引用。字符串池由String类私有维护。
这里又引入了字符串池这个概念。
字符串池
字符串池存放的是常量池中字符串对象的引用,而不是字符串对象。通过第一种字面量赋值法创建的字符串会放在常量池中,字符串池就会存储这个字符串对象的引用,当再次在常量池创建字符串时,会先从字符串池查看是否有此字符串的等值引用,如果有的话,直接指向此引用对应的对象。
而第二种方式创建的字符串,会在字符串池中查找是否有与构造参数等值的字符串,以此决定是否需要在常量池新建字符串,然后拷贝常量池中字符串在Heap创建一个新的字符串。

如图,在堆中会在常量池中创建一个名为original的新字符串,然后拷贝并在堆中生成一个新字符串。注释中也提到,除非你需要一个字符串的显式副本,否则不需要使用这个构造函数,因为字符串是不可变的。
这里使用intern()测试一下字符串池:
public static void main(String[] args) {
//第一部分 测试
String str1 = "abc";
String str2 = new String("abc");
System.out.println(str1.intern() == str1); //true
System.out.println(str1.intern() == str2); //false
System.out.println(str1.intern() == str2.intern()); //true
//第二部分 测试通过char[]创建字符串后,引用是否会进入字符串池
String str3 = new String(new char[]{'g', 'h'});
String str4 = "gh";
System.out.println(str3.intern() == str3); //false
System.out.println(str3.intern() == str4); //true
//第三部分 测试char[]创建的字符串调用intern()后引用是否进入字符串池
String str3 = new String(new char[]{'g', 'h'});
str3.intern();
String str4 = "gh";
System.out.println(str3.intern() == str3); //true
System.out.println(str3.intern() == str4); //true
}
以上三部分代码是独立测试。
第一部分:str1在常量池创建了abc,并将引用放入字符串池,str2拷贝常量池中的abc并在堆中创建新字符串。intern()从字符串池中获取的是常量池中str1的abc引用。
第二部分:str3通过char[]在堆中创建了字符串,不是在常量池,所以gh的引用不会自动放入字符串池。str4在常量池创建了gh,所以字符串池中保存了str4的gh引用。intern()从字符串池中获取的是常量池中str4的gh引用。
第三部分:str3通过char[]在堆中创建了字符串,不是在常量池,所以gh的引用不会自动放入字符串池,但是它调用intern()手动将str3的gh的引用添加到了字符串池中。当str4使用字面量赋值创建时,查询到字符串池中有gh的引用,str4就指向了str3的gh引用。intern()从字符串池中获取的是堆中str3的gh引用。
从上面的代码中也得出结论:intern()可以将堆中创建的且字符串池没有等值引用的字符串引用放入字符串池。
同时,这也能说明String为什么不可变这个问题。
因为这样可以保证多个引用可以同时指向字符串池中的同一个对象。如果字符串是可变的,其中的一个引用操作改变了对象的值,对其他引用会有影响,这样显然是不可以的。
言归正传
回到知乎上的问题。在常量池创建了"string"并将其引用放入字符串池,str1调用intern()返回的是常量池中的引用,而str1指向的是堆中的引用,所以输出为false。
而StringBuilder的toString()是通过char[]创建字符串:

在堆中创建了abcdef之后,str2调用intern()将堆中引用放入字符串池并返回此引用,与str2指向堆中同一个字符串对象,所以输出为true。
结语
Java中有时候很小的问题也会发散出很多知识点,不论是底层还是JVM的理论学习,结合应用案例会理解的更加深刻。就像文中提到的常量池就是class文件结构和类加载理论学习的一部分。

从Java的字符串池、常量池理解String的intern()的更多相关文章
- 从字符串到常量池,一文看懂String类设计
从一道面试题开始 看到这个标题,你肯定以为我又要讲这道面试题了 // 这行代码创建了几个对象? String s3 = new String("1"); 是的,没错,我确实要从这里 ...
- Java提高篇之常量池
一.相关概念 1. 什么是常量 用final修饰的成员变量表示常量,值一旦给定就无法改变! final修饰的变量有三种:静态变量.实例变量和局部变量,分别表示三种类型的常量. 2. Class文件中的 ...
- Java基础系列2:深入理解String类
Java基础系列2:深入理解String类 String是Java中最为常用的数据类型之一,也是面试中比较常被问到的基础知识点,本篇就聊聊Java中的String.主要包括如下的五个内容: Strin ...
- 【JVM】Java 8 中的常量池、字符串池、包装类对象池
1 - 引言 2 - 常量池 2.1 你真的懂 Java的“字面量”和“常量”吗? 2.2 常量和静态/运行时常量池有什么关系?什么是常量池? 2.3 字节码下的常量池以及常量池的加载机制 2.4 是 ...
- Java中几种常量池的区分
转载自:https://tangxman.github.io/2015/07/27/the-difference-of-java-string-pool/ 在java的内存分配中,经常听到很多关于常量 ...
- java虚拟(一)--java内存区域和常量池概念
一.java运行时数据区 也可以称为java内存区域,和java内存模型不是一回事,不要弄混,这里基于jdk1.8之前 1.1.方法区 线程共享,类装载过程中产生的java.lang.Class对象保 ...
- Java内存中的常量池
1,java内存模型简介 <深入理解java虚拟机>里将java内存分为如下五个模块: 堆-堆是所有线程共享的,主要用来存储对象. 其中,堆可分为:新生代和老年代两块区域.使用NewRat ...
- String字符串针对常量池的优化
String对象是java语言中重要的数据类型,但是不是基本数据类型.相对于c语言的char java做了一些封装和延伸. 针对常量池的优化:当两个String拥有相同的值时,它们只引用常量池中的同一 ...
- Java基础二:常量池
目录: 自动装箱与拆箱 常量池 ==与equals()区别 1. 自动装箱与拆箱 Java是一个近乎纯洁的面向对象编程语言,但是为了编程的方便还是引入了基本数据类型,但是为了能够将这些基本数据类型当成 ...
随机推荐
- STL-Vector容量问题:
1.clear,erase ,pop_back() 函数只删除对象,并没有释放vec中的内存,若对象是指针还需要delete:2.在erase,clear,pop_back()删除对象的后,size改 ...
- 题解-CF1139D Steps to One
题面 CF1139D Steps to One 一个数列,每次随机选一个 \([1,m]\) 之间的数加在数列末尾,数列中所有数的 \(\gcd=1\) 时停止,求期望长度 \(\bmod 10^9+ ...
- hiveSQL和MySQL区别
1.hive支持按行分割,按字段分割,如按','分割: lateral view explode(split( , ',')) 2.hive不支持等值连接,即不支持where a.id = b.id的 ...
- Android Studio中SVN的使用
1.忽略文件 1)这种方式,每次新建一个项目都要添加,并不是全局的. .idea文件夹 .gradle文件夹 所有的build文件夹 所有的.iml文件 local.properties文件 2)使用 ...
- python协程需要注意的
python协程需要注意的点 都在注释里 # -*- coding: utf-8 -*- import asyncio import time from geeker import schedule ...
- 云原生网络代理(MOSN)的进化之路
本文系云原生应用最佳实践杭州站活动演讲稿整理.杭州站活动邀请了 Apache APISIX 项目 VP 温铭.又拍云平台开发部高级工程师莫红波.蚂蚁金服技术专家王发康.有赞中间件开发工程师张超,分享云 ...
- Spark 源码浅读-SparkSubmit
Spark 源码浅读-任务提交SparkSubmit main方法 main方法主要用于初始化日志,然后接着调用doSubmit方法. override def main(args: Array[St ...
- 工具-chrome相关-安装crx包及错误解决(99.3.1)
@ 目录 1.安装教程 2.程序包无效:"CRX_HEADER_INVALID" 1.安装教程 在浏览器上输入 chrome://extensions 并且选择开发者模式 将.cr ...
- Docker 部署 _实现每日情话 定时推送(apscheduler)
由于最近工作比较忙,后续博客可能更新不及时,哈哈 前言: 由于python对于微信推送不够友好,需要扫码登录,短信接口需要RMB.我就想到了qq邮箱发送到好友,然而微信有qq邮箱提醒功能,就实现了我需 ...
- python并发编程——多线程
编程的乐趣在于让程序越来越快,这里将给大家介绍一个种加快程序运行的的编程方式--多线程 1 著名的全局解释锁(GIL) 说起python并发编程,就不得不说著名的全局解释锁(GIL)了.有兴趣的同 ...