首页 > 其他分享 >String 类和常量池

String 类和常量池

时间:2024-01-24 23:12:28浏览次数:19  
标签:String 对象 池中 intern 字符串 常量

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

String str1 = "abcd";
String str2 = new String("abcd");
System.out.println(str1==str2); //false

这两种不同的创建方法是有差别的:

第一种方式是在常量池中获取对象("abcd" 属于字符串字面量,因此编译时期会在常量池中创建一个字符串对象);

第二种方式一共会创建两个字符串对象(前提是 String Pool 中还没有 "abcd" 字符串对象)。

  • "abcd" 属于字符串字面量,因此编译时期会在常量池中创建一个字符串对象,该字符串对象指向这个 "abcd" 字符串字面量;
  • 使用 new 的方式会在堆中创建一个字符串对象。

str1 指向常量池中的 “abcd”,而 str2 指向堆中的字符串对象。

2、intern() 方法

intern() 方法设计的初衷,就是重用 String 对象,以节省内存消耗。

JDK6:当调用intern方法的时候,如果字符串常量池先前已创建出该字符串对象,则返回常量池中的该字符串的引用。否则,将此字符串对象添加到字符串常量池中,并且返回该字符串对象的引用。

JDK6+:当调用intern方法的时候,如果字符串常量池先前已创建出该字符串对象,则返回常量池中的该字符串的引用。否则,如果该字符串对象已存在与Java堆中,则将堆中对此对象的引用添加到字符串常量池中,并且返回该引用;如果堆中不存在,则在常量池中创建该字符串并返回其引用。

在 JVM 运行时数据区中的方法区有一个常量池,但是发现在 JDK 1.6 以后常量池被放置在了堆空间,因此常量池位置的不同影响到了 String 的 intern() 方法的表现。

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);

JDK 1.6 及以下

  • 上述代码输出结果:
false
false
  • 解释:

在 JDK 1.6 中所有的输出结果都是 false,因为 JDK 1.6 以及以前版本中,常量池是放在 PermGen 区(属于方法区)中的,而方法区和堆区是完全分开的。

使用引号声明的字符串会直接在字符串常量池中生成的,而 new 出来的 String 对象是放在堆空间中的。所以两者的内存地址肯定是不相同的,即使调用了 intern() 方法也是不影响的。

intern() 方法在 JDK 1.6 中的作用:比如 String s = new String("1");,再调用 s.intern(),此时返回值还是字符串"1",表面上看起来好像这个方法没什么用处。但实际上,在 JDK1.6 中:检查字符串常量池里是否存在 "1" 这么一个字符串,如果存在,就返回池里的字符串;如果不存在,该方法会把 "1" 添加到字符串常量池中,然后再返回它的引用

JDK 1.6 及以上

  • 上述代码输出结果:
false
true
  • 解释:

String s= new String("1") 生成了字符串常量池中的 "1" 和堆空间中的字符串对象。

s.intern() s 对象去字符串常量池中寻找后,发现 "1" 已存在于常量池中。

String s2 = "1" 生成 s2 的引用指向常量池中的 "1" 对象。

显然,s 和 s2 的引用地址是不同的。

String s3 = new String("1") + new String("1") 在字符串常量池中生成 "1",并在堆空间中生成 s3 引用指向的对象(内容为 "11")。 注意此时常量池中是没有 "11" 对象

s3.intern()将 s3 中的 "11" 字符串放入字符串常量池中。 JDK 1.6 的做法是直接在常量池中生成一个 "11" 的对象。但在 JDK 1.7 中,常量池中不需要再存储一份对象了,可以直接存储堆中的引用。这份引用直接指向 s3 引用的对象,也就是说 s3.intern() == s3 会返回 true。

String s4 = "11", 这一行代码会直接去常量池中创建,但是发现已经有这个对象了,此时 s4 就是指向 s3 引用对象的一个引用。因此 s3 == s4 返回了true。

3、字符串拼接

String str1 = "str";
String str2 = "ing";
		  
String str3 = "str" + "ing";//常量池中的对象
String str4 = str1 + str2; //TODO:在堆上创建的新的对象	  
String str5 = "string";//常量池中的对象
System.out.println(str3 == str4);//false
System.out.println(str3 == str5);//true
System.out.println(str4 == str5);//false

注意:尽量避免多个字符串拼接,因为这样会重新创建对象。 如果需要改变字符串的话,可以使用 StringBuilder 或者 StringBuffer

面试题:String s1 = new String("abc");问创建了几个对象?

创建2个字符串对象(前提是 String Pool 中还没有 "abcd" 字符串对象)。

  • "abc" 属于字符串字面量,因此编译时期会在常量池中创建一个字符串对象,指向这个 "abcd" 字符串字面量;
  • 使用 new 的方式会在堆中创建一个字符串对象。

