常量池、栈、堆的比较

http://www.cnblogs.com/Eason-S/p/5658230.htmlhtml

JAVA中,有六个不一样的地方能够存储数据:java

1.寄存器:最快的存储区,位于不一样于其余存储区的地方——处理器内部。寄存器的数量极其有限,因此寄存器由编译器根据需求进行分配。你不能直接控制,也不能在程序中感受到寄存器存在的任何迹象。面试

2. 栈:存放基本类型的变量数据和对象的引用。位于通用RAM中,但经过它的“堆栈指针”能够从处理器哪里得到支持。堆栈指针若向下移动,则分配新的内存;若向上移动,则释放那些内存。这是一种快速有效的分配存储方法,仅次于寄存器。建立程序时候,JAVA编译器必须知道存储在堆栈内全部数据的确切大小和生命周期,由于它必须生成 相应的代码,以便上下移动堆栈指针。这一约束限制了程序的灵活性。安全

3. 堆:一种通用性的内存池(也存在于RAM中),用于存放因此的JAVA对象。堆不一样于堆栈的好处是:编译器不须要知道要从堆里分配多少存储区域,也没必要知道存储的数据在堆里存活多长时间。所以,在堆里分配存储有很大的灵活性。当你须要建立一个对象的时候,只须要new写一行简单的代码,当执行这行代码时,会自动在堆里进行存储分配。固然,为这种灵活性必需要付出相应的代码。用堆进行存储分配比用堆栈进行存储存储须要更多的时间。  jvm

4. 静态域:存放静态成员(static定义的) 。操作系统

5. 常量池:存放字符串常量和基本类型常量(public static final)。 常量值一般直接存放在程序代码内部,这样作是安全的,由于它们永远不会被改变。.net

6. 非RAM存储:硬盘等永久存储空间。若是数据彻底存活于程序以外,那么它能够不受程序的任何控制,在程序没有运行时也能够存在。 设计

这里咱们主要关心栈,堆和常量池,对于栈和常量池中的对象能够共享,对于堆中的对象不能够共享。指针

栈中的数据大小和生命周期是能够肯定的,当没有引用指向数据时,这个数据就会消失。堆中的对象的由垃圾回收器负责回收,所以大小和生命周期不须要肯定,具备很大的灵活性。    htm

(1)对于字符串:其对象的引用都是存储在栈中的,若是是编译期已经建立好(直接用双引号定义的)的就存储在常量池中,若是是运行期(new出来的)才能肯定的就存储在堆中。对于equals相等的字符串,在常量池中永远只有一份,在堆中有多份。

例如:

复制代码

1 String s1 = "china";
2 String s2 = "china";
3 String s3 = "china";
4 String ss1 = new String("china");
5 String ss2 = new String("china");
6 String ss3 = new String("china");

复制代码

对于经过new产生一个字符串(假设为”china”)时,会先去常量池中查找是否已经有了”china”对象,若是没有则在常量池中建立一个此字符串对象,而后堆中再建立一个常量池中此”china”对象的拷贝对象。

这也就是有道面试题:String s = new String(“xyz”);产生几个对象?答:一个或两个,若是常量池中原来没有”xyz”,就是两个。

 

(2)对于基础类型的变量和常量:变量和引用存储在栈中,常量存储在常量池中。

例如:

复制代码

1 int i1 = 9;
2 int i2 = 9;
3 int i3 = 9; 
4 public static final int INT1 = 9;
5 public static final int INT2 = 9;
6 public static final int INT3 = 9;

复制代码

对于成员变量和局部变量:成员变量就是方法外部,类的内部定义的变量;局部变量就是方法或语句块内部定义的变量。局部变量必须初始化。

形式参数是局部变量,局部变量的数据存在于栈内存中。栈内存中的局部变量随着方法的消失而消失。

成员变量存储在堆中的对象里面,由垃圾回收器负责回收。

下面给出一个实例:

复制代码

