简介

String.intern()方法是一种手动将字符串加入常量池中的native方法,原理如下:如果在当前类的常量池中存在与调用intern()方法的字符串等值的字符串,就直接返回常量池中相应字符串的引用,否则在常量池中复制一份该字符串(Jdk7中会直接在常量池中保存当前字符串的引用),并将其引用返回;因此,只要是堆中等值的String对象,使用intern()方法返回的都是常量池中同一个String引用,所以,这些等值的String对象通过intern()后使用==是可以匹配的。

预备知识

一、创建String对象的两种方式

创建String对象分为在堆中创建和在字符串常量池中创建。

1、在字符串常量池中创建String对象

String str1 = "计算机";
String str2 = "计算机";
System.out.println(str1 == str2);

第一行代码

  • 执行到括号中“计算机”的时候,会在字符串常量池中添加“计算机”对象。
  • 然后将栈中的str1变量指向该对象。

第二行代码

  • 执行到括号中“计算机”的时候,发现字符串常量池已存在“计算机”对象。
  • 为避免重复创建对象,将栈中的str2变量指向字符串常量池已存在的对象,字符串常量池存在的原因正是如此。

补充:

//常量字符串的"+"操作,编译阶段直接会合成为一个字符串。
String str = "计算" + "机"; //编译时合并成String str = "计算机";//对于final字段,编译期直接进行了常量替换。
final String str1 = "计算";
final String str2 = "机";
String str3 = str1 + str2; //编译时直接替换成了String str3 = "计算" + "机";

2、在堆中创建String对象

String str1 = new String("计算机");
String str2 = new String("计算机");
System.out.println(str1 == str2);

第一行代码

  • 执行到括号中“计算机”的时候,会在字符串常量池中添加“计算机”对象。
  • 执行完毕后会在堆中创建“计算机”对象,并将栈中的str1变量指向该对象。

第二行代码

  • 执行到括号中“计算机”的时候,发现字符串常量池已存在“计算机”对象。
  • 执行完毕后会因为new关键字在堆中创建出一个新的对象,并将栈中的变量str2指向该对象。

二、不同jdk版本中的intern()实现

jvm对字符串常量池在不同的jdk版本有不同的划分,内存的变化也会影响intern方法的执行。

JDK6 JDK7及之后
intern()方法 会把首次遇到的字符串实例复制到常量池中,并返回此引用 会把首次遇到的字符串实例的引用添加到常量池中,并返回此引用
  • Jdk6中常量池位于PermGen(永久代)中,PermGen是一块主要用于存放已加载的类信息和字符串池的大小固定的区域。Jdk6中使用intern()方法的主要问题就在于常量池被保存在PermGen中:首先,PermGen是一块大小固定的区域,一般不同的平台PermGen的默认大小也不相同,大致在32M到96M之间。所以不能对不受控制的运行时字符串(如用户输入信息等)使用intern()方法,否则很有可能会引发PermGen内存溢出;其次String对象保存在Java堆区,Java堆区与PermGen是物理隔离的,因此如果对多个不等值的字符串对象执行intern操作,则会导致内存中存在许多重复的字符串,会造成性能损失。
  • Jdk7将常量池从PermGen区移到了Java堆区,执行intern操作时,如果常量池已经存在该字符串,则直接返回字符串引用,否则复制该字符串对象的引用到常量池中并返回。堆区的大小一般不受限,所以将常量池从PremGen区移到堆区使得常量池的使用不再受限于固定大小。除此之外,位于堆区的常量池中的对象可以被垃圾回收。当常量池中的字符串不再存在指向它的引用时,JVM就会回收该字符串。

实例解析

代码一

String s1 = new String("1"); //常量池中创建"1",堆中创建"1"
s1.intern();          //常量池中已有"1",所以jdk6和jkd7都是返回指向常量池"1"的引用,//但因为该语句没有赋值操作,所以s1仍指向堆中"1"
String s2 = "1";      //s2指向常量池中已存在的"1"
System.out.println(s1==s2);  //s1指向堆中"1",s2指向常量池中"1",false

jdk6、7、8都返回false

代码二

String s3 = new String("1") + new String("1"); //常量池生成一个"1",堆生成一个"11"//s3指向堆中"11"//中间还有2个匿名的new String("1")暂不讨论
s3.intern();             //因为常量池中不存在"11",//jdk6会将堆中"11"复制到常量池中,//jdk7则将堆中"11"的引用添加到常量池中,//此时s3仍指向堆中"11"
String s4 = "11";        //因为常量池中已存在"11"或其引用,s4指向常量池中"11"
System.out.println(s3==s4);  //jdk6中,s4指向常量池中"11",s3指向堆中"11",false//jdk7中,s4指向常量池中指向堆中"11"的引用,true

在jdk6中返回false,jdk7及以上返回true

适用场景

Jdk6 中常量池位于PremGen区,大小受限,不建议使用String.intern()方法,不过Jdk7 将常量池移到了Java堆区,大小可控,可以重新考虑使用String.intern()方法。

