初步了解JVM第二篇

2019-12-25  本文已影响0人  java星星

点关注,不迷路;持续更新Java相关技术及资讯!!!

在一篇《初步了解JVM第一篇》中,我们已经了解了:

那在这一篇中我们来聊一聊方法区、栈和堆。

5.方法区

在JVM的架构图中,Java栈、本地方法栈、程序计数器都是线程私有的。而方法区跟堆一样,是一个内存共享的区域,他的主要作用就是存储每一个类的结构信息,例如运行时常量池(Runtime Constant Pool)、字段和方法数据、构造函数和普通方法的字节码内容。

再简单来说方法区就是一个类的模板,在上一篇我们已经说了ClassLoader将class文件加载完成之后会把类的字节码内容放到方法区中,就像把Car.class文件通过类加载器加载后,会把car这个类的结构信息存放在方法区中。当你要实例化的时候再通过这个模板去new出你想要的car1,car2,car2,而你创建出来这些类对象是存放在堆(heap)中的。

图一是方法区中存放的内容

image

方法区的实现:

方法区只是一个定义、一个规范。在不同的虚拟机里头实现是不一样的。这里我们主要介绍的是JDK7和JDK8的实现方式

永久代

在JDK7中方法区的实现方式叫永久代,但是它存储的部分数据是存放在JVM的一块地方的,这会造成一个问题:

当类加载太多了,可能会导致内存栈溢出:java.lang.OutOfMemoryError: PermGen这样一来就不够灵活,为了提高灵活性(这只是其中一个原因)就有了元空间

元空间:

在JDK8中,JVM的开发者就把永久代移除了,移至元空间中。其实作用是差不多的,只是元空间不再使用JVM的内存了,而是直接使用本地堆内存(native heap),说白了就是直接使用系统的内存,这样就几乎不会发生内存溢出的情况,提高了灵活性。

所以为什么在网上会看到关于方法区很多不同的说法就是因为方法区的实现方式在不同的JVM中是不同,最典型的就是永久代和元空间。

以上我们总结出:

以上就是方法区的介绍,在介绍堆的时候还会提及。

6.Stack栈

栈是一个线程私有的,主要用来管理Java程序的运行。是在线程创建的时候创建的,它的生命周期跟随这线程的结束而结束,当线程结束了栈的内存也就释放了,对于栈来说,不会存在垃圾回收问题,因为只要线程一结束该栈就结束了。

栈中主要存储的内容:

栈就类似一个子弹夹,它的特点就是“后进先出,先进后出”,在Java中需要实现很多方法,而这些方法就是一个一个被压进栈中的,然后再依次调用。在平常中,我们所说的Java中的方法在栈其实有一个专有名词叫栈帧,栈帧主要存放三类数据:

栈运行原理:

Java中的方法存放在栈中,但是这些方法到底是怎么执行的呢?

接下来我们就用一个例子来说明一下:

<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: &quot;Courier New&quot; !important; font-size: 12px !important;">package testJVM;

public class TestStack {
    public  static void method_one(){
        System.out.println("This is the method_one");
    }
    public static void method_two(){
        System.out.println("This is the method_two");
    }
    public static void main(String[] args) {
        System.out.println("This is the main method");
        //调用方法一
 method_one();
        //调用方法二
 method_two();
        //输出程序结束
        System.out.println("The program is finish");
    }
}

以上的运行结果为:

image

这样的输出结果,相信已经在大家的预料之中,但是这些方法在栈中是怎么运行的呢?废话不说,上图二

image

我们都知道main方法是一切程序的入口,所以程序一执行碰到的是main方法,main方法就第一个入栈了,所以他们的执行过程是这样的:

所以就形成了图二,当他运行的时候:

这样就形成了类似一条执行链,依次执行了main方法。

总结栈运行原理:

栈中的数据都是以栈帧(Stack Frame)的格式存在,栈帧是一个内存区块,是一个数据集,是一个有关方法(Method)和运行期数据的数据集,当一个方法A被调用时就产生了一个栈帧 F1,并被压入到栈中, A方法又调用了 B方法,于是产生栈帧 F2 也被压入栈, B方法又调用了 C方法,于是产生栈帧 F3 也被压入栈, 执行完毕后,先弹出F3栈帧,再弹出F2栈帧,再弹出F1栈帧…… 遵循“先进后出”和“后进先出”原则。每个方法执行的同时都会创建一个栈帧,用于存储局部变量表、操作数栈、动态链接、方法出口等信息,每一个方法从调用直至执行完毕的过程,就对应着一个栈帧在虚拟机中入栈到出栈的过程。栈的大小和具体JVM的实现有关,通常在256K~756K之间,与等于1Mb左右。

栈溢出

讲完了栈的内容,现在我们来看一个大家在实际开发中会碰到的一个错误,请看下列代码:

<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: &quot;Courier New&quot; !important; font-size: 12px !important;">package testJVM;

public class TestStack {
    public  static void method_one(){
        //递归调用
 method_one();
    }
    public static void main(String[] args) {
        method_one();
    }
}

上述是一个递归调用的例子,现在来执行一下,看看会出现一个什么结果:

image

相信大家多多少少都会遇到过上述的错误,栈溢出。原因如下:

由于我们的方法method_one一直在递归调用自己,而且并没有停止的条件。所以method_one这个方法就会被一直压入栈中,JVM中的内存又是有限的,上述我们也提到了Java中的栈是随着线程的生命周期结束而结束的,不会存在垃圾回收机制,内存得不到释放而方法又不断的进栈,最终内存不够造成栈溢出的现象。图三

image

以上就是本人对栈的理解,最后来到了重头戏堆(heap),那就下篇再进行介绍吧,哈哈哈。

在下篇将会介绍:

上一篇 下一篇

猜你喜欢

热点阅读