JVM基础解析(一)

 

Java里面有 JDK ,JRE, JVM ,这三者的关系是怎么样的呢?

JDK是编译时环境: 整个Java的核心,包括了Java运行环境JRE、Java工具和Java基础类库

JRE是运行时环境: Java virtual machine(JVM),runtime class libraries和Java application launcher

JVM是运行时环境:整个Java实现跨平台的最核心的部分,所有的Java程序会首先被编译为.class的类文件,这种类文件可以在虚拟机上执行。

JDK包含JRE,JRE包含JVM。

关系如下图:

 

 

 

JVM解析

JVM内存结构如下图:(绿色标记的堆与方法区为线程共享区,所有线程安全问题都是从这儿引发的; 黄色标记区为线程独享区,和线程安全问题无关

程序计数器(Program Counter),Java堆(Heap),Java虚拟机栈(Stack),本地方法栈(Native Stack),方法区(Method Area)

 

 

>> 程序计数器(Program Counter)

  主要执行指令

  每个Java类编译后都会生成一个 *.class文件,例如我定义一个 App.java类,内容如下:

  App.java

package com.imodule.dataImport;

public class App {

	public int add(){
		int a = 11;
		int b = 22;
		int c = (a+b)*10;
		return c;
	}
	
	public static void main(String[] args) {
		App app = new App();
		int result  = app.add();
		System.out.println("result = "+result);
	}

}

  生成的对应的App.class文件内容如下:(这些内容我们看不懂哈哈哈哈,所以下面会使用javap命令来分解)

  App.class

漱壕   4 ;
 
 # $
  #
  %	 & ' (
  # )
  *
  +
  ,
 - . / <init> ()V Code LineNumberTable LocalVariableTable this Lcom/imodule/dataImport/App; add ()I a I b c main ([Ljava/lang/String;)V args [Ljava/lang/String; app result 
SourceFile App.java   com/imodule/dataImport/App   0 1 2 java/lang/StringBuilder 	result =  3 4 3 5 6 7 8 9 : java/lang/Object java/lang/System out Ljava/io/PrintStream; append -(Ljava/lang/String;)Ljava/lang/StringBuilder; (I)Ljava/lang/StringBuilder; toString ()Ljava/lang/String; java/io/PrintStream println (Ljava/lang/String;)V !  
             /     *??                              c     <=`
h>?               
 	    *               	    
     	       q     '?Y?L+?=??Y??	?
???          
   
  &          '          
       !    "

   我们在windows打开cmd命令窗口,进入App.class文件存在的目录。(我的目录在这里: dataImport\target\classes\com\imodule\dataImport\App.class ,其中开始的 dataImport是我的项目名)

     使用命令 javap -c App ,就会在下方输出指令信息

 

  我们可以用一个文件接收输出内容 ,这里将内内容存储在同级目录的 App.txt下

 

  最后 Apptxt文件中就会出现 程序计数器需要执行的指令啦(我们会发现指令是按大小顺序排列的,但是不是递增的)

  App.txt

Compiled from "App.java"
public class com.imodule.dataImport.App {
  public com.imodule.dataImport.App();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public int add();
    Code:
       0: bipush        11
       2: istore_1
       3: bipush        22
       5: istore_2
       6: iload_1
       7: iload_2
       8: iadd
       9: bipush        10
      11: imul
      12: istore_3
      13: iload_3
      14: ireturn

  public static void main(java.lang.String[]);
    Code:
       0: new           #2                  // class com/imodule/dataImport/App
       3: dup
       4: invokespecial #3                  // Method "<init>":()V
       7: astore_1
       8: aload_1
       9: invokevirtual #4                  // Method add:()I
      12: istore_2
      13: getstatic     #5                  // Field java/lang/System.out:Ljava/io/PrintStream;
      16: new           #6                  // class java/lang/StringBuilder
      19: dup
      20: invokespecial #7                  // Method java/lang/StringBuilder."<init>":()V
      23: ldc           #8                  // String result =
      25: invokevirtual #9                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      28: iload_2
      29: invokevirtual #10                 // Method java/lang/StringBuilder.append:(I)Ljava/lang/StringBuilder;
      32: invokevirtual #11                 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
      35: invokevirtual #12                 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
      38: return
}

 这里我们就来讲一下指令, add()方法里面的指令执行过程:

public int add();
Code:
0: bipush 11   # 将常量 11 压入栈中 (当int取值-1~5采用iconst指令,取值-128~127采用bipush指令,取值-32768~32767采用sipush指令,取值-2147483648~2147483647采用 ldc 指令。)
2: istore_1    # 将数值 11 从操作栈中取出,存储到局部变量 a
3: bipush 22  # 将常量 22 压入栈中
5: istore_2  # 将数值 22 从操作栈中取出,存储到局部变量 b
6: iload_1   # 将局部变量 a 加载到操作栈
7: iload_2   # 将局部变量 b 加载到操作栈
8: iadd  # 执行加运算  局部变量a+b, 结果为33
9: bipush 10 # 将常量10压入栈
11: imul  # 执行乘运算,(a+b)*10,结果为 330
12: istore_3 # 将计算结果 330从栈中取出,存储到局部变量 c
13: iload_3 # 将局部变量 c 加载到操作栈
14: ireturn # 将结果值返回

 

>>Java堆(Heap) 先进先出

  存储对象

  实例变量、new出来的对象、数组等信息

   关于JVM内存的分配图:(其中永久代为方法区)

  堆内存中垃圾回收机制回收过程分析~

  首先对象会存储堆内存的 年轻代的 Eden区,

  进行垃圾回收后 存活的对象会移到 Survivor 去的 From, (From 和 To会一直有一个空间为空,用于来回复制存活对象 大概移动15~16次)

  存活的对象会进入老年代

  

  关于GC:

  虚拟机在进行MinorGC(新生代的GC)的时候,会判断要进入OldGeneration区域对象的大小,是否大于Old Generation剩余空间大小,如果大于就会发生Full GC。

  刚分配对象在Eden中,如果空间不足尝试进行GC,回收空间,如果进行了MinorGC空间依旧不够就放入Old Generation,如果OldGeneration空间还不够就OOM了。

  虚拟机中存在三种垃圾回收现象,minor GC、major GC和full GC。对新生代进行垃圾回收叫做minor GC,对老年代进行垃圾回收叫做major GC,同时对新生代、老年代和永久代进行垃圾回收叫做full GC。

>>Java虚拟机栈(Stack) 后进先出 First in last out

  存储栈帧

  栈帧包括 局部变量表、操作栈、动态链接、方法出口

  方法存储在栈里面,一个方法对应一个栈帧。

  递归方法,就是自己调用自己,会有N个栈帧存储在栈中,且都符合后进先出的规则。 如果是个死循环就会出现栈溢出: Stackoverflow ................

 

>>本地方法栈(Native Stack)

  JNI(Java Native Interface)

  本地方法栈,使用native修饰的方法,是指Java调用非Java代码的接口,方法是由非Java语言实现。

  DLL(Dinamic Link Library): 动态链接库文件

 

>>方法区(Method Area)

  主要存放已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据

  (比如spring 使用IOC或者AOP创建bean时,或者使用cglib,反射的形式动态生成class信息等)。

 

ProcessOn绘图工具地址: https://www.processon.com/diagraming/5d254c70e4b0fdb331d8fa9e

 posted on 2019-07-10 10:39  阿叮339  阅读(225)  评论(0编辑  收藏  举报