转自:http://www.importnew.com/10756.html
在理解字符串常量前,我们先熟悉一下如何创建一个字符串,在Java中有两种方法可以创建一个字符串对象:
- 使用new运算符。例如:
1
|
String str = new String( "Hello" );
|
- 使用字符串常量或者常量表达式。例如:
1
2
|
String str= "Hello" ; //(字符串常量) 或者
String str= "Hel" + "lo" ; //(字符串常量表达式).
|
这些字符串的创建方式之间有什么区别呢?在Java中,equals方法被认为是对象的值进行深层次的比较,而操作符==是进行的浅层次的比较。equals方法比较两个对象的内容而不是引用。==两侧是引用类型(例如对象)时,如果引用是相同的-即指向同一个对象-则执行结果为真。如果是值类型(例如原生类型),如果值相同,则执行结果为真。equals方法在两个对象具有相同内容时返回真-但是,java.lang.Object类中的equals方法返回真-如果类没有覆盖默认的equals方法,如果两个引用指向同一个对象。
让我们通过下面的例子来看看这两种字符串的创建方式之间有什么区别吧。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
public class DemoStringCreation {
public static void main(String args[]) {
String str1 = "Hello" ;
String str2 = "Hello" ;
System.out.println( "str1 and str2 are created by using string literal." );
System.out.println( " str1 == str2 is " + (str1 == str2));
System.out.println( " str1.equals(str2) is " + str1.equals(str2));
String str3 = new String( "Hello" );
String str4 = new String( "Hello" );
System.out.println( "str3 and str4 are created by using new operator." );
System.out.println( " str3 == str4 is " + (str3 == str4));
System.out.println( " str3.equals(str4) is " + str3.equals(str4));
String str5 = "Hel" + "lo" ;
String str6 = "He" + "llo" ;
System.out.println( "str5 and str6 are created by using string constant expression." );
System.out.println( " str5 == str6 is " + (str5 == str6));
System.out.println( " str5.equals(str6) is " + str5.equals(str6));
String s = "lo" ;
String str7 = "Hel" + s;
String str8 = "He" + "llo" ;
System.out.println( "str7 is computed at runtime." );
System.out.println( "str8 is created by using string constant expression." );
System.out.println( " str7 == str8 is " + (str7 == str8));
System.out.println( " str7.equals(str8) is " + str7.equals(str8));
}
} |
输出结果为:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
str1 and str2 are created by using string literal. str1 == str2 is true
str1.equals(str2) is true
str3 and str4 are created by using new operator.
str3 == str4 is false
str3.equals(str4) is true
str5 and str6 are created by using string constant expression. str5 == str6 is true
str5.equals(str6) is true
str7 is computed at runtime. str8 is created by using string constant expression. str7 == str8 is false
str7.equals(str8) is true
|
使用相同的字符序列而不是使用new关键字创建的两个字符串会创建指向Java字符串常量池中的同一个字符串的指针。字符串常量池是Java节约资源的一种方式。
字符串常量池
字符串的分配,和其他的对象分配一样,耗费高昂的时间与空间代价。JVM为了提高性能和减少内存开销,在实例化字符串常量的时候进行了一些优化。为了减少在JVM中创建的字符串的数量,字符串类维护了一个字符串池,每当代码创建字符串常量时,JVM会首先检查字符串常量池。如果字符串已经存在池中,就返回池中的实例引用。如果字符串不在池中,就会实例化一个字符串并放到池中。Java能够进行这样的优化是因为字符串是不可变的,可以不用担心数据冲突进行共享。例如:
1
2
3
4
5
6
7
8
9
|
public class Program
{ public static void main(String[] args)
{
String str1 = "Hello" ;
String str2 = "Hello" ;
System.out.print(str1 == str2);
}
} |
其结果是:
1
|
true |
不幸的是,当使用:
1
|
String a= new String( "Hello" );
|
一个字符串对象在字符串常量池外创建,即使池里存在相同的字符串。考虑到这些,要避免new一个字符串除非你明确的知道需要这么做!例如:
1
2
3
4
5
6
7
8
9
10
|
public class Program
{ public static void main(String[] args)
{
String str1 = "Hello" ;
String str2 = new String( "Hello" );
System.out.print(str1 == str2 + " " );
System.out.print(str1.equals(str2));
}
} |
结果是:
1
|
false true
|
JVM中有一个常量池,任何字符串至多维护一个对象。字符串常量总是指向字符串池中的一个对象。通过new操作符创建的字符串对象不指向字符串池中的任何对象,但是可以通过使用字符串的intern()方法来指向其中的某一个。java.lang.String.intern()返回一个保留池字符串,就是一个在全局字符串池中有了一个入口。如果以前没有在全局字符串池中,那么它就会被添加到里面。例如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
public class Program
{ public static void main(String[] args)
{
// Create three strings in three different ways.
String s1 = "Hello" ;
String s2 = new StringBuffer( "He" ).append( "llo" ).toString();
String s3 = s2.intern();
// Determine which strings are equivalent using the ==
// operator
System.out.println( "s1 == s2? " + (s1 == s2));
System.out.println( "s1 == s3? " + (s1 == s3));
}
} |
输出是:
1
2
|
s1 == s2? false
s1 == s3? true
|
为了优化空间,运行时实例创建的全局字符串常量池中有一个表,总是为池中每个唯一的字符串对象维护一个引用。这就意味着它们一直引用着字符串常量池中的对象,所以,在常量池中的这些字符串不会被垃圾收集器回收。
Java语言规范第三版中的字符串常量
每一个字符串常量都是指向一个字符串类实例的引用。字符串对象有一个固定值。字符串常量,或者一般的说,常量表达式中的字符串都被使用方法 String.intern进行保留来共享唯一的实例。
1
2
3
4
5
6
7
8
9
10
11
12
13
|
package testPackage;
class Test {
public static void main(String[] args) {
String hello = "Hello" , lo = "lo" ;
System.out.print((hello == "Hello" ) + " " );
System.out.print((Other.hello == hello) + " " );
System.out.print((other.Other.hello == hello) + " " );
System.out.print((hello == ( "Hel" + "lo" )) + " " );
System.out.print((hello == ( "Hel" +lo)) + " " );
System.out.println(hello == ( "Hel" +lo).intern());
}
} class Other { static String hello = "Hello" ; }
|
编译单元:
1
2
|
package other;
public class Other { static String hello = "Hello" ; }
|
产生输出:
1
|
true true true true false true
|
这个例子说明了六点:
- 同一个包下同一个类中的字符串常量的引用指向同一个字符串对象;
- 同一个包下不同的类中的字符串常量的引用指向同一个字符串对象;
- 不同的包下不同的类中的字符串常量的引用仍然指向同一个字符串对象;
- 由常量表达式计算出的字符串在编译时进行计算,然后被当作常量;
- 在运行时通过连接计算出的字符串是新创建的,因此是不同的;
- 通过计算生成的字符串显示调用intern方法后产生的结果与原来存在的同样内容的字符串常量是一样的。
相关推荐
第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: 揭秘JVM字符串常量池和Java堆-01第4节: ...
在Java编程语言中,字符串常量池(String Constant Pool)是一个重要的概念,它与程序的内存管理和性能优化密切相关。理解这个概念对于任何Java开发者来说都至关重要。字符串常量池是Java虚拟机(JVM)在运行时为...
为了提高效率和内存管理,Java虚拟机(JVM)引入了字符串常量池这一概念。字符串常量池是一个特殊的区域,它存储了程序中所有的字符串常量,避免了多次创建相同的字符串对象。本文将详细探讨字符串常量池的存储方式...
**StringTable(字符串常量池)详解** 在Java编程语言中,`StringTable`是一个重要的概念,它涉及到字符串对象的创建、存储以及内存管理。理解`StringTable`的工作原理对于优化程序性能和节省内存资源至关重要。 #...
String int 字符串常量池 包装类型 函数参数 值传递引用传递 的 内存分配例子——源码 public static void fun_ref (Ref_test ref_out){ Ref_test ref_in=new Ref_test(); ref_in.s1="in"; //ref_out.s1=...
Java中的字符串常量池是Java虚拟机(JVM)为了优化字符串对象的使用而设计的一个特殊内存区域。这个池主要用于存储字符串字面量,也就是在程序中直接出现的字符串值,比如`"hello"`。其核心目的是减少内存的消耗和...
Java中的字符串常量池是Java虚拟机(JVM)为了优化字符串对象的使用而设立的一个特殊区域,它存储了程序中所有的字符串字面量。在Java中,字符串是不可变的,这意味着一旦创建,就不能修改。这个特性使得字符串常量...
Java String 字符串常量池解析 Java 中的字符串常量池是一种为了提高性能和减少内存开销的机制。它是 JVM 实例化字符串常量时进行的一些优化,主要是为了减少字符串对象的创建和存储。 字符串常量池的设计思想是...
本文将深入探讨C#中的CLR(Common Language Runtime)内存字符串常量池,以及它如何处理和优化字符串。 首先,我们需要理解C#中的内存模型。在C#中,内存分为两种主要区域:堆(Heap)和栈(Stack)。栈主要用于...
"深入解析JVM之内存结构及字符串常量池" JVM(Java Virtual Machine)是Java语言的核心组件之一,负责将Java代码编译成机器代码并执行。JVM的内存结构是Java开发者需要了解的基础知识之一,本文将深入解析JVM之内存...
字符串常量池和intern 字符串常量池是Java语言中的一种机制,旨在节省空间和提高性能。它是一个内存区域,所有的Java类共享这个池子。字符串常量池的设计思想是为了解决字符串的频繁创建问题,减少内存开销和提高...
在实际开发中,我们还需要关注JVM的其他重要概念,如类加载机制(加载、验证、准备、解析、初始化)、类文件结构(魔数、版本号、常量池等)、垃圾回收算法(如标记-清除、复制、标记-整理、分代收集)以及执行引擎...
- **字符串字面量的处理**:当程序尝试创建一个新的字符串字面量时,JVM首先会检查字符串常量池中是否存在该字符串。如果存在,则返回已存在的字符串对象;如果不存在,则创建一个新的字符串对象并将其放入字符串...
在JDK 1.6及以前,字符串常量池位于方法区(也称为永久代),而在JDK 1.7中,随着永久代逐渐被移除,字符串常量池被移到堆中。到了JDK 1.8及以后,永久代完全消失,元空间取代了它的位置,但字符串常量池仍然存在于...
在 Java6 中,字符串常量池是放在 Perm 空间的,而从 Java7 开始,字符串常量池被移到 Heap 空间。 Perm 空间是一个固定大小的内存区域,用于存储类的元数据、方法、字段和字符串常量池等信息。当 Perm 空间接近满的...
而在第二行比较中,由于 `"hello"` 和 `"world"` 都是字符串字面量,它们会被合并成一个 `"helloworld"` 字符串字面量,这使得 `s3` 与 `"hello" + "world"` 实际上都指向字符串常量池中的同一个对象。 通过使用反...
- 字符串常量池是JVM内存中的一个特殊区域,用于存放所有的字符串字面量。当创建一个`String`对象时,如果常量池中已经存在相同内容的字符串,那么将返回该字符串的引用,而不是创建新的对象。 4. **字符串比较**...
jvm如何处理长字符串?java的classs文件中,constant_utf8_info的长度是u2,也就是说,一个字符串最长是65535个字节,但是,在本机做测试,超过这个长度的字符串也是允许的,原因是什么?
字符串常量池在JDK8之前位于永久代,但在变化后,它被移出永久代,仍然保留在堆内存中,可能是为了更方便地进行垃圾回收。与此相似,整型常量池也独立出来,用于存储基本类型整数的常量。 整体来看,JDK8的JVM内存...