Loading [MathJax]/jax/input/TeX/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Java的intern()函数和字符串常量池

Java的intern()函数和字符串常量池

作者头像
用户7886150
修改于 2021-04-06 02:44:24
修改于 2021-04-06 02:44:24
6670
举报
文章被收录于专栏:bit哲学院bit哲学院

参考链接: Java字符串之-intern()

// ==与equals的区别:  // ==:  // 1、比较的是操作符两端的操作数是否是同一个对象  // 2、两边的操作数必须是同一类型的(可以是父子类之间)才能编译通过。  // 3、比较的是地址,如果是具体的阿拉伯数字的比较,值相等则为TRUE,如:  // int a=10 与 long b=10L 与 double c=10.0都是相同的(为true),因为他们都指向地址为10的堆  // equals:  // 1、比较的是两个对象的内容是否一样  String s=”abce”是一种非常特殊的形式,和new 有本质的区别.  *  * 它是java中唯一不需要new 就可以产生对象的途径.  *  * 以 String s=”abce”;形式赋值在java中叫直接量,它是在常量池中而不是象new 一样放在压缩堆中.  * 这种形式的字符串,在JVM内部发生字符串拘留,即当声明这样的一个字符串后,JVM会在常量池中先查找有有没有一个值为”abcd”的对象,如果有,就会把它赋给当前引用.即原来那个引用和现在这个引用指点向了同一对象,如果没有,则在常量池中新创建一个”abcd”,下一次如果有String  * s1 = “abcd”;又会将s1指向”abcd”这个对象,即以这形式声明的字符串,只要值相等,任何多个引用都指向同一对象. 

public class Test1 {

