Java知识荟萃(2019年4月23日)
石彬的架构笔记
JVM知识
JVM大致分为:程序计数器、虚拟机栈、堆、方法区、本地方法栈这几个部分.
下面从一个简单的java代码来分析这些区域的作用.
假如有如下代码:
public class HelloWord {
public static void main(String[] args) {
System.out.println("Hello Word");
}
}
上面的.java
代码首先会通过编译器编译为.class
的字节码文件.字节码文件是计算器可以理解的语言,它大概是下面这样的:
1. 程序计数器
在这里,0:aload_o
这样的叫字节码指令,它对应了一条一条的机器指令,计算器只有读到这种机器码指定才知道具体是干什么.
接下来,在执行字节码指令时,JVM里面程序计数器就记录每个线程当前执行的字节码指令的位置,记录当前线程目前执行到了那一条字节码指令.
因为会有多个线程来并发执行各种不同的代码,所以每个线程都有自己的一个程序计数器.
2. Java虚拟机栈
在方法中,经常会定义一些局部变量,比如下面的name:
public void sayHello() {
String name = "hello";
}
在JVM中,有一块保存每个方法内的局部变量数据的区域,这就是java虚拟机栈.而且,每个线程都有自己的java虚拟机栈.
如果线程执行了一个方法,那么就会为这个方法创建对应的一个栈帧,栈帧中包含该方法的局部变量表、操作数栈、动态链接、方法出口等.
在上面的代码中,一个线程调用了sayHello
方法,那么就会为sayHello
方法创建一个栈帧,压入线程自己的java虚拟机栈中,如下图:
如果sayHello
方法调用了另外一个greeting
方法,如下:
这时会给greeting
方法又创建一个栈帧,压入java虚拟机栈里.
接着如果greeting
方法执行完毕了,会把greeting
方法对应的栈帧从虚拟机栈中出栈.
所以,Java虚拟机栈的作用:调用执行任何方法的时候,给方法创建栈帧,然后入栈,执行完毕后出栈.
Java堆内存
Java堆中存放着我们创建的各种对象,如下面的方法:
public void sayHello(String name) {
Student student = new Student(name);
student.study();
}
上面的代码中我们创建了一个student对象,这个对象中包含一个name成员变量.那么这个student就会存放在堆内存中.
Java堆内存区域里放入类似Student的对象,在栈帧的局部变量表中,存放Student对象的地址.如下:
方法区/Metaspace
JDK1.8以前,这个区域叫方法区,1.8以后改名为Metaspace
,主要存放我们自己写的类相关的信息,常量池等.
本地方法栈
在很多JDK的底层API中,会有下面这样的调用:
public native int hashCode();
在调用这种native方法时,就会有线程对应的本地方法栈,类似于java虚拟机栈,不过这里存放的是各种native方法的局部变量表等.