Java内存分析

 
2009-03-03 10:39:30
 
在JAVA中,有六个不同的地方可以存储数据:
1. 寄存器(register)。这是最快的存储区,因为它位于不同于其他存储区的地方——处理器内部。但是寄存器的数量极其有限,所以寄存器由编译器根据需求进行分配。你不能直接控制,也不能在程序中感觉到寄存器存在的任何迹象。
2. 堆栈(stack)。位于通用RAM中,但通过它的“堆栈指针”可以从处理器哪里获得支持。堆栈指针若向下移动,则分配新的内存;若向上移动,则释放那些内存。这是一种快速有效的分配存储方法,仅次于寄存器。创建程序时候,JAVA编译器必须知道存储在堆栈内所有数据的确切大小和生命周期,因为它必须生成相应的代码,以便上下移动堆栈指针。这一约束限制了程序的灵活性,所以虽然某些JAVA数据存储在堆栈中——特别是对象引用,但是JAVA对象不存储其中。
3. 堆(heap)。一种通用性的内存池(也存在于RAM中),用于存放所以的JAVA对象。堆不同于堆栈的好处是:编译器不需要知道要从堆里分配多少存储区域,也不必知道存储的数据在堆里存活多长时间。因此,在堆里分配存储有很大的灵活性。当你需要创建一个对象的时候,只需要new写一行简单的代码,当执行这行代码时,会自动在堆里进行存储分配。当然,为这种灵活性必须要付出相应的代码。用堆进行存储分配比用堆栈进行存储存储需要更多的时间。
4. 静态存储(static storage)。这里的“静态”是指“在固定的位置”。静态存储里存放程序运行时一直存在的数据。你可用关键字static来标识一个对象的特定元素是静态的,但JAVA对象本身从来不会存放在静态存储空间里。
5. 常量存储(constant storage)。常量值通常直接存放在程序代码内部,这样做是安全的,因为它们永远不会被改变。有时,在嵌入式系统中,常量本身会和其他部分分割离开,所以在这种情况下,可以选择将其放在ROM中
6. 非RAM存储。如果数据完全存活于程序之外,那么它可以不受程序的任何控制,在程序没有运行时也可以存在。
 
上面这段话摘取之《Thinking in Java》』
 
---------------------------------------------------------------------
堆是一个运行时数据区,类的对象从中分配空间。这些对象通过new建立,它们不需要程序代码来显式的释放。堆是由垃圾回收来负责的,堆的优势是可以动态地分配内存大小,生存期也不必事先告诉编译器,因为它是在运行时动态分配内存的,Java的垃圾收集器会自动收走这些不再使用的数据。但缺点是,由于要在运行时动态分配内存,存取速度较慢。java中的对象和数组都存放在堆中。
栈的优势是,存取速度比堆要快,仅次于寄存器,栈数据可以共享。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。栈中主要存放一些基本类型的变量(,int, short, long, byte, float, double, boolean, char)和对象引用。
栈有一个很重要的特殊性,就是存在栈中的数据可以共享。假设我们同时定义:
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, 它是由编译器完成的,它有利于节省空间。而一个对象引用变量修改了这个对象的内部状态,会影响到另一个对象引用变量。
 
以上内容也是摘抄自网上。
 
