1. JVM
1.1 运行时数据区内存模型
运行时数据区是JVM把自己管理的内存部分抽象出来的模型,抽象出来的不同的数据区域,以便于管理,具体有如下几个区域。
1.1.1 分类
程序计数器,堆,栈,本地方法栈,方法区
1.1.2 程序计数器
在线程中,记录每个线程当前执行的语句行数,不会发生OOM,每个线程都有。
1.1.3 栈
栈是描述方法,描述线程的一块区域,每个线程都拥有一个栈,栈中的栈帧表示方法,用FILO的方法表示方法之间的调用。栈帧中有局部变量,局部变量包括,当前方法所在的对象的this指针,定义的局部变量,方法参数,方法内catch的异常中的参数。操作数栈,操作数栈是用来做计算的,因为JVM是一种面向栈的计算语言,普通的计算机有寄存器的概念,用0至15个寄存器做加减,但是JVM用的是栈的方法做计算,比如a+b,在JVM中是将a和b入栈,再使用一个iadd命令,从栈顶弹出a和b做加法,这种就类似于后缀表达式计算方法。栈中还有返回地址和一些其他信息等。
1.1.4 本地方法栈
本地方法栈是在JVM调用本地的native方法的时候会使用的栈,在JDK源代码中有很多方法都是本地的native修饰的方法,这种方法jdk是不实现的,直接调用操作系统的一些源语,JVM只是获取返回值。比如arraycopy方法等。这里也有程序计数器,但是没行号。
1.1.5 堆
堆中用来分配new创建的对象,堆分为新生代和老年代,长时间存在的对象或者较大的对象会进入老年代,刚生成的对象会进入新生代,新生代分为两类区域,一部分为Eden此部分较大占据百分之80的空间,另一部分为Survivor空间,一般会有两个Survivor空间各占百分之10左右。用来存放新生代的对象。
1.1.6 方法区
理论上方法区的位置在堆上,但是方法区和堆存放的数据却天差地别。方法区中存放类的信息,当类加载进来的时候会把字节码的信息存在方法区中,而且类文件里的类变量也存放在方法区中。
从1.8开始,方法区不再作为JVM内存中的一个部分而是将方法区的数据放在本地内存上,作为元空间,这样的好处在于降低了方法区OOM的可能性,因为方法区主要加载的就是类的信息,如果类的信息过大可能导致方法区OOM,而这种OOM通常不是我们操作带来的问题,不是很好处理。
1.1.7 直接内存
在JVM中有一块直接存储数据的区域,在JDK1.4之后,这块区域用给NIO做数据的缓冲,通过一个native的方法来分配出一块区域,通过一个在堆中的对象直接操作者部分内存。避免了在堆和系统内存中不断的复制空间的问题。