JVM堆栈,类加载过程

Java堆、栈

JAVA在程序运行时,在内存中划分5片空间进行数据的存储。分别是:1:寄存器。2:本地方法区。3:方法区。4:栈。5:堆。

基本要点:

基本数据类型、局部变量都是存放在栈内存中的,用完就消失。 new创建的实例化对象及数组,是存放在堆内存中的,用完之后靠垃圾回收机制不定期自动消除。

示例

  1. main()
    int x=1;
    show ()
    int x=2

    主函数main()中定义变量int x=1,show()函数中定义变量int x=1。最后show()函数执行完毕。

    以上程序执行步骤:

    第1步——main()函数是程序入口,JVM先执行,在栈内存中开辟一个空间,存放int类型变量x,同时附值1。 第2步——JVM执行show()函数,在栈内存中又开辟一个新的空间,存放int类型变量x,同时附值2。  此时main空间与show空间并存,同时运行,互不影响。 第3步——show()执行完毕,变量x立即释放,空间消失。但是main()函数空间仍存在,main中的变量x仍然存在,不受影响。

    示意图如下:

  2. main()
    int[] x=new int[3];
    x[0]=20

    主函数main()中定义数组x,元素类型int,元素个数3。

    以上程序执行步骤 第1步——执行int[] x=new int[3]; 隐藏以下几分支 JVM执行main()函数,在栈内存中开辟一个空间,存放x变量(x变量是局部变量)。 同时,在堆内存中也开辟一个空间,存放new int[3]数组,堆内存会自动内存首地址值,如0x0045。 数组在栈内存中的地址值,会附给x,这样x也有地址值。所以,x就指向(引用)了这个数组。此时,所有元素均未附值,但都有默认初始化值0。

    第2步——执行x[0]=20 即在堆内存中将20附给[0]这个数组元素。这样,数组的三个元素值分别为20,0,0

    示图如下:

  3. main()
    int[] x=new int[3];
    x[0]=20
    x=null;

    以上步骤执行步骤 第1、2步——与示例2完全一样,略。

    第3步——执行x=null; null表示空值,即x的引用数组内存地址0x0045被删除了,则不再指向栈内存中的数组。此时,堆中的数组不再被x使用了,即被视为垃圾,JVM会启动垃圾回收机制,不定时自动删除。

    示图如下

  4. main()
    int[] x=new int[3];
    int[] y=x;
    y[1]=100
    x=null;

    以上步骤执行步骤

    第1步——与示例2第1步一致,略。 第2步——执行int[] y=x, 在栈内存定义了新的数组变量内存y,同时将x的值0x0045附给了y。所以,y也指向了堆内存中的同一个数组。 第3步——执行y[1]=100 即在堆内存中将20附给[0]这个数组元素。这样,数组的三个元素值分别为0,100,0 第4步——执行x=null 则变量x不再指向栈内存中的数组了。但是,变量y仍然指向,所以数组不消失。

    示图如下

  5. Car c=new Car;
    c.color="blue";
    Car c1=new Car;
    c1.num=5;

    虽然是个对象都引用new Car,但是是两个不同的对象。每一次new,都产生不同的实体

  6. Car c=new Car;
    c.num=5;
    Car c1=c;
    c1.color="green";
    c.run();

    Car c1=c,这句话相当于将对象复制一份出来,两个对象的内存地址值一样。所以指向同一个实体,对c1的属性修改,相当于c的属性也改了。

栈:

函数中定义的基本类型变量,对象的引用变量都在函数的栈内存中分配。 栈内存特点,数数据一执行完毕,变量会立即释放,节约内存空间。 栈内存中的数据,没有默认初始化值,需要手动设置。

堆:

堆内存用来存放new创建的对象和数组。 堆内存中所有的实体都有内存地址值。 堆内存中的实体是用来封装数据的,这些数据都有默认初始化值。 堆内存中的实体不再被指向时,JVM启动垃圾回收机制,自动清除,这也是JAVA优于C++的表现之一(C++中需要程序员手动清除)。

Java 中的堆是 JVM 所管理的最大的一块内存空间,主要用于存放各种类的实例对象。   在 Java 中,堆被划分成两个不同的区域:新生代 ( Young )、老年代 ( Old )。新生代 ( Young ) 又被划分为三个区域:Eden、From Survivor、To Survivor。   这样划分的目的是为了使 JVM 能够更好的管理堆内存中的对象,包括内存的分配以及回收。 新生代:Young Generation,主要用来存放新生的对象。   老年代:Old Generation或者称作Tenured Generation,主要存放应用程序声明周期长的内存对象。

