Java内存分析

本文涉及的产品
对象存储 OSS,OSS 加速器 50 GB 1个月
简介:
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); 
        } 
}
以下为对内存的分析:
 
 
成员变量:方法外部,类的内部定义的变量。
局部变量:方法或语句块内部定义的变量。
 
再贴上一张程序执行过程的图片(截取自尚学堂):









本文转自 yzzh9 51CTO博客,原文链接:http://blog.51cto.com/java999/134359,如需转载请自行联系原作者
相关实践学习
对象存储OSS快速上手——如何使用ossbrowser
本实验是对象存储OSS入门级实验。通过本实验,用户可学会如何用对象OSS的插件,进行简单的数据存、查、删等操作。
目录
相关文章
|
9月前
|
安全 Java 应用服务中间件
Spring Boot + Java 21:内存减少 60%,启动速度提高 30% — 零代码
通过调整三个JVM和Spring Boot配置开关,无需重写代码即可显著优化Java应用性能:内存减少60%,启动速度提升30%。适用于所有在JVM上运行API的生产团队,低成本实现高效能。
1073 3
|
10月前
|
安全 Java 编译器
new出来的对象,不一定在堆上?聊聊Java虚拟机的优化技术:逃逸分析
逃逸分析是一种静态程序分析技术,用于判断对象的可见性与生命周期。它帮助即时编译器优化内存使用、降低同步开销。根据对象是否逃逸出方法或线程,分析结果分为未逃逸、方法逃逸和线程逃逸三种。基于分析结果,编译器可进行同步锁消除、标量替换和栈上分配等优化,从而提升程序性能。尽管逃逸分析计算复杂度较高,但其在热点代码中的应用为Java虚拟机带来了显著的优化效果。
319 4
|
10月前
|
存储 缓存 Java
Java数组全解析:一维、多维与内存模型
本文深入解析Java数组的内存布局与操作技巧,涵盖一维及多维数组的声明、初始化、内存模型,以及数组常见陷阱和性能优化。通过图文结合的方式帮助开发者彻底理解数组本质,并提供Arrays工具类的实用方法与面试高频问题解析,助你掌握数组核心知识,避免常见错误。
|
12月前
|
存储 弹性计算 缓存
阿里云服务器ECS经济型、通用算力、计算型、通用和内存型选购指南及使用场景分析
本文详细解析阿里云ECS服务器的经济型、通用算力型、计算型、通用型和内存型实例的区别及适用场景,涵盖性能特点、配置比例与实际应用,助你根据业务需求精准选型,提升资源利用率并降低成本。
735 3
|
8月前
|
Java 大数据 Go
从混沌到秩序:Java共享内存模型如何通过显式约束驯服并发?
并发编程旨在混乱中建立秩序。本文对比Java共享内存模型与Golang消息传递模型,剖析显式同步与隐式因果的哲学差异,揭示happens-before等机制如何保障内存可见性与数据一致性,展现两大范式的深层分野。(238字)
244 4
|
8月前
|
存储 缓存 Java
【深入浅出】揭秘Java内存模型(JMM):并发编程的基石
本文深入解析Java内存模型(JMM),揭示synchronized与volatile的底层原理,剖析主内存与工作内存、可见性、有序性等核心概念,助你理解并发编程三大难题及Happens-Before、内存屏障等解决方案,掌握多线程编程基石。
|
8月前
|
存储 Java Go
【Java】(3)8种基本数据类型的分析、数据类型转换规则、转义字符的列举
牢记类型转换规则在脑海中将编译和运行两个阶段分开,这是两个不同的阶段,不要弄混!
364 2
|
8月前
|
设计模式 缓存 Java
【JUC】(4)从JMM内存模型的角度来分析CAS并发性问题
本篇文章将从JMM内存模型的角度来分析CAS并发性问题; 内容包含:介绍JMM、CAS、balking犹豫模式、二次检查锁、指令重排问题
214 1
|
8月前
|
Java Go 开发工具
【Java】(9)抽象类、接口、内部的运用与作用分析,枚举类型的使用
抽象类必须使用abstract修饰符来修饰,抽象方法也必须使用abstract修饰符来修饰,抽象方法不能有方法体。抽象类不能被实例化,无法使用new关键字来调用抽象类的构造器创建抽象类的实例。抽象类可以包含成员变量、方法(普通方法和抽象方法都可以)、构造器、初始化块、内部类(接 口、枚举)5种成分。抽象类的构造器不能用于创建实例,主要是用于被其子类调用。抽象类中不一定包含抽象方法,但是有抽象方法的类必定是抽象类abstract static不能同时修饰一个方法。
335 1
|
9月前
|
缓存 监控 Kubernetes
Java虚拟机内存溢出(Java Heap Space)问题处理方案
综上所述, 解决Java Heap Space溢出需从多角度综合施策; 包括但不限于配置调整、代码审查与优化以及系统设计层面改进; 同样也不能忽视运行期监控与预警设置之重要性; 及早发现潜在风险点并采取相应补救手段至关重要.
1017 17