Java内存管理的进一步理解-模拟过程图解

Java内存管理的进一步理解-模拟过程图解--转载

java的内存管理分为:

1、堆内存;2、栈内存;3、方法区;4、本地方法区

/*

1:方法区 

    方法区存放装载的类数据信息包括: 

    (1):基本信息: 

    1)每个类的全限定名 

     2)每个类的直接超类的全限定名(可约束类型转换) 

    3)该类是类还是接口 

    4)该类型的访问修饰符 

    5)直接超接口的全限定名的有序列表 

    (2):每个已装载类的详细信息: 

    1)运行时常量池: 

    存放该类型所用的一切常量(直接常量和对其它类型、字段、方法的符号引用),它们以数组形式通过索引被访问,是外部调用与类联系及类型对象化的桥梁。它是类文件(字节码)常量池的运行时表示。(还有一种静态常量池,在字节码文件中)。 

    2)字段信息: 

    类中声明的每一个字段的信息(名,类型,修饰符)。 

    3)方法信息: 

    类中声明的每一个方法的信息(名,返回类型,参数类型,修饰符,方法的字节码和异常表)。 

    4)静态变量 

    5)到类 classloader 的引用:即到该类的类装载器的引用。 

    6)到类 class 的引用: 

    虚拟机为每一个被装载的类型创建一个 class 实例, 用来代表这个被装载的类。 

2:栈内存 

    Java栈内存以帧的形式存放本地方法的调用状态(包括方法调用的参数,局部变量,中间结果等)。每调用一个方法就将对应该方法的方法帧压入 Java 栈,成为当前方法帧。当调用结束(返回)时,就弹出该帧。 

     编译器将源代码编译成字节码(.class)时, 就已经将各种类型的方法的局部变量, 操作数栈大小确定并放在字节码中,随着类一并装载入方法区。当调用方法时,通过访问方法区中的类的信息,得到局部变量以及操作数栈的大小。 

    也就是说: 在方法中定义的一些基本类型的变量和对象的引用变量都在方法的栈内存中分配。 当在一段代码块定义一个变量时,Java 就在栈中为这个变量分配内存空间,当超过变量的作用域后,Java 会自动释放掉为该变量所分配的内存空间, 该内存空间可以立即被另作它用。 

栈内存的构成: 

    Java 栈内存由局部变量区、操作数栈、帧数据区组成。 

    (1):局部变量区为一个以字为单位的数组,每个数组元素对应一个局部变量的值。调用方法时,将方法的局部变量组成一个数组,通过索引来访问。若为非静态方法,则加入一个隐含的引用参数 this,该参数指向调用这个方法的对象。而静态方法则没有this参数。因此,在静态方法里无法访问对象信息。 

    (2):操作数栈也是一个数组,但是通过栈操作来访问。所谓操作数是那些被指令操 

作的数据。当需要对参数操作时如 a=b+c,就将即将被操作的参数压栈,如将 b 和 c 压栈, 

然后由操作指令将它们弹出,并执行操作。虚拟机将操作数栈作为工作区。 

    (3):帧数据区处理常量池解析,异常处理等 

3:堆内存 

    堆内存用来存放由 new 创建的对象和数组。在堆中分配的内存,由 Java 虚拟机的自动垃圾回收器来管理。 

    在堆中产生了一个数组或对象后,还可以在栈中定义一个特殊的变量, 让栈中这个变量的取值等于数组或对象在堆内存中的首地址,栈中的这个变量就成了数组或对象的引用变量。 
    引用变量就相当于是为数组或对象起的一个名称, 以后就可以在程序中使用栈中的引用变量来访问堆中的数组或对象。 

4:本地方法栈内存 

    与调用的本地方法的语言相关,如:调用的是一个c语言方法则为一个c栈。本地方法可以回调java方法。若有java方法调用本地方法,虚拟机就运行这个本地方法。 

    在虚拟机看来运行这个本地方法就是执行这个java方法,如果本地方法抛出异常,虚拟机就认为是这个java方法抛出异常。 

    Java通过Java本地接口JNI(Java Native Interface)来调用其它语言编写的程序,在Java里面用native修饰符来描述一个方法是本地方法。

*/