永久代:(方法区,不属于java堆,另一个别名为“非堆Non-Heap”但是一般查看PrintGCDetails都会带上PermGen区)是指内存的永久保存区域,主要存放Class和Meta的信息,Class在被 Load的时候被放入PermGen space区域. 它和和存放Instance的Heap区域不同,GC(Garbage Collection)不会在主程序运行期对PermGen space进行清理,所以如果你的应用会加载很多Class的话,就很可能出现PermGen space错误

回收方法区(附加补充)   很多人认为方法区(或者HotSpot虚拟机中的永久代[PermGen])是没有垃圾收集的,java虚拟机规范中确实说过可以不要求虚拟机在方法区实现垃圾收集,而且在方法去中进行垃圾收集的“性价比”一般比较低:在堆中,尤其是在新生代中,常规应用进行一次垃圾收集一般可以回收70%-95%的空间,而永久代的垃圾收集效率远低于此。   永久代的垃圾收集主要回收两部分内容:废弃的常量和无用的类。   废弃的常量:回收废弃常量与回收java堆中的对象非常类似。以常量池字面量的回收为例,加入一个字符串“abc"已经进入了常量池中,但是当前系统没有任何一个String对象是叫做"abc"的,换句话说,就是有任何String对象应用常量池中的"abc"常量,也没有其他地方引用了这个字面量,如果这时发生内存回收,而且必要的话,这个“abc”常量就会被系统清理出常量池。常量池中的其他类(接口)、方法、字段的符号引用也与此类似。(注:jdk1.7及其之后的版本已经将字符串常量池从永久代中移出)   无用的类:类需要同时满足下面3个条件才能算是“无用的类”:

  • 该类所有的实例都已经被回收,也就是java堆中不存在该类的任何实例。

  • 加载该类的ClassLoader已经被回收

  • 该类对应的java.lang.Class对象没有在任何地方被引用,无法在任何地方通过反射访问该类的方法。

JVM类加载过程

  1. 加载 虚拟机需完成以下三件事:

    • 通过一个类的全限定名来获取定义此类的二进制字节流

    • 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构

    • 在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问接口

  2. 连接

    2.1 验证:确保被加载的类的正确性 确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全

    • 文件格式验证:验证字节流是否符合Class文件格式的规范,如:是否以模数0xCAFEBABE开头,主次版本号是否在当前虚拟机处理范围内等等

    • 元数据验证:对字节码描述的信息进行语义分析,以保证其描述的信息符合Java语言规范的要求:如:这个类是否有父类,是否实现了父类的抽象方法,是否重写了父类的final方法,是否继承了被final修饰的类等

    • 字节码验证:通过数据流和控制流分析,确定程序语义是合法的,符合逻辑的,

    • 符号引用验证:确保解析动作能正确执行;如:通过符号引用能找到对应的类和方法,符号引用中类,属性,方法的访问属性是否能被当前类访问等

    2.2 准备:为类的静态变量分配内存,并将其赋默认值 为类变量分配内存并设置类变量初始值,这些内存都将在方法区中分配,注意

    • 只对static修饰的静态变量进行内存分配,赋默认值(如0,0L,null,false等)

    • 对final修饰的常量直接赋初值

    2.3 解析: 将常量池中的符号引用替换为直接引用(内存地址)的过程

    符号引用:一组符号来描述目标,可以是任何字面量,属于编译原理方面的概念如:包括和接口的全限定名,字段的名称和描述符,方法的名称和描述符 ​ 直接引用:直接指向目标的指针,相对偏移量或一个间接定位到目标的句柄,如指向方法区某个类的一个指针

    假设:一个类有一个静态变量,该静态变量是一个自定义的类型,那么经过解析后,该静态变量将是一个指针,指向该类在方法区的内存地址

  3. 初始化:为类的静态变量赋初值 定义静态变量时指定初始值,如private static String x = “123”;

    在静态代码块里为静态变量赋值,如static{x=“123”;}

    clinit与init

    在编译生成class文件时,编译器会产生两个方法加于class文件中,一个是类的初始化方法clinit,另一个是实例的初始化方法init

    clinit:

    clinit指的是类构造器,主要作用是在类加载过程中的初始化阶段进行执行,执行内容包括静态变量初始化和静态块的执行

    注意:

    • 如果类中没有静态变量或静态代码块,那么clinit方法不会被生成

    • 在执行clinit方法时,必须先执行父类的clinit方法。

    • clinit方法只执行一次

    init:

    init指的是实例构造器,主要作用是在类实例化过程中执行,执行内容包括成员变量初始化和代码块的执行。

    注意:

    • 如果类中没有成员变量和代码块,那么init方法不会被生成

    • 在执行init方法时,必须先执行父类的init方法

    • init方法没实例化一次就会执行一次

    • init方法先为实例变量分配内存空间,再执行默认值,然后根据源码中的顺序执行赋初值或代码块

上一篇:guava缓存的get方法的回调函数讲解一下


下一篇:Java Web 4 Maven