(字符串常量"abc"在编译期就已经确定放入常量池,而 Java 堆上的"abc"是在运行期初始化阶段才确定)。

String s1 = new String("abc");// 堆内存的地址值
String s2 = "abc";
System.out.println(s1 == s2);// 输出false
//因为一个是堆内存,一个是常量池的内存,故两者是不同的。
System.out.println(s1.equals(s2));// 输出true

String 类和常量池

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

String str1 = "abcd";
String str2 = new String("abcd");
System.out.println(str1==str2); //false

这两种不同的创建方法是有差别的:

第一种方式是在常量池中获取对象("abcd" 属于字符串字面量,因此编译时期会在常量池中创建一个字符串对象);

第二种方式一共会创建两个字符串对象(前提是 String Pool 中还没有 "abcd" 字符串对象)。

  • "abcd" 属于字符串字面量,因此编译时期会在常量池中创建一个字符串对象,该字符串对象指向这个 "abcd" 字符串字面量;
  • 使用 new 的方式会在堆中创建一个字符串对象。

str1 指向常量池中的 “abcd”,而 str2 指向堆中的字符串对象。

2、intern() 方法

intern() 方法设计的初衷,就是重用 String 对象,以节省内存消耗。

JDK6:当调用intern方法的时候,如果字符串常量池先前已创建出该字符串对象,则返回常量池中的该字符串的引用。否则,将此字符串对象添加到字符串常量池中,并且返回该字符串对象的引用。

JDK6+:当调用intern方法的时候,如果字符串常量池先前已创建出该字符串对象,则返回常量池中的该字符串的引用。否则,如果该字符串对象已存在与Java堆中,则将堆中对此对象的引用添加到字符串常量池中,并且返回该引用;如果堆中不存在,则在常量池中创建该字符串并返回其引用。

在 JVM 运行时数据区中的方法区有一个常量池,但是发现在 JDK 1.6 以后常量池被放置在了堆空间,因此常量池位置的不同影响到了 String 的 intern() 方法的表现。

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);

JDK 1.6 及以下

  • 上述代码输出结果:
false
false
  • 解释:

在 JDK 1.6 中所有的输出结果都是 false,因为 JDK 1.6 以及以前版本中,常量池是放在 PermGen 区(属于方法区)中的,而方法区和堆区是完全分开的。

使用引号声明的字符串会直接在字符串常量池中生成的,而 new 出来的 String 对象是放在堆空间中的。所以两者的内存地址肯定是不相同的,即使调用了 intern() 方法也是不影响的。

intern() 方法在 JDK 1.6 中的作用:比如 String s = new String("1");,再调用 s.intern(),此时返回值还是字符串"1",表面上看起来好像这个方法没什么用处。但实际上,在 JDK1.6 中:检查字符串常量池里是否存在 "1" 这么一个字符串,如果存在,就返回池里的字符串;如果不存在,该方法会把 "1" 添加到字符串常量池中,然后再返回它的引用

JDK 1.6 及以上

  • 上述代码输出结果:
false
true
  • 解释:

String s= new String("1") 生成了字符串常量池中的 "1" 和堆空间中的字符串对象。

s.intern() s 对象去字符串常量池中寻找后,发现 "1" 已存在于常量池中。

String s2 = "1" 生成 s2 的引用指向常量池中的 "1" 对象。

显然,s 和 s2 的引用地址是不同的。

String s3 = new String("1") + new String("1") 在字符串常量池中生成 "1",并在堆空间中生成 s3 引用指向的对象(内容为 "11")。 注意此时常量池中是没有 "11" 对象

s3.intern()将 s3 中的 "11" 字符串放入字符串常量池中。 JDK 1.6 的做法是直接在常量池中生成一个 "11" 的对象。但在 JDK 1.7 中,常量池中不需要再存储一份对象了,可以直接存储堆中的引用。这份引用直接指向 s3 引用的对象,也就是说 s3.intern() == s3 会返回 true。

String s4 = "11", 这一行代码会直接去常量池中创建,但是发现已经有这个对象了,此时 s4 就是指向 s3 引用对象的一个引用。因此 s3 == s4 返回了true。

3、字符串拼接

String str1 = "str";
String str2 = "ing";
		  
String str3 = "str" + "ing";//常量池中的对象
String str4 = str1 + str2; //TODO:在堆上创建的新的对象	  
String str5 = "string";//常量池中的对象
System.out.println(str3 == str4);//false
System.out.println(str3 == str5);//true
System.out.println(str4 == str5);//false

