什么是JVM
JVM 全称 Java Virtual Machine,它是一种规范。JVM 是一个虚拟化的操作系统,类似于 Linux 或者 Windows 的操作系统,只是它架在操作系统上,接收字节码,把字节码翻译成操作系统上的机器码且进行执行。
为什么需要学习JVM
Android的运行环境都是在ART虚拟机或者是Dalvik虚拟机上,为什么Android程序员需要学习JVM,理解JVM可以帮助我们更好的了解Java内存区域、对象的创建和内存分配、垃圾的回收以及常见的垃圾回收算法等等,然后将其运用到Android开发中,有助于处理app中的内存问题。
JVM的内存区域
在JVM中内存主要分为堆、程序计数器、方法区、虚拟机栈和本地方法栈等
同时按照与线程的关系也可以这么划分区域:
- 线程私有区域:一个线程拥有单独的一份内存区域。
- 线程共享区域:被所有线程共享,并且只有一份。
虚拟机栈
虚拟机栈是先进后出(FILO)的数据结构,是每个线程私有的,线程在运行时,在执行每个方法的时候都会打包成一个栈帧,存储了局部变量表
,操作数栈
,动态链接
,方法出口
等信息,然后放入栈。每个时刻正在执行的当前方法就是虚拟机栈顶的栈桢。方法的执行就对应着栈帧在虚拟机栈中入栈和出栈的过程。
栈帧
在每个 Java 方法被调用的时候,都会创建一个栈帧,并入栈。一旦方法完成相应的调用,则出栈。
栈帧一般包含四个区域:(局部变量表、操作数栈、动态连接、返回地址)
-
局部变量表:
用于存放我们的局部变量的(方法中的变量),主要存放我们的 Java 的八大基础数据类型,如果是局部的一些对象,只需要存放它的一个引用地址即可。
-
操作数栈
存放 java 方法执行的操作数的,它就是一个栈,先进后出的栈结构,操作数栈,就是用来操作的,操作的元素可以是任意的 java 数据类型。
-
动态连接
每个栈帧中都包含一个在常量池中对当前方法的引用, 目的是支持方法调用过程的动态连接。
-
返回地址
正常退出,即正常执行到任何方法的返回字节码指令,如 RETURN、IRETURN、ARETURN 等、
异常退出
我们来写一个简单的java代码查看汇编指令
public class Stack {
public static void main(String[] args) {
int a = 10;
int b = 20;
int c = a + b;
System.out.println(c);
}
}
执行 javac
生成class,再执行 javap -v Stack.class
Classfile /C:/Users/Jack/Desktop/Stack.class
Last modified 2021-11-24; size 406 bytes
MD5 checksum 38b185e462a0e2aa0a06afdc67dfe12c
Compiled from "Stack.java"
public class com.test.Stack
minor version: 0
major version: 52
flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
#1 = Methodref #5.#14 // java/lang/Object."<init>":()V
#2 = Fieldref #15.#16 // java/lang/System.out:Ljava/io/PrintStream;
#3 = Methodref #17.#18 // java/io/PrintStream.println:(I)V
#4 = Class #19 // com/Stack
#5 = Class #20 // java/lang/Object
#6 = Utf8 <init>
#7 = Utf8 ()V
#8 = Utf8 Code
#9 = Utf8 LineNumberTable
#10 = Utf8 main