1 class BirthDate {
 2     private int day;
 3     private int month;
 4     private int year;    
 5     public BirthDate(int d, int m, int y) {
 6         day = d; 
 7         month = m; 
 8         year = y;
 9     }
10     省略get,set方法………
11 }
12 
13 public class Test{
14     public static void main(String args[]){
15         int date = 9;
16         Test test = new Test();      
17         test.change(date); 
18         BirthDate d1= new BirthDate(7,7,1970);       
19     }  
20 
21     public void change(int i){
22      i = 1234;
23     }
24 }

复制代码

对于以上这段代码,date为局部变量,i,d,m,y都是形参为局部变量,day,month,year为成员变量。下面分析一下代码执行时候的变化:

1. main方法开始执行:int date = 9;

date局部变量,基础类型,引用和值都存在栈中。

2. Test test = new Test();

test为对象引用,存在栈中,对象(new Test())存在堆中。

3. test.change(date);

i为局部变量,引用和值存在栈中。当方法change执行完成后,i就会从栈中消失。

4. BirthDate d1= new BirthDate(7,7,1970);  

d1为对象引用,存在栈中,对象(new BirthDate())存在堆中,其中d,m,y为局部变量存储在栈中,且它们的类型为基础类型,所以它们的数据也存储在栈中。day,month,year为成员变量,它们存储在堆中(new BirthDate()里面)。当BirthDate构造方法执行完以后,d,m,y将从栈中消失。

5.main方法执行完以后,date变量,test,d1引用将从栈中消失,new Test(),new BirthDate()将等待垃圾回收。

 

参考:<http://www.iteye.com/topic/634530>

        <http://www.cnblogs.com/xiohao/p/4296088.html>

        《深刻理解java虚拟机———jvm高级特性与最佳实践》

 

JAVA -Xms -Xmx -XX:PermSize -XX:MaxPermSize 区别

JVM按照其存储数据的内容将所需内存分配为堆区与非堆区两个部分:所谓堆区即为经过new的方式建立的对象(类实例)所占用的内存空间;非堆区即为代码、常量、外部访问(如文件访问流所占资源)等。

 

常见参数种类(配置内存):(-Xms 、-Xmx、-XX:newSize、-XX:MaxnewSize、-Xmn)、(-XX:PermSize、-XX:MaxPermSize)。能够从列举的方式上看出个人用意,参数的配置是分组的,前者是用来配置堆区的,后者是用来配置非堆区的。
    第一组配置参数:-Xms 、-Xmx、-XX:newSize、-XX:MaxnewSize、-Xmn

    
    一、-Xms :表示java虚拟机堆区内存初始内存分配的大小,一般为操做系统可用内存的1/64大小便可,但仍需按照实际状况进行分配。有可能真的按照这样的一个规则分配时,设计出的软件尚未可以运行得起来就挂了。
    二、-Xmx: 表示java虚拟机堆区内存可被分配的最大上限,一般为操做系统可用内存的1/4大小。可是开发过程当中,一般会将 -Xms 与 -Xmx两个参数的配置相同的值,其目的是为了可以在java垃圾回收机制清理完堆区后不须要从新分隔计算堆区的大小而浪费资源。

接下来要讲述的三个参数是用来控制新生代内存大小的。

    一、-XX:newSize:表示新生代初始内存的大小,应该小于 -Xms的值;
    二、-XX:MaxnewSize:表示新生代可被分配的内存的最大上限;固然这个值应该小于 -Xmx的值;
    三、-Xmn:至于这个参数则是对 -XX:newSize、-XX:MaxnewSize两个参数的同时配置,也就是说若是经过-Xmn来配置新生代的内存大小,那么-XX:newSize = -XX:MaxnewSize = -Xmn,虽然会很方便,但须要注意的是这个参数是在JDK1.4版本之后才使用的。

 

上面所述即为java虚拟机对外提供的可配置堆区的参数,接下来说述java虚拟机对非堆区内存配置的两个参数:

    一、-XX:PermSize:表示非堆区初始内存分配大小,其缩写为permanent size(持久化内存)
    二、-XX:MaxPermSize:表示对非堆区分配的内存的最大上限。

相关文章
相关标签/搜索