`
geeksun
  • 浏览: 961287 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Java堆和栈、String Pool(字符串池)

    博客分类:
  • jvm
阅读更多

Thinking in Java中详细介绍了当程序运行的时候,具体的内存分配。
可以分为寄存器、堆栈、堆、常量存储、非RAM存储。
1.   堆栈比较
     栈(stack)与堆(heap)都是Java用来在RAM中存放数据的地方。
     栈的优势是,存取速度比堆要快,仅次于CPU的寄存器
,栈数据可以共享。。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。栈中主要存放基本类型变量(primitive types), 共有8种,即int, short, long, byte, float, double, boolean, char,和对象的句柄。

     堆是运行时数据区,优势是可以动态地分配内存大小,生存期不必事先告诉编译器,Java的垃圾收集器会自动收走这些不再使用的数据。但缺点是,由于要 在运行时动态分配内存,存取速度较慢。 堆中主要存放对象,包括基本类型的包装类,如Integer, String, Double等将相应的基本数据类型包装起来的类,这些类数据全部存在于堆中。

     栈的数据可以共享,可通过下面例子来说明:

 

int a = 3;
int b = 3;

      编译器先处理int a = 3; 首先在栈中创建一个变量为a的引用,然后在栈中查找是否有3这个值,如果没找到,就将3这个值存进来,然后将a指向3.

    再处理int b = 3;在创建b的引用变量后,因为栈中已有3存在,b引用直接指向3。这样,a和b便指向了同一个变量3。

        这时,如果再令a=4;那么编译器会重新搜索栈中是否有4值,如果没有,则将4存放进来,并令a指向4;如果已经有了,则直接将a指向这个地址。因此a值的改变不会影响到b的值。

        要注意这种数据的共享与两个对象的引用同时指向一个对象的这种共享是不同的,因为这种情况a的修改并不会影响到b, 它是由编译器完成的,它有利于节省空间。而一个对象引用变量修改了这个对象的内部状态,会影响到另一个对象引用变量。

2.   String Pool(字符串池)

String是一个特殊的包装类数据。可以用:

String str = new String("abc");
String str = "abc"; 
      两种方式来创建,第一种是在堆里创建String对象,第二种是先在栈中创建一个String类的对象引用变量str,然后查找字符串池(位于常量池)中有没有存放"abc",如果没有,则将"abc"放入字符串池(string pool),并令str指向"abc",如果有则直接指向"abc"。

        当第一次使用某个字符串直接量时,JVM会将它放入字符串池进行缓存,这有点像是享元模式的应用。在一般情况下,字符串池中的字符串对象不会被垃圾回收器回收,当程序再次需要使用该字符串时,无需重新创建一个新的字符串而是直接让引用变量指向字符串池中已有的字符串。

      在一般情况下,字符串池中的字符串对象不会被垃圾回收。当程序再次需要使用该字符串时,无需重新创建一个新的字符串就可以直接让引用变量直接指向字符串中已有的字符串。

      常量池(constant pool)指的是在编译期被确定,并被保存在已编译的.class文件中的一些数据。它包括了关于类、方法、接口等中的常量,也包括字符串常量。


     比较类里面的数值是否相等时,用equals()方法;当测试两个包装类的引用是否指向同一个对象时,用==,下面用例子说明上面的理论。 

String str1 = "abc";
String str2 = "abc";
System.out.println(str1==str2);              //true 
  可以看出str1和str2是指向同一个对象的。

String str1 =new String ("abc");
String str2 =new String ("abc");
System.out.println(str1==str2);              // false
  用new的方式是生成不同的对象。每一次生成一个。

     使用String str = "abc"的方式,可以在一定程度上提高程序的运行速度,因为JVM会自动根据字符串池的实际情况来决定是否有必要创建新对象。而对于String str = new String("abc");的代码,则一概在堆中创建新对象,而不管其字符串值是否相等,是否有必要创建新对象,从而加重了程序的负担。

 另一方面, 要注意: 我们在使用诸如String str = "abc";的格式定义类时,总是想当然地认为,创建了String类的对象str。担心陷阱!对象可能并没有被创建!而可能只是指向一个先前已经创建的对象。只有通过new()方法才能保证每次都创建一个新的对象。

    由于String类的immutable性质,当String变量需要经常变换其值时,应该考虑使用StringBuffer类,以提高程序效率。

 String.intern():

 再补充介绍一点:存在于.class文件中的常量池,在运行期被JVM装载,并且可以扩充。String的intern()方法就是扩充常量池的一个方法;当一个String实例str调用intern()方法时,Java查找常量池中是否有相同Unicode的字符串常量,如果有,则返回其的引用,如果没有,则在常量池中增加一个Unicode等于str的字符串并返回它的引用;看下列代码 :

String s0= “kvill”;  
String s1=new String(”kvill”);  
String s2=new String(“kvill”);  
System.out.println( s0==s1 );  
System.out.println( “**********” );  
s1.intern();  
s2=s2.intern(); //把常量池中“kvill”的引用赋给s2  
System.out.println( s0==s1);  
System.out.println( s0==s1.intern() );  
System.out.println( s0==s2 );  

 结果为:

false  
**********  
false //虽然执行了s1.intern(),但它的返回值没有赋给s1  
true //说明s1.intern()返回的是常量池中”kvill”的引用  
true  
       关于equals()和==:

这个对于String简单来说就是比较两字符串的Unicode序列是否相当,如果相等返回true;而==是比较两字符串的地址是否相同,也就是是否是同一个字符串的引用。

       关于String是不可变的

这一说又要说很多,大家只要知道String的实例一旦生成就不会再改变了,比如说:String str=”kv”+”ill”+” “+”ans”;
就是有4个字符串常量,首先”kv”和”ill”生成了”kvill”存在内存中,然后”kvill”又和” “ 生成 ”kvill “存在内存中,最后又和生成了”kvill ans”;并把这个字符串的地址赋给了str,就是因为String的“不可变”产生了很多临时变量,这也就是为什么建议用StringBuffer的原因了,因为StringBuffer是可改变的。

      因为字符串池中的字符串对象不会被垃圾回收,所以当某个字符串池中的字符串对象失去引用时,它将变成垃圾,而字符串池又不回收。于是便产生了java内存泄露。示例如下:

public class StringDemo {
    public static void main(String args[]){
        String str1 = "abc";
        String str3 = "def";
        str3 = str1 ;
        System.out.println(str3);
        System.out.println(str1);
    }
}
输出:

abc

"def"失去引用,将成为垃圾,进而造成内存泄露。

分享到:
评论

相关推荐

    Java堆,栈和常量池详解

    ### Java堆、栈和常量池详解 #### Java内存模型概览 在深入探讨Java中的堆、栈以及常量池之前,我们先来简要回顾一下Java内存模型的基本概念。Java程序运行时会使用到不同的内存区域来存储各种类型的数据,这些...

    java 内存中 堆、栈、常量池、方法区的总结

    在Java内存管理中,堆(Heap)、栈(Stack)、常量池(Constant Pool)和方法区(Method Area)是四个核心概念,它们在Java程序运行时扮演着不同的角色。 首先,方法区是用来存放类的信息、常量、静态变量等数据的...

    java内存机制,分析堆和栈的存储特点资料.pdf

    ` 这种情况下,Java会检查字符串常量池(String Pool)中是否存在相同的字符串"abc"。如果不存在,它会将"abc"放入池中并创建一个引用str1指向这个字符串。如果已经存在,str1则直接引用池中的"abc"。 2. 使用`new`...

    Java中的堆、栈和常量池_.docx

    字符串对象的引用存储在栈中,而实际的字符串内容可能存储在常量池(编译期已知的字符串)或堆中(运行期创建的字符串)。例如: ```java String s1 = "china"; // 字符串常量,存储在常量池 String s2 = "china"; ...

    java中的栈(深层了解java虚拟机对对象的内存分布)

    后者利用了字符串常量池(String Constant Pool)的概念,当字符串字面量首次出现时,会在常量池中创建一个新的字符串对象,之后相同的字符串字面量将直接引用池中的已有对象,避免了重复创建相同内容的字符串。...

    String类创建对象问题

    `,即先在栈中创建字符数组,再基于该数组创建 `String` 对象,并将该对象的引用加入到字符串池中。 总结来说,了解 `String` 类的创建过程及其在内存中的存储方式对于理解 Java 中字符串的工作机制至关重要。正确...

    基于Java中字符串内存位置详解

    4. 常量池(Constant Pool):存储字符串字面量和编译期常量。在JDK7之后,常量池位于堆内存中,包含字符串字面量(如`"holten"`)和其他常量。 接下来,我们将分析不同情况下字符串在内存中的位置: 1. 显式声明...

    飞加Java学习笔记_内存管理

    当调用`intern()`方法时,如果字符串池中还没有当前`String`对象表示的字符串,那么这个字符串将被添加到字符串池中,并返回字符串池中的这个字符串的引用;如果字符串池中已经有相同的字符串,那么将返回字符串池中...

    创建string对象过程的内存分配:

    - 接着,在 **堆** 中通过调用 `String` 构造器创建一个新的字符串对象 `"abc"`,并将 `s` 引用指向这个新对象。 这种方式下,无论 **常量池** 中是否已有相同的字符串对象,都会在 **堆** 中创建一个新的对象。这...

    JAVA内存分配与管理.doc

    Java的字符串池优化了字符串的内存使用,避免了相同字符串的多次创建。 在栈中,多个变量可以指向同一个堆中的对象,例如在示例代码中,`int a = 3; int b = 3;`,a和b都指向栈中相同的值3,这是一种空间的优化。 ...

    java内存分配[归类].pdf

    1. 字符串常量池(String Pool)位于堆内存中,当使用字面量创建字符串时,如 `String str = "abc"`,如果池中已有"abc",则直接引用;否则,将"abc"放入池中并创建引用。 2. 使用 `new String("abc")` 创建字符串时...

    String对象的内存分析

    3. 常量池(Constant Pool):是堆内存中的一部分,用来存储字符串常量和基本类型常量,以及`public static final`修饰的变量。常量池中的数据可以被多个对象共享。 4. 静态存储:用于存储`static`定义的成员,这些...

    java内存分配描述

    如果两个字符串常量具有相同的值,它们在常量池中只会占用一个位置,这也是Java字符串池的概念。 6. **非RAM存储(Non-RAM Storage)**:包括硬盘和其他持久化存储,用于存储持久化的数据,如数据库文件、配置文件...

    深入Java核心 Java内存分配原理精讲

    `的方式创建对象时,如果字符串池中已经存在“abc”这个字符串,则直接返回该字符串的引用;否则会将新字符串添加到字符串池中,并返回该字符串的引用。因此,使用`==`比较两个String对象是否相等时,实际上是比较...

    Java 堆内存与栈内存详细介绍

    例如,当你声明一个`String`变量并赋值为一个字符串字面量时,字符串的值会存储在常量池(Constant Pool)中,而变量本身会存储在栈中,指向常量池中的地址。如果你使用`new String("example")`创建一个新的字符串...

    深入Java核心 Java内存分配原理精讲.docx

    对于String对象,由于其特殊性,常量池中的字符串可以通过字符串字面量引用,这种方式能有效利用字符串池,减少内存开销。 Java内存管理的另一个重要概念是垃圾回收(Garbage Collection, GC)。GC负责自动回收不再...

    关于java内存的小常识,以helloworld程序为例

    4. **常量池(Constant Pool)**:位于方法区中,存储字符串常量和符号引用。在第三步中,"HelloWorld"字符串被查找并在常量池中添加,然后将引用赋予`s`。 5. **共享代码区(Shared Code Area)**:这是JVM为类的...

    java技术面试必问:JVM-内存模型讲解.docx

    字面量包括字符串(String a= "b”)、基本类型的常量(final修饰的变量),符号引用则包括类和方法的全限定名(例如String这个类,它的全限定名就是Java/lang/String)、字段的名称和描述符以及方法的名称和描述符...

    深入Java内存分配

    3. **Java堆**(Java Heap):这是一种动态内存管理模型,允许以任意顺序在运行时进行存储空间的分配和回收。堆中存储的数据往往是大小、数量和生命周期在编译时不确定的。所有Java对象的内存都是在堆中分配的。 ##...

Global site tag (gtag.js) - Google Analytics