注意:尽量避免多个字符串拼接,因为这样会重新创建对象。 如果需要改变字符串的话,可以使用 StringBuilder 或者 StringBuffer

面试题:String s1 = new String("abc");问创建了几个对象?

创建2个字符串对象(前提是 String Pool 中还没有 "abcd" 字符串对象)。

  • "abc" 属于字符串字面量,因此编译时期会在常量池中创建一个字符串对象,指向这个 "abcd" 字符串字面量;
  • 使用 new 的方式会在堆中创建一个字符串对象。

(字符串常量"abc"在编译期就已经确定放入常量池,而 Java 堆上的"abc"是在运行期初始化阶段才确定)。

String s1 = new String("abc");// 堆内存的地址值
String s2 = "abc";
System.out.println(s1 == s2);// 输出false
//因为一个是堆内存,一个是常量池的内存,故两者是不同的。
System.out.println(s1.equals(s2));// 输出true

参考:

标签:String,对象,池中,intern,字符串,常量
From: https://www.cnblogs.com/i9code/p/17986075

相关文章

  • 8 种基本类型的包装类和常量池
    Java基本类型的包装类的大部分都实现了常量池技术,即Byte,Short,Integer,Long,Character,Boolean;这5种包装类默认创建了数值[-128,127]的相应类型的缓存数据,但是超出此范围仍然会去创建新的对象。两种浮点数类型的包装类Float,Double并没有实现常量池技术。valueOf()方......
  • string 函数
    在C++中,string类型是处理字符串的一种方便的方式,它包含了许多有用的成员函数来进行字符串操作。以下是一些常用的string函数的示例说明:构造函数和赋值:创建空字符串:stringstr;使用字符串常量初始化:stringstr="Hello";使用字符数组初始化:charcharArray[]="World";......
  • 一文搞清楚Java中的方法、常量、变量、参数
    写在开头在上一篇文章:一文搞清楚Java中的包、类、接口中我们讲了Java中的包、类和接口,今天继续将剩下的方法、常量、变量以及参数梳理完。Java中的变量与常量在JVM的运转中,承载的是数据,而数据的一种变现形式就是“量”,量分为:常量与变量,我们在数学和物理学中已经接触过变量的概......
  • Go语言核心36讲 37 | strings包与字符串操作
    在上一篇文章中,我介绍了Go语言与Unicode编码规范、UTF-8编码格式的渊源及运用。Go语言不但拥有可以独立代表Unicode字符的类型rune,而且还有可以对字符串值进行Unicode字符拆分的for语句。除此之外,标准库中的unicode包及其子包还提供了很多的函数和数据类型,可以帮助我们解析各......
  • toString、求平均数工具类
     1/**2*构造器私有化3*/4privateArraysUtils(){}56//toString()工具类静态方法、工具方法7publicstaticStringtoString(int[]arr){8Stringresult="[";9for(inti=0;i<arr.length;i++)......
  • QOJ 2486 Build the String
    考虑当字符串全为\(\texttt{b}\)时,可以通过\(\text{copy}\)\(n-1\)次再\(\text{fuse}\)\(n\)次。这启发从连续段来做,先按顺序构造出一个个连续段,最后\(\text{fuse}\)合为这个串。若第一个连续段为\(\texttt{a}\),则可以通过\(\text{swap}\)事先交换\(\texttt{ab}......
  • StringGrid1单元格内绘图
     varRect:Trect;beginRect:=bg.CellRect(4,3);bg.Canvas.Brush.Color:=clwhite;bg.Canvas.FillRect(rect);bg.Canvas.Draw(rect.Left+trunc((rect.Right-rect.Left-tx2.Width)/2)//单元格水平居中,rect.Top+tr......
  • 21String类的实现
    String类的实现//#include<string>#include<iostream>usingnamespacestd;classString{private: char*_pstr; friendStringoperator+(constString&s1,constString&s2); friendostream&operator<<(ostream&out,constS......
  • 22String字符串和vector对象的迭代器iterator实现
    String字符串对象的迭代器iterator实现泛型算法参数接收的都是迭代器泛型算法是一组全局的函数,适用于所有容器基于第二点,泛型算法有一套方法可以统一地遍历所有容器的元素classString{public: //嵌套定义iterator类 classiterator { private: char*_p;//没有用......
  • CF1830C Hyperregular Bracket Strings
    HyperregularBracketStringsLuoguCF1830C题目描述给定一个数\(n\)和\(k\)个区间\(\left[l_i,r_i\right]\in[1,n]\)。我们定义,对于一个长度为\(n\)的,仅由(和)组成的合法括号序列,如果它的每一个区间\(\left[l_i,r_i\right]\)内的子串都是合法括号序列,那么这个......