 public static void main(String[] args) {

  String a = new String("ab"); // a 为一个引用

  String b = new String("ab"); // b为另一个引用,对象的内容一样

  String aa = "ab"; // 放在常量池中

  String bb = "ab"; // 从常量池中查找

  if (aa == bb) // true

   System.out.println("aa==bb");

  if (a == b) // false

   System.out.println("a==b");

  if (a.equals(b)) // true

   System.out.println("aEQb");

  if (42 == 42.0)  // true

   System.out.println("true");

 }

也可以这么理解:  String str = “hello”; 先在内存中(常量池中)找是不是有”hello” 这个对象,如果有,就让str指向那个”hello”.如果内存里没有”hello”,就创建一个新的对象保存”hello”。  String str=new String (“hello”)  就是不管内存里是不是已经有”hello”这个对象,都新建一个对象保存”hello”。  String a = new String(“ab”);是在堆里面建立的对象 String,a和”ab”  aa=”ab”;是查找栈里有没有”ab”,如果有就用aa引用它,如果没有就把“ab”存进栈 

String str1 = new StringBuilder("计算机").append("软件").toString();

System.out.println(str1.intern() == str1);

String str2 = new StringBuilder("ja").append("va").toString();

System.out.println(str2.intern() == str2); 

这段代码在JDK1.6中运行,会得到两个false,而在JDK1.7中运行,会得到一个true和一个false。产生差异的原因是:在JDK1.6中,intern()方法会把首次遇到的字符串实例复制到永久代(常量池)中,返回的也是永久代中这个字符串实例的引用,而由StringBuilder创建的字符串实例在Java堆上,所以必然不是同一个引用,将返回false。而JDK1.7(以及部分其他虚拟机,例如JRockit)的intern()实现不会再复制实例,只是在常量池中记录首次出现的实例引用, 因此intern()返回的引用和有StringBuilder创建的那个字符串实例是同一个。对str2比较返回false是因为”java”这个字符串在执行StringBuilder.toString()之前已经出现过,字符串常量池中已经有它的引用,不符合“首次出现”的原则,而“计算机软件”这个字符串则是首次出现的,因此返回true。

本文系转载,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文系转载,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
字符串常量池_字符串常量池溢出
我们知道字符串的分配和其他对象分配一样,是需要消耗高昂的时间和空间的,而且字符串我们使用的非常多。JVM为了提高性能和减少内存的开销,在实例化字符串的时候进行了一些优化:使用字符串常量池。
全栈程序员站长
2022/09/19
6930
字符串常量池_字符串常量池溢出
深入理解字符串常量池
在JVM中,为了减少字符串对象的重复创建,维护了一块特殊的内存空间,这块内存就被称为字符串常量池。
烟雨星空
2020/06/16
1.4K0
字符串常量池理解「建议收藏」
在JVM中,为了减少字符串对象的重复创建,维护了一块特殊的内存空间,这块内存就被称为字符串常量池。
全栈程序员站长
2022/09/19
7090
字符串常量池理解「建议收藏」
JVM - 深入剖析字符串常量池
看 1.8 , 疯狂的intern, 抛出了 heap oom ,由此可以推断出 1.8中的字符串常量池 是在堆中。
小小工匠
2021/08/17
6530
彻底弄懂字符串常量池等相关问题
  在平时我们使用字符串一般就是拿来直接搞起,很少有深入的去想过这方面的知识,导致别人在考我们的时候,会问 String str = new String("123"); 这个一行代码执行创建了几个对象, String str1= str + new String("456");这行代码中str1存储在内存的哪个位置,堆or 字符串常量区(方法区)? 会把我们问的哑口无言了;哈哈哈哈,其实也不是水平问题,是我们平时可以仔细的去总结该类问题,下面就详细的对这类问题进行总结;
小勇DW3
2018/08/30
7030
常量池之字符串常量池String.intern()
运行时常量池是方法区(PermGen)的一部分。 需要提前了解: 1. JVM内存模型。 2. JAVA对象在JVM中内存分配 常量池的好处 常量池是为了避免频繁的创建和销毁对象而影响系统性能,其实现了对象的共享。 Java的自动装箱中其实就使用到了运行时常量池。详见:Java 自动装箱与拆箱的实现原理 还有字符串常量池。 字符串进入到常量池的两种方法: 1. new String()的实例调用intern()方法。     执行intern()方法时,若常量池中不存在等值的字符串,JVM就会在常
java404
2018/05/18
1.1K0
jvm字符串常量池_java 常量池
字符串对象:比如new String(“abc”),或者直接String s=”str”,后面的”str”也是一个字符串对象。
全栈程序员站长
2022/09/19
6000
jvm字符串常量池_java 常量池
jvm常量池和字符串常量池_常量池中的字符串是对象吗
在Java开发中不管是前后端交互的JSON串,还是数据库中的数据存储,我们常常需要使用到String类型的字符串。作为最常用也是最基础的引用数据类型,JVM为String提供了字符串常量池来提高性能,本篇文章我们一起从底层JVM中认识并学习字符串常量池的概念和设计原理。
全栈程序员站长
2022/09/19
6500
jvm常量池和字符串常量池_常量池中的字符串是对象吗
常量池和堆的区别_字符串常量池在堆中还是方法区
目录: 1.常量池与Class常量池 2.运行时常量池 运行时常量池的简介 方法区的Class文件信息,Class常量池和运行时常量池的三者关系 3.字符串常量池 字符串常量池的简介 采用字面值的方式创建字符串对象 采用new关键字新建一个字符串对象 字符串池的优缺点 4.字符串常量池和运行时常量池之间的藕断丝连 常量池和字符串常量池的版本变化 String.intern在JDK6和JDK7之后的区别(重难点) 字符串常量池里存放的是引用还是字面量
全栈程序员站长
2022/09/19
1.4K0
常量池和堆的区别_字符串常量池在堆中还是方法区
String:字符串常量池
作为最基础的引用数据类型,Java 设计者为 String 提供了字符串常量池以提高其性能,那么字符串常量池的具体原理是什么,我们带着以下三个问题,去理解字符串常量池:
全栈程序员站长
2021/06/17
7720
String:字符串常量池
Java字符串常量池_字符串常量池溢出
Java 常量池详解(二)class文件常量池 和 Java 常量池详解(三)class运行时常量池
全栈程序员站长
2022/09/19
1.2K0
Java字符串常量池_字符串常量池溢出
JVM之字符串常量池
StringTable为什么要调整 ①永久代permSize默认比较小; ②永久代的垃圾回收频率低;
程序员阿杜
2021/07/05
3280
JVM之字符串常量池
Java中的字符串常量池
Java中字符串对象创建有两种形式,一种为字面量形式,如String str = "droid";,另一种就是使用new这种标准的构造对象的方法,如String str = new String("droid");,这两种方式我们在代码编写时都经常使用,尤其是字面量的方式。然而这两种实现其实存在着一些性能和内存占用的差别。这一切都是源于JVM为了减少字符串对象的重复创建,其维护了一个特殊的内存,这段内存被成为字符串常量池或者字符串字面量池。
技术小黑屋
2018/09/05
1.3K0
什么是字符串常量池_常量池中的字符串是对象吗
JDK1.8-1.9,String底层从char数组变成了byte数组,原因是部分字符仅占一个byte,而堆中含有大量的String字符串,该优化能节省较多空间。
全栈程序员站长
2022/09/19
5910
什么是字符串常量池_常量池中的字符串是对象吗
java8以后字符串常量池的位置,以及元空间的探秘,使用VisualVM进行实战验证
  在网上看了很多博客,解释也比较多,关于字符串常量池的具体位置难以分辨谁真谁假。
全栈程序员站长
2022/09/19
1.7K0
java8以后字符串常量池的位置,以及元空间的探秘,使用VisualVM进行实战验证
Jvm常量池、运行时常量池、字符串常量池理解
是.class文件的常量池,也可以理解为一张表,虚拟机指令根据这张常量表找到要执行的类名,方法名,参数类型,字面量等信息
gang_luo
2020/08/13
2K0
Jvm常量池、运行时常量池、字符串常量池理解
从String的intern()到常量池
通常,根据"==比较的是地址,equals比较的是值"介个定理就能得到结果。但是String有些特殊,通过new String(string)生成的两个同值的字符串地址就不相等,用其他方式来生成的两个同值字符串地址就相等。
叫我阿柒啊
2022/05/09
3150
从String的intern()到常量池
String字符串—详细总结
不可变的,每一次修改实际上生成新的字符串,并且该字符串的值是修改后的值。new String都是在堆上创建字符串对象
用户5325874
2020/01/16
4780
String字符串—详细总结
String类和常量池内存分析例子以及8种基本类型
       如果运行时常量池中已经包含一个等于此 String 对象内容的字符串,则直接返回常量池中该字符串的引用;如果没有, 那么
砖业洋__
2023/05/06
2210
再议String-字符串常量池与String.intern()
来源:blog.csdn.net/gcoder_/article/details/106644312
Java小咖秀
2021/08/05
3800
再议String-字符串常量池与String.intern()
相关推荐
字符串常量池_字符串常量池溢出
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档