JVM的内存结构之线程私有部分与线程共享部分

在这里插入图片描述
Java内存模型即途中的Runtime Data Area区域
其中,一部分是线程私有的部分,另外一部分是线程共享的部分。

Java内存模型之线程私有部分

私有部分为:程序计数器、虚拟机栈、本地方法栈

  1. 程序计数器
  • 代表当前线程所执行的字节码行号指示器(是逻辑计数器而不是物理计数器)
  • 改变计数器的值来选取下一条需要执行的指令
  • 和线程是一对一的关系,即“线程私有”
  • 对Java方法计数,如果是native方法则计数器的值为Undefined
  • 不会发生内存泄漏
  1. 虚拟机栈(Stack)
  • Java方法执行的内存模型
  • 有固定容量
  • 由虚拟机自动管理的
  • 包含多个栈帧,当方法调用结束时,帧才会被销毁

在这里插入图片描述
其中,局部变量表包含方法执行过程中的所有变量,操作数栈:入栈,出栈,复制,交换,产生消费变量
编写下面这个类,先对其用javac进行编译再用javap进行反编译

package com.interview.javabasic.jvm.model;
public class ByteCodeSample {
    public static int add (int a, int b) {
        int c = 0;
        c = a + b;
        return c;
    }
}

在这里插入图片描述
反编译的内容:

D:\ProgrammingLife\Java\JavaBasic\src>javap -verbose com/interview/javabasic/jvm/model/ByteCodeSample.class
Classfile /D:/ProgrammingLife/Java/JavaBasic/src/com/interview/javabasic/jvm/model/ByteCodeSample.class
  Last modified 2020-2-26; size 304 bytes
  MD5 checksum 63484f98f7e61cd31624692497744b3c
  Compiled from "ByteCodeSample.java"
public class com.interview.javabasic.jvm.model.ByteCodeSample
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #3.#12         // java/lang/Object."<init>":()V
   #2 = Class              #13            // com/interview/javabasic/jvm/model/ByteCodeSample
   #3 = Class              #14            // java/lang/Object
   #4 = Utf8               <init>
   #5 = Utf8               ()V
   #6 = Utf8               Code
   #7 = Utf8               LineNumberTable
   #8 = Utf8               add
   #9 = Utf8               (II)I
  #10 = Utf8               SourceFile
  #11 = Utf8               ByteCodeSample.java
  #12 = NameAndType        #4:#5          // "<init>":()V
  #13 = Utf8               com/interview/javabasic/jvm/model/ByteCodeSample
  #14 = Utf8               java/lang/Object
{
  public com.interview.javabasic.jvm.model.ByteCodeSample();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 3: 0

  public static int add(int, int);
    descriptor: (II)I
    flags: ACC_PUBLIC, ACC_STATIC
    Code:
      stack=2, locals=3, args_size=2
         0: iconst_0
         1: istore_2
         2: iload_0
         3: iload_1
         4: iadd
         5: istore_2
         6: iload_2
         7: ireturn
      LineNumberTable:
        line 5: 0
        line 6: 2
        line 7: 6
}
SourceFile: "ByteCodeSample.java"

在这里插入图片描述
额外补充:递归为什么会引发java.lang.StackOverflowError异常?

递归过于深,由于每次线程执行一个方法,都会创建对应的栈帧,并压入虚拟机栈中,当栈帧数超出虚拟栈深度时就会发生异常

  1. 本地方法栈
  • 与虚拟机栈类似,主要作用于标注了native的方法

Java内存模型之线程共享部分

共享部分为:MetaSpace(元空间),Java堆

  1. 元空间(MetaSpace)与永久代(PermGen)的区别:(用元空间代替永久代)
  • 元空间使用的是本地内存,而永久代使用的是jvm的内存
  • 字符串常量池存在永久代中,容易出现性能问题和内存溢出
  • 类和方法的信息大小难以确定,给永久代的大小指定带来困难
  • 永久代会为GC带来不必要的复杂性
  1. Java堆(Heap),在虚拟机启动时创建
  • 对象实例的分配区域(Java堆的唯一目的)
  • GC管理的主要区域
发布了22 篇原创文章 · 获赞 23 · 访问量 6281

猜你喜欢

转载自blog.csdn.net/bob_man/article/details/104514961