下面通过一个简单的代码示例,理解Java中,内存是怎么进行分配与管理的。示例如下:

public class JavaRamExplain {
    public static void main(String[] args) {
        Person.whatCountry();  

        Person sam = new Person("sam", 20);
        sam.introduceSelf();
    }
}
class Person {  

    private String name;
    private int age;
    public static String country = "china";  

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }  

    public void introduceSelf() {
        System.out.println("My name‘s " + name + ",i‘m " + age + " years old.");
    }  

    public static void whatCountry() {
        System.out.println("This people come from " + country);
    }
} 

当开始运行JavaRamExplain类时,JVM便会开始进行内存的分配管理工作。

图解过程为:

(图片较大,看不清可以图片右键-查看图像,进行查看原图)

可以看到到第八步工作结束后:

栈内存中的数据全部被释放(所以说栈内存中的数据的生命周期是已知并固定的,因为随着方法的执行结束,栈内存便会进行释放);

堆内存中的Person对象,现已没有任何对象引用指向它,所以它将被视作内存中的“垃圾”,等待回收。(堆内存中的数据由java的垃圾回收机制自动进行处理,所以其生命周期不确定)

由图中也可以看到:

类的方法会进行压栈和弹栈,对应的,方法中用到的参数(变量)便会相应的在栈内存中进行存储和释放。所以栈内存用于存放局部变量(包括基本类型和引用类型)。

类的对象会被存放到堆内存中,相应的该对象所包含的成员变量也会被存放到该空间。

类本身将在jvm通过.class运行该类时就会被加载到内存中的方法区内,顾名思义,方法区用于保存类的方法代码,其中普通方法被存放在非静态区,静态成员(方法和变量)被存放在方法区中专门的静态区当中。

栈内存是方法的运行区域(因为方法中定义的局部变量需要在栈里开辟空间),方法区是方法的存放区域。

原文地址:https://www.cnblogs.com/lipeineng/p/8358823.html

时间: 2024-11-29 09:04:09

Java内存管理的进一步理解-模拟过程图解的相关文章

Java内存管理第二篇 - 内存的分配

Java内存管理无非就是对内存进行分配和释放.对于分配来说,基本类型和对象的引用存储到栈中,常量存储到常量池中,对象存储到堆上,这是一般的分配.而对于回收来说要复杂的多,如果回收不好,还可能造成分配出去的内存得不到回收而造成内存泄漏. 这一篇将简单介绍一下Java内存的分配,下一篇将介绍内存的回收及内存泄漏等知识. 1.JVM内存模型 1.程序计数器(Program Counter Register): 程序计数器是一个比较小的内存区域,用于指示当前线程所执行的字节码执行到了第几行,可以理解为是

java内存管理机制

JAVA 内存管理总结 1. java是如何管理内存的 Java的内存管理就是对象的分配和释放问题.(两部分) 分配 :内存的分配是由程序完成的,程序员需要通过关键字new 为每个对象申请内存空间 (基本类型除外),所有的对象都在堆 (Heap)中分配空间.释放 :对象的释放是由垃圾回收机制决定和执行的,这样做确实简化了程序员的工作.但同时,它也加重了JVM的工作.因为,GC为了能够正确释放对象,GC必须监控每一个对象的运行状态,包括对象的申请.引用.被引用.赋值等,GC都需要进行监控. 2. 

Java内存管理第三篇 - 内存可能产生的问题

Java内存在分配和回收的过程中会产品很多的问题,下面来说一说可能会产生的问题. 1.垃圾处理 从程序运行的根节点出发,遍历整个对象引用,查找存活的对象.那么在这种方式的实现中,垃圾回收从哪儿开始的呢?即,从哪儿开始查找哪些对象是正在被当前系统使用的.上面分析的堆和栈的区别,其中栈是真正进行程序执行地方,所以要获取哪些对象正在被使用,则需要从Java栈开始.同时,一个栈是与一个线程对应的,因此,如果有多个线程的话,则必须对这些线程对应的所有的栈进行检查. 同时,除了栈外,还有系统运行时的寄存器等