intern()方法优点:执行速度非常快,直接使用==进行比较要比使用equals()方法快很多;内存占用少。虽然intern()方法的优点看上去很诱人,但由于intern()操作每次都需要与常量池中的数据进行比较以查看常量池中是否存在等值数据,同时JVM需要确保常量池中的数据的唯一性,这就涉及到加锁机制,这些操作都是有需要占用CPU时间的,所以如果进行intern操作的是大量不会被重复利用的String的话,则有点得不偿失。由此可见,String.intern()主要 适用于只有有限值,并且这些有限值会被重复利用的场景,如数据库表中的列名、人的姓氏、编码类型等。

String的intern()方法浅析相关推荐

  1. Java之String系列--intern方法的作用及原理

    原文网址:Java之String系列--intern方法的作用及原理_IT利刃出鞘的博客-CSDN博客 简介 本文介绍Java的String的intern方法的原理. 常量池简介 在 JAVA 语言中 ...

  2. Java String的intern方法

    1. 首先String不属于8种基本数据类型,String是一个对象. 因为对象的默认值是null,所以String的默认值也是null:但它又是一种特殊的对象,有其它对象没有的一些特性. 2. ne ...

  3. [转]String 之 new String()和 intern()方法深入分析

    引入 String,是 Java 中除了基本数据类型以外,最为重要的一个类型了.很多人会认为他比较简单.但是和 String 有关的面试题有很多,下面我随便找两道面试题,看看你能不能都答对: Q1:S ...

  4. String中intern()方法

    本文参考 https://blog.csdn.net/believesoul/article/details/79588305 ,我对这篇文章进行了一些小小的整理和修改.这篇文章很好的对intern( ...

  5. String的intern方法演示及各种字符串的拼接对比

    演示String的intern方法,以及各种拼接字符串的区别 package string;/*** @Author gzx* @create 2022-1-14 jdk8*/ public clas ...

  6. String中intern方法的作用

    前言 读完这篇文章你可以了解,String对象在虚拟机内存中的存放,intern的作用,这么多String对象的创建到底有什么区别,String 创建的对象有几个!! 进入正题 先科普几个知识点 1. ...

  7. java intern_java String的intern方法

    我们知道再jvm的运行时内存可以分为堆.方法区.程序计数器.虚拟机栈和本地方法栈.而在方法区中有一个字符串常量池,用来保存字符串这个不可变量.如果我们使用String str=new String(& ...

  8. 字符斜杠是合法常量吗_【面试秘籍】你对String的intern方法了解吗

    我们先来看个例子: public class StringTest { public static void main(String[] args) { String a = "A" ...

  9. JDK8中String的intern()方法详细解读【内存图解+多种例子+1.1w字长文】

    文章目录 一.前言 二.图文理解String创建对象 1.例子一 2.例子二 3.例子三 4.例子四 5.例子五 6.例子六 三.深入理解intern()方法 1. 源码查看 2. 例子一 3. 例子 ...

  10. java String的intern()方法

    本文已参与「新人创作礼」活动,一起开启掘金创作之路. 本文章所用jdk版本为jdk1.8 先看第一个例子 public class HelloWorld {public static void mai ...

最新文章

  1. 【30集iCore3_ADP出厂源代码(ARM部分)讲解视频】30-10底层驱动之I2C
  2. arcgis Manager 登陆时提示Incorrect login Information
  3. jquery可编辑表格(版本二)
  4. c# streamReader转XmlDocument读取节点
  5. zookeeper 屁民
  6. 边缘计算如何实现海量IoT数据就地处理
  7. android 多线程断点续传下载 三
  8. python拼图游戏代码的理解_有意思的JS(1)拼图游戏 玩法介绍及其代码实现
  9. Node.js:Webpack
  10. Angular动态注册组件(controller,service...)
  11. p750服务器配置信息,P750的一些配置
  12. photoshop cc2019快捷键
  13. 十三年来,淘宝走过的大数据之路
  14. 【环境搭建】机械革命 Z3 Air 双系统 原生Windows10 +Ubuntu 20.04.2.0 LTS 桌面版 配置
  15. 万字阐述智能驾驶汽车安全体系
  16. 使用物联网卡发送短信
  17. 1000元购物卷买票
  18. horizontal sagittal and coronal planes (水平面,矢状面,冠状面)
  19. C/C++仓库管理系统
  20. 金蝶kis标准版如何修改服务器地址,金蝶kis怎么设置服务器地址

热门文章

  1. MySQL中的自增主键用完了怎么办?
  2. CSDN怎么自动生成目录
  3. 洛谷P3398 仓鼠找suger
  4. 人工智能前沿——AI技术在医疗领域的应用
  5. [渝粤教育] 宁波财经学院 金融工程学 参考 资料
  6. java实现图片拼接
  7. 商汤科技面试——实习面试案例总结
  8. 【读书笔记】--少有人走的路①:心智成熟的旅程
  9. DEL: 华为无线modem变无线路由器 2
  10. 一个月薪 12000 的北京程序员的真实生活