`
javaEE_dev
  • 浏览: 26401 次
  • 性别: Icon_minigender_1
  • 来自: 西安
最近访客 更多访客>>
文章分类
社区版块
存档分类
最新评论

Java内存分析

阅读更多
在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);
        }
}以下为对内存的分析:



成员变量:方法外部,类的内部定义的变量。
局部变量:方法或语句块内部定义的变量。

再贴上一张程序执行过程的图片(截取自尚学堂):
分享到:
评论

相关推荐

    jProfiler7 java内存分析 linux版本

    本篇文章将详细探讨jProfiler7在Java内存分析上的核心功能、使用方法以及在Linux环境中的配置和应用。 1. **内存分析概述** - 内存分析是识别和解决Java应用程序中的内存泄漏、过度对象创建和内存消耗过高问题的...

    JAVA内存分析 - V1.0.0.zip

    Java内存分析是Java开发中非常重要的一个环节,它关乎到应用程序的性能和稳定性。"JAVA内存分析 - V1.0.0.zip" 提供了一套工具来帮助开发者深入理解并优化Java应用程序的内存使用情况。这个压缩包包含了两个关键文件...

    java内存解析(讲的很清楚的视频)

    java内存解析(讲的很清楚的视频)java内存解析(讲的很清楚的视频)java内存解析(讲的很清楚的视频)java内存解析(讲的很清楚的视频)java内存解析(讲的很清楚的视频)java内存解析(讲的很清楚的视频)java内存...

    Java 内存分析工具

    Java内存分析工具是Java开发过程中不可或缺的调试利器,主要用于检测和解决内存泄漏问题。内存泄漏是程序运行过程中,无法释放不再使用的内存空间,随着时间推移,会导致系统资源耗尽,性能急剧下降,甚至可能导致...

    java内存分析-内存泄露问题.rar

    Java内存分析是一个重要的主题,尤其是在开发复杂且性能要求高的应用程序时。内存泄露是Java程序员需要密切关注的问题,因为它们可能导致系统性能下降,甚至可能导致系统崩溃。本文将深入探讨Java内存分析和内存泄露...

    Mac OS java内存分析工具MAT

    **Mac OS上的Java内存分析工具MAT** 在Java应用程序开发中,内存管理是至关重要的,特别是对于大型、长期运行的应用程序,内存泄漏可能导致性能下降甚至系统崩溃。为了有效地诊断和解决这些问题,开发人员需要借助...

    Java内存分析工具ji

    Java内存分析工具ji Java内存分析工具ji是Java程序员不可或缺的工具之一,它可以帮助开发者快速地排查和解决Java内存泄漏问题。在本文中,我们将详细介绍Java内存泄漏的定义、原因、排查方法、解决方案,并提供一个...

    java内存分析工具dumpAnalyzer

    Java内存分析是优化应用程序性能的关键环节,特别是在处理大型或资源密集型应用时。"dumpAnalyzer"是一个专门用于此目的的工具,它提供了图形化的界面,帮助开发者深入理解系统的内存状况,找出潜在的性能瓶颈。本文...

    java内存分析工具介绍

    Java内存分析工具是Java开发过程中不可或缺的利器,主要用于检测和解决程序中的内存泄漏问题,以及进行性能监控。本文将深入探讨几种常用的Java内存分析工具,它们的特点、使用方法以及如何利用这些工具有效地优化...

    ibm HeapAnalyzer java内存分析工具 ha457.jar

    IBM HeapAnalyzer是一款强大的Java内存分析工具,主要用于诊断和解决Java应用程序中的内存泄漏问题。这款工具通过对Java堆内存的深入分析,帮助开发者定位那些占用过多内存的对象,从而优化应用性能。在Java开发过程...

    java内存分析常用工具和指令

    介绍了几种jdk自带的内存分析工具的简单使用,和一些java内置的命令

    基于eclipse的java内存分析工具MAT

    Java内存分析是优化应用程序性能和解决内存泄漏问题的关键步骤,尤其在使用Eclipse IDE时,MAT(Memory Analyzer Tool)是一个非常强大的工具。MAT基于Eclipse平台,为Java开发者提供了深入的内存剖析功能,帮助我们...

    MAT-Memory Analyzer Tool Java内存泄漏分析工具1.5

    MAT 是一个开源的java内存分析工具,能够快速的分析dump文件,可以直观的看到各个对象在内存占用的量大小,以及类实例的数量,对象之间的引用关系,找出对象的GC Roots相关的信息,此外还能生成内存泄露报表,疑似...

    Java内存分配全面解析

    Java内存分配全面解析 Java程序在执行过程中,其内存分配主要涉及到JVM(Java Virtual Machine,Java虚拟机)的不同区域。这些区域包括寄存器、栈、堆、常量池、代码段和数据段。理解这些内存区域的工作原理对于...

    MAT Java内存分析工具

    MAT(Memory Analyzer Tool)是Eclipse基金会开发的一款强大的Java内存分析工具,专为解决Java应用程序的内存泄漏和性能优化问题而设计。它提供了丰富的功能,包括堆dump分析、对象引用链追踪、内存消耗报告等,使得...

    Java内存分析技术在教学中的应用.pdf

    Java内存分析技术在教学中的应用 Java内存分析技术在教学中的应用是Java程序设计课程教学中的一件利器。本文介绍了Java内存分析技术及其在教学中的典型应用。Java程序运行在Java虚拟机(JVM)中,与其他一些语言...

    Java+内存分析工具+MAT

    Java内存分析是一个关键的优化步骤,特别是在开发大型的、高性能的应用程序时。MAT(Memory Analyzer Tool)是由IBM开发的一款强大的Java堆内存分析器,它的全名是Eclipse Memory Analyzer。这款工具是开源的,完全...

    JAVA内存泄漏分析工具

    MAT(Memory Analyzer Tool)是IBM开发的一款强大的Java内存分析工具,它是标题中提到的具体分析工具之一。MAT提供了一种可视化的界面,通过分析heap dump文件,能够帮助我们深入理解Java应用程序的内存使用情况,找...

    JAVA内存分析指引201007_V0.2

    ### JAVA内存分析指引 #### 一、环境说明与配置 在深入探讨JAVA内存分析之前,有必要先理解本文档所基于的环境。文档指出,生产环境中主要使用的是WebSphere5和WebSphere6,这两种版本的安装目录及日志路径在...

    java内存分析培训材料0918.pptx

    java内存分析培训材料,包括jvm内存分析工具、JVM工具与MAT介绍

Global site tag (gtag.js) - Google Analytics