java内存管理浅析

首先感谢强大的网络资源,本博文是根据网络上的各种资源进行整合,然后加入自己的理解而成,可能会与其它网络资源有重复,望其他作者多多包涵.由于初学java,如有不准确的描述还请读者指正.下面正式切入正题: 众所周知,java和C++都是面向对象的编程语言,但是与C++相比,java上手比较容易,而且使用方便.小弟对c++了解不是很多,但是有一点是C++初学者最为头痛的问题,那就是内存管理,这也正是C++和java之间很大的一个区别.在C++中,内存是依靠程序员自己来管理的,编写程序过程中稍有不慎就会

JAVA内存管理再解

首先我们要明白一点,我们所使用的变量就是一块一块的内存空间!! 一.内存管理原理:   在java中,有java程序.虚拟机.操作系统三个层次,其中java程序与虚拟机交互,而虚拟机与操作系统间交互!这就保证了java程序的平台无关性!下面我们从程序运行前,程序运行中.程序运行内存溢出三个阶段来说一下内存管理原理! 1.程序运行前:JVM向操作系统请求一定的内存空间,称为初始内存空间!程序执行过程中所需的内存都是由java虚拟机从这片内存空间中划分的. 2.程序运行中:java程序一直向java

揭开Java内存管理的面纱

前言 相对于C.C++这些高性能语言,Java有着让此类程序员羡慕的功能:内存自动管理.似乎这样,Java程序员不用再关心内存,也不用去了解相关知识.但结果真的是这样吗?特别对于我们这种Android程序员来说,对内存可是吃得死死的,一旦出现较为复杂的内存泄露和溢出方面的问题,简直就是噩梦.因此,对Java内存管理有个大体的了解似乎已经成为一个合格的Android程序员必备的技能,就算是新进的Kotlin同样是基于JVM的.不如趁此机会,大家一起来揭开它的面纱. 对象 Java是一门面向对象的编

java内存管理机制(一)-运行时数据区

前言 本打算花一篇文章来聊聊JVM内存管理机制,结果发现越扯越多,于是分了三遍文章(文章讲解JVM以Hotspot虚拟机为例,jdk版本为1.8),本文为其中第一篇.from java内存管理机制(一)-运行时数据区  1. java内存管理机制-运行时数据区 2. java内存管理机制-内存分配 3. java内存管理机制-垃圾回收 正文 C++与java之间有一堵由内存动态分配和垃圾收集技术所围成的“高墙”,墙外的人想进去,墙里的人却想出来…… 与C.C++程序员时刻要关注着内存的分配与释放

Java内存管理-掌握虚拟机类加载器(五)

勿在流沙筑高台,出来混迟早要还的. 做一个积极的人 编码.改bug.提升自己 我有一个乐园,面向编程,春暖花开! 上一篇介绍虚拟机类加载机制,讲解了类加载机制中的三个阶段,分别是:加载.连接(验证.准备.解析).初始化 ,知道了类加载的机制.下面我们就要知道类到底是通过什么方式加载到内存中的,也就是本文要介绍的类加载器,类加载器就是加载类的信息到内存中. 本文地图 : 一.什么是类加载器(ClassLoader) 虚拟机设计团队把类加载阶段中的”通过一个类的全限定名来获取描述此类的二进制字节流“

java内存管理

一.jvm内存结构 程序计数器(Program Counter Register).JVM虚拟机栈(JVM Stacks).本地方法栈(Native Method Stacks).堆(Heap).方法区(Method Area) (1)PCR 跟随线程生命周期,记录当前执行到的.class字节码行数,用于多线程操作 (2)JVM Stacks 跟随线程生命周期,在方法执行中存储数据 (3)Native Method Stacks 处理native方法,如object中的hashCodes()等