int a = 1,到底存在于JVM的哪里

我是谁,我在哪? QAQ
学习本篇文章需要对JVM内存结构和类加载机制有一定了解~

int作为Java八大基本数据类型之一,大家最熟悉不过了。
如果说我们学习java写的第一行代码是

System.out.println("Hello World");

那么我们学习的第二行就是这个了吧

int a = 1;

那到底a存放于哪里,1又放在哪里了呢?
答案当然不会是唯一的,要是唯一的我还会专门写嘛哈哈!
先说下答案,你答对了么?
1 a作为类的成员变量,存放于方法区中;1保存在堆(Heap)的实例中
2 a作为方法局部变量,存放于Java虚拟机栈(JVM Stacks)的局部变量表中;1也保存在栈内存中。

一 作为类的成员变量
有如下代码

public class IntTest {
    int a = 1;
}

我们通过javap反编译class文件

javap -v -p -l IntTest
Classfile /Users/xxxxx/IdeaProjects/demo/target/classes/com/example/demo/IntTest.class
  Last modified 2020-4-20; size 315 bytes
  MD5 checksum 93d441915a8d7bc6db8efbbd7cff6158
  Compiled from "IntTest.java"
public class com.example.demo.IntTest
  // 4个字节用于存储版本号
  // 前2个字节存储次版本号
  minor version: 0
  // 后2个存储主版本号
  major version: 52
  // ACC_PUBLIC:是否为Public类型 ACC_SUPER:是否允许使用invokespecial字节码指令的新语义
  flags: ACC_PUBLIC, ACC_SUPER
// 用于存放常量的常量池,主要存储了字面量以及符号引用
Constant pool:
   #1 = Methodref          #4.#16         // java/lang/Object."<init>":()V
   #2 = Fieldref           #3.#17         // com/example/demo/IntTest.a:I
   #3 = Class              #18            // com/example/demo/IntTest
   #4 = Class              #19            // java/lang/Object
   #5 = Utf8               a
   #6 = Utf8               I
   #7 = Utf8               <init>
   #8 = Utf8               ()V
   #9 = Utf8               Code
  #10 = Utf8               LineNumberTable
  #11 = Utf8               LocalVariableTable
  #12 = Utf8               this
  #13 = Utf8               Lcom/example/demo/IntTest;
  #14 = Utf8               SourceFile
  #15 = Utf8               IntTest.java
  #16 = NameAndType        #7:#8          // "<init>":()V
  #17 = NameAndType        #5:#6          // a:I
  #18 = Utf8               com/example/demo/IntTest
  #19 = Utf8               java/lang/Object
// 方法表集合
{
  int a;
    descriptor: I
    flags:

  public com.example.demo.IntTest();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=2, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: aload_0
         5: iconst_1
         6: putfield      #2                  // Field a:I
         9: return
      LineNumberTable:
        line 3: 0
        line 4: 4
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      10     0  this   Lcom/example/demo/IntTest;
}
// 源码文件名称
SourceFile: "IntTest.java"

如果我们执行new IntTest();操作,a变量会在构造函数中初始化,iconst_1通过putfield赋予了值。
JVM在加载Class时,创建instanceKlass,表示其元数据,包括常量池、字段、方法等,存放在方法区,所以成员变量a就在方法区咯。
在new一个对象时,JVM创建instanceOopDesc,来表示这个对象,存放在堆区,其引用,存放在栈区;所以1存放在堆中。
ps:不懂可以咕鸽"oop_klass模型",或者移步我的另一篇文章。

二 作为方法局部变量

public class IntTest {
    public void getValue() {
        int a = 1;
    }
}

反编译如下

  public void getValue();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=2, args_size=1
         0: iconst_1
         1: istore_1
         2: return
      LineNumberTable:
        line 5: 0
        line 6: 2
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       3     0  this   Lcom/example/demo/IntTest;
            2       1     1     a   I

当代码执行到 int a = 1;的时候JVM会
iconst_1 int型常量1进栈
istore_1 栈顶int数值存入第2局部变量
所以a作为局部变量会放在局部变量表中,1则是虚拟机在栈空间内开辟了一处地址存放,a指向1,且1在栈中可以被其他局部变量共享,int b = 1; 中b发现栈内存中已经有了1,也指向1

现在你了解了么QAQ

以上均为个人理解,如有不足之处,欢迎在评论区留言。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容