---------------------------------------------------------------------
下面我自己来举几个例子:
public class TestStr { 
  public static void main(String[] args) { 
    // 以下两条语句创建了1个对象。"凤山"存储在字符串常量池中 
    String str1 = "凤山"
    String str2 = "凤山"
    System.out.println(str1==str2);//true 
     
    //以下两条语句创建了3个对象。"天峨",存储在字符串常量池中,两个new String()对象存储在堆内存中 
    String str3 = new String("天峨"); 
    String str4 = new String("天峨"); 
    System.out.println(str3==str4);//false 
     
    //以下两条语句创建了1个对象。9是存储在栈内存中 
    int i = 9; 
    int j = 9; 
    System.out.println(i==j);//true 
     
      //以下两条语句创建了1个对象。1对象存储在栈内存中 
    Integer l = 1;//装箱 
    Integer k = 1;//装箱 
    System.out.println(l==k);//true 
     
    //由于没有了装箱,以下两条语句创建了2个对象。两个1对象存储在堆内存中 
    Integer l1 = new Integer(1); 
    Integer k1 = new Integer(1); 
    System.out.println(l1==k1);//false 
     
    //以下两条语句创建了1个对象。i1,i2变量存储在栈内存中,两个256对象存储在堆内存中 
         Integer i1 = 256; 
                Integer i2 = 256; 
         System.out.println(i1==i2);//false 
  } 
 
 
对于以上最后两个关于Integer对象的例子,在自动装箱时对于值从–128到127之间的值,使用一个实例。
下面是对字符串常量池()的一个例子:
String s1 = "aaa" + "bbb"; //产生了1个对象。 
由于常量的值在编译的时候就被确定了。在这里,"aaa"和"bbb"都是常量,因此变量s1 的值在编译时就可以确定。这行代码编译后的效果等同于: 
String s1 ="aaabbb"; 
因此这里只创建了一个对象"aaabbb",并且它被保存在字符串池里了。
 
String str1 = "凤山";
String str2 = "凤山";
以上两条语句只在常量池中保存了一个"凤山"对象。
 
String str3 = new String("天峨");
String str4 = new String("天峨");
以上两条语句创建了3个对象,首先在字符串常量池中创建一个"天峨"对象,接着在堆内存中创建两个new String()对象,里面保存的是指向"天峨"对象的引用。
另:“==“在判断对象时,其实是根据对象在堆栈中的地址判断对象是不是一样,而不是根据hashcode 值。
 
 
在网上看见这段对Java String中的HashCode和equal的总结比较好,记录如下:
    1. hashSet中比较是否重复的依据是a.hasCode()=b.hasCode() && a.equals(b)
    2. String的hashCode依据: 以依赖于char[i]的int值以和char[i]的排列序的算法计算出的(可以去看看源码)。不依赖String的ref.
    3. String的equals依据: a==b || ( a.length=b.length && { a[i]=b[i] } )
    4. 只有用a==b时比校的才是比校的ref,也就是说这时才是比校是a与b是不是同一个对象
    5. 结论: 两个不同ref的String可能会被认为是集合中的同一个元素。
 
---------------------------------------------------------------------
下面分析一个代码示例:
class BirthDate { 
        private int day; 
        private int month; 
        private int year;         
        public BirthDate(int d, int m, int y) { 
                day = d;    
                month = m;    
                year = y; 
        } 
        省略get,set方法。。。 
        public void display() { 
        System.out.println 
                (day + " - " + month + " - " + year); 
        } 

    
public class Test{ 
        public static void main(String args[]){ 
                Test test = new Test(); 
                int date = 9; 
                BirthDate d1= new BirthDate(7,7,1970); 
                BirthDate d2= new BirthDate(1,1,2000);         
                test.change1(date); 
                test.change2(d1); 
                test.change3(d2); 
                d1.display(); 
                d2.display(); 
        } 
         
        public void change1(int i){ 
        i = 1234; 
        } 
         
        public void change2(BirthDate b) { 
        b = new BirthDate(22,2,2004); 
        } 
         
        public void change3(BirthDate b) { 
        b.setDay(22); 
        } 
}
 
以下为对内存的分析:
 
 
 
成员变量:方法外部,类的内部定义的变量。
局部变量:方法或语句块内部定义的变量。
 
再贴上一张程序执行过程的图片(截取自尚学堂):
 
 

本文出自 “青山” 博客,请务必保留此出处http://java999.blog.51cto.com/259217/134359

 
更多0
 
 卡卡罗宾逊
1人
了这篇文章
类别:J2se复习┆技术圈(0)┆阅读(3258)┆评论(5) ┆ 推送到技术圈返回首页
 

文章评论

 
2009-03-03 11:39:50
内存分析这部分内容讲的很好

2009-08-07 10:59:17
String str3 = new String("天峨"); 
String str4 = new String("天峨"); 
str3和str4的hashCode值是一样的呀?

2009-08-07 11:01:31
hashCode值一样也就是三个对象吧

2009-08-14 09:45:26
回复 abccof:[3楼]
判断是不是相同对象是根据引用地址来判断的(也就是通过==来判断),对于
String str3 = new String("天峨"); 
String str4 = new String("天峨"); 
因为两个new String()对象的值("天峨")是一样的,所以他们的hashcode是一样的,但他们的引用时不一样的。
对于这两个语句可以这样理解生成几个对象:
先是在字符串常量池中生成了("天峨")这个对象,然后在栈内存中生成两个new String()对象,他们保存的内容为指向常量池中的("天峨")对象的引用。因此为产生了3个对象。

2012-04-05 06:45:03
上面string内容有误! "123"+"123";不会在string连接池中生成"123123" 需要intern一下 ~~

 

posted @   歌颂者  阅读(256)  评论(0编辑  收藏  举报
编辑推荐:
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
· Linux系统下SQL Server数据库镜像配置全流程详解
· 现代计算机视觉入门之:什么是视频
阅读排行:
· Sdcb Chats 技术博客:数据库 ID 选型的曲折之路 - 从 Guid 到自增 ID,再到
· 99%的人不知道,桥接模式失败的真正原因是它!
· .NET Core GC压缩(compact_phase)底层原理浅谈
· Winform-耗时操作导致界面渲染滞后
· Phi小模型开发教程:C#使用本地模型Phi视觉模型分析图像,实现图片分类、搜索等功能
点击右上角即可分享
微信分享提示