首页 > 代码库 > 浅谈JAVA中字符串常量的储存位置

浅谈JAVA中字符串常量的储存位置

在讲述这些之前我们需要一些预备知识:

java中的内存被分成以下部分:

1、栈区:由编译器自动分配释放,具体方法执行结束后,系统自动释放JVM内存资源。

其作用有保存局部变量的值,包括:1.用来保存基本数据类型的值;2.保存类的实例,即堆区对象的引用(指针)。也可以用来保存加载方法时的帧。

2、堆区:一般由程序员分配释放,JVM不定时查看这个对象,如果没有引用指向这个对象就回收。

其作用为用来存放动态产生的数据,比如new出来的对象。注意创建出来的对象只包含属于各自的成员变量,并不包括成员方法。

因为同一个类的对象拥有各自的成员变量,存储在各自的堆中,但是他们共享该类的方法,并不是每创建一个对象就把成员方法复制一次。

3、代码区存放程序中方法的二进制代码,而且是多个对象共享一个代码空间区域

4、数据区:用来存放static定义的静态成员。

5、常量池:JVM为每个已加载的类型维护一个常量池,常量池就是这个类型用到的常量的一个有序集合。包括直接常量(基本类型,String)和对其他类型、方法、字段的符号引用。池中的数据和数组一样通过索引访问。由于常量池包含了一个类型所有的对其他类型、方法、字段的符号引用,所以常量池在Java的动态链接中起了核心作用。常量池存在于堆中

下图大致描述了JAVA的内存分配

技术分享

 

接来下我们来看一段代码实例:

 1 public class TestStringConstant {
 2     public static void main(String args[]) {
 3         // 字符串常量,分配在Data Segment,编译器会对其进行优化,
 4         // 即当一个字符串已经存在时,不再重复创建一个相同的对象,而是直接将s2也指向"hello".
 5         String s1 = "hello";   
 6         String s2 = "hello";
 7         // new出来的对象,分配在heap中.s3与s4虽然它们指向的字符串内容是相同的,但是是两个不同的对象.
 8         // 因此==进行比较时,其所存的引用是不同的,故不会相等
 9         String s3 = new String("world");      
10         String s4 = new String("world");
11         
12         System.out.println(s1 == s2);   // true
13         System.out.println(s3 == s4);   // false
14         System.out.println(s3.equals(s4));   // true   
15         // String中equals方法已经被重写过,比较的是内容是否相等.
16     }
17 }

通过以上相信大家对于字符串常量的分配区域以及java的内存分配有了一个较为形象的了解。

下面是一些相关知识点的补充与注意事项:

1.分清什么是实例什么是对象。Class a= new Class();此时a叫实例,而不能说a是对象。实例在栈中,对象在堆中,操作实例实际上是通过实例的指针间接操作对象。多个实例可以指向同一个对象。

2.栈中的数据和堆中的数据销毁并不是同步的。方法一旦结束,栈中的局部变量立即销毁,但是堆中对象不一定销毁。因为可能有其他变量也指向了这个对象,直到栈中没有变量指向堆中的对象时,它才销毁,而且还不是马上销毁,要等垃圾回收扫描时才可以被销毁。

3.以上的栈、堆、代码段、数据段等等都是相对于应用程序而言的。每一个应用程序都对应唯一的一个JVM实例,每一个JVM实例都有自己的内存区域,互不影响。并且这些内存区域是所有线程共享的。这里提到的栈和堆都是整体上的概念,这些堆栈还可以细分。

4.类的成员变量在不同对象中各不相同,都有自己的存储空间(成员变量在堆中的对象中)。而类的方法却是该类的所有对象共享的,只有一套,对象使用方法的时候方法才被压入栈,方法不使用则不占用内存。

注:以上仅仅谈及了stack与heap这两个部分,但是还有一个较为麻烦的部分——常量池。本文这里对此不再叙述,想要了解的可移步以下内容:

http://www.blogjava.net/Jack2007/archive/2008/05/21/202018.html

http://developer.51cto.com/art/201009/225071_1.htm

 

浅谈JAVA中字符串常量的储存位置