细说jvm(二)、java对象创建过程

简介: 细说jvm(二)、java对象创建过程

上篇大体说了jvm运行时候的数据区域,这篇我们来说说对象创建的过程。注意哈,这篇文章应该是对你写高性能的代码是有帮助的。

对象真的是振奋单身狗们心灵的一个词,在面向对象编程里面就更爽了,想new多少就new多少,想new什么样的就new什么样的。

好了,说正经的,对象创建是我们编程中做的非常频繁的一件事情,那么对象创建都经历了什么过程呢?我们这篇里面仅仅说遇到new指令时候所进行的操作。

我们以下面这段代码为例


// 这个类仅仅有个main方法,main方法仅仅new了个对象而已
public class TestNew {
    public static void main(String[] args) {
        new MyEntity(1,"a");
    }
}


MyEntity.java


@Setter
@Getter
// 仅仅是两个属性以及空参和无参构造
public class MyEntity {
    private Integer id;
    private String name;
    public MyEntity(Integer id) {
        this.id = id;
    }
    public MyEntity(Integer id, String name) {
        this.id = id;
        this.name = name;
    }
}


我们不多BB,直接javap -v TestNew.class,来看main方法字节码(别害怕,我一句一句解释给你)

如下是main方法对应字节码


1686814320545.png


为了方便讲解,我把它粘了下来:


// 注意哈,你需要重点关注的只有我注释的5-12的地方,其他的跟着看一遍就行
// 注意哈,你需要重点关注的只有我注释的5-12的地方,其他的跟着看一遍就行
// 注意哈,你需要重点关注的只有我注释的5-12的地方,其他的跟着看一遍就行   重要的事情说三遍
public static void main(java.lang.String[]);
    // 1. 这个descriptor 描述的是方法的参数是string类型的数组,最后这个大写的V表示是个void的返回值类型
    descriptor: ([Ljava/lang/String;)V
    // 2. 方法描述符,下面的两个值说明是个public 的 static 方法
    flags: ACC_PUBLIC, ACC_STATIC
    // 3. Code表示是方法的代码部分
    Code:
    // 4.栈桢中的操作数栈的深度是4,局部变量个数是1,方法参数的个数是1
      stack=4, locals=1, args_size=1
      // 5.遇到了一条new指令,然后创建了一个空的对象,执行默认的初始化逻辑(这里没有为成员属性赋值)
         0: new           #2                  // class com/example/demo/asmtest/MyEntity
         // 6. 把对象的引用压到操作数栈的栈顶
         3: dup
         // 7. 把int类型值为1的这个数压到操作数栈栈顶
         4: iconst_1
         // 8. 给1执行了一下Integer.valueof方法,这个因为id类型是Integer,但你给的1实际是int
         5: invokestatic  #3                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
         // 9. 把字符串”a“从常量池压到栈顶,ldc这个指令是把int,float,string类型的值从常量池加载到栈顶
         8: ldc           #4                  // String a
         // 10. 调用MyEntity的构造方法。为什么是这时候调用?因为需要的两个变量这时候已经准备好了
        10: invokespecial #5                  // Method com/example/demo/asmtest/MyEntity."<init>":(Ljava/lang/Integer;Ljava/lang/String;)V
         // 11. 上一步创建出来的对象由于没有用变量去接收,所以直接用pop指令弹出
        13: pop
        // 12. 方法结束返回到调用它的地方
        14: return
        // 13. LineNumberTable是方法的行号表,这是调试用的
      LineNumberTable:
        line 8: 0
        line 9: 14
        // 14。 LocalVariableTable是局部变量表,也是属于调试信息
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      15     0  args   [Ljava/lang/String;
        // 15. MethodParameters是方法参数表
    MethodParameters:
      Name                           Flags
      args


我们一点一点的来说,第5句这里其实做了很多事情,大概如下:

  1. 首先检查这个new指令的参数是否能在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已经被加载、验证、准备、解析、和初始化过。如果没有那就表明该类还没有被虚拟机加载无法创建,需要先执行类的加载过程(这里这个类其实就是MyEntity)。
  2. 类加载检查之后,jvm在java堆中按照 “指针碰撞” 或者 “空闲列表” 的方式为新生的对象分配内存,分配内存的大小在类加载完成后便可完全确定(由于我们使用的垃圾回收器是CMS,所以是按照空闲列表的方式来的)。
  3. 内存分配完成之后,jvm需要将分配到的内存空间进行初始化零值(不包含对象头),该步骤保证了对象的实例字段在Java代码中可以不赋初始值就可以直接使用,程序能访问到这些字段的数据类型所对应的零值。(实例变量无需初始化即可被程序使用,区别于局部变量的必须初始化才能被程序使用)
  4. 接下来对对象进行必要的设置:这个对象是哪个类的实例、如何找到类的元数据信息、对象的哈希码、对象的GC分代年龄等信息、这些信息存放于对象的对象头信息中。

到这里new指令做的事情就完了,接下来要填充对象实例了,也就是该给成员属性赋值了,但是成员属性的值也需要一个准备过程,你肯定觉得很奇怪,为什么还需要准备呢?我不是已经在new MyEntity(1,“a”)中把值给了吗?这是因为你给的1其实是int类型,但是id这个成员属性其实是Integer类型,因此需要装箱,因此就有了8这里这个,然后“a”是在字符串常量池的,需要从常量池中把引用加载到栈顶来,也就是9这里的操作,这里都准备好了,才在10调用了MyEntity类的构造方法初始化成员变量。10这里做的事情可以概括如下:

5.开始执行方法进行对象的初始化,按照程序猿的意愿初始化对象,到这里一个成品的对象就算是加载完了

我上面说的12345就是对象创建的完整过程,细节你可以对着字节码的看看,下一篇说对象创建过程的内存分配。

有什么问题不明白也可以在评论区问我,我并非没耐心的人,只是比较忙,你问的问题我看到的话一定会认真给你解释。

目录
相关文章
|
2天前
|
存储 Java
java的对象详解
在Java中,对象是根据类模板实例化的内存实体,具有唯一标识符、属性及行为。通过`new`关键字实例化对象并用构造方法初始化。变量存储的是对象引用而非对象本身,属性描述对象状态,方法定义其行为。Java利用垃圾回收机制自动处理不再使用的对象内存回收,极大地简化了对象生命周期管理,同时对象具备封装、继承和多态性,促进了代码的重用与模块化设计。这使得Java程序更易于理解、维护和扩展。
|
4天前
|
Java
Java 对象和类
在Java中,**类**(Class)和**对象**(Object)是面向对象编程的基础。类是创建对象的模板,定义了属性和方法;对象是类的实例,通过`new`关键字创建,具有类定义的属性和行为。例如,`Animal`类定义了`name`和`age`属性及`eat()`、`sleep()`方法;通过`new Animal()`创建的`myAnimal`对象即可调用这些方法。面向对象编程通过类和对象模拟现实世界的实体及其关系,实现问题的结构化解决。
|
8天前
|
存储 Java 程序员
优化Java多线程应用:是创建Thread对象直接调用start()方法?还是用个变量调用?
这篇文章探讨了Java中两种创建和启动线程的方法,并分析了它们的区别。作者建议直接调用 `Thread` 对象的 `start()` 方法,而非保持强引用,以避免内存泄漏、简化线程生命周期管理,并减少不必要的线程控制。文章详细解释了这种方法在使用 `ThreadLocal` 时的优势,并提供了代码示例。作者洛小豆,文章来源于稀土掘金。
|
9天前
|
安全 前端开发 Java
浅析JVM invokedynamic指令与Java Lambda语法的深度融合
在Java的演进历程中,Lambda表达式无疑是Java 8引入的一项革命性特性,它极大地简化了函数式编程在Java中的应用,使得代码更加简洁、易于阅读和维护。而这一切的背后,JVM的invokedynamic指令功不可没。本文将深入探讨invokedynamic指令的工作原理及其与Java Lambda语法的紧密联系,带您领略这一技术背后的奥秘。
9 1
|
2天前
|
Java 程序员
Java编程中的对象和类: 初学者指南
【9月更文挑战第9天】在Java的世界中,对象和类构成了编程的基石。本文将引导你理解这两个概念的本质,并展示如何通过它们来构建你的程序。我们将一起探索类的定义,对象的创建,以及它们如何互动。准备好了吗?让我们开始这段Java的旅程吧!
|
9天前
|
存储 Java
Java编程中的对象序列化与反序列化
【9月更文挑战第2天】在Java的世界里,对象序列化和反序列化就像是给数据穿上了一件隐形的斗篷。它们让数据能够轻松地穿梭于不同的系统之间,无论是跨越网络还是存储在磁盘上。本文将揭开这层神秘的面纱,带你领略序列化和反序列化的魔法,并展示如何通过代码示例来施展这一魔法。
11 0
|
11天前
|
C# 开发者 Windows
震撼发布:全面解析WPF中的打印功能——从基础设置到高级定制,带你一步步实现直接打印文档的完整流程,让你的WPF应用程序瞬间升级,掌握这一技能,轻松应对各种打印需求,彻底告别打印难题!
【8月更文挑战第31天】打印功能在许多WPF应用中不可或缺,尤其在需要生成纸质文档时。WPF提供了强大的打印支持,通过`PrintDialog`等类简化了打印集成。本文将详细介绍如何在WPF应用中实现直接打印文档的功能,并通过具体示例代码展示其实现过程。
46 0
|
11天前
|
数据库 C# 开发者
WPF开发者必读:揭秘ADO.NET与Entity Framework数据库交互秘籍,轻松实现企业级应用!
【8月更文挑战第31天】在现代软件开发中,WPF 与数据库的交互对于构建企业级应用至关重要。本文介绍了如何利用 ADO.NET 和 Entity Framework 在 WPF 应用中访问和操作数据库。ADO.NET 是 .NET Framework 中用于访问各类数据库(如 SQL Server、MySQL 等)的类库;Entity Framework 则是一种 ORM 框架,支持面向对象的数据操作。文章通过示例展示了如何在 WPF 应用中集成这两种技术,提高开发效率。
25 0
|
11天前
|
存储 Java
Java编程中的对象和类
在Java的世界中,“对象”与“类”是构建一切的基础。就像乐高积木一样,类定义了形状和结构,而对象则是根据这些设计拼装出来的具体作品。本篇文章【8月更文挑战第31天】 将通过一个简单的例子,展示如何从零开始创建一个类,并利用它来制作我们的第一个Java对象。准备好让你的编程之旅起飞了吗?让我们一起来探索这个神奇的过程!
|
24天前
|
Java Docker 索引
记录一次索引未建立、继而引发一系列的问题、包含索引创建失败、虚拟机中JVM虚拟机内存满的情况
这篇文章记录了作者在分布式微服务项目中遇到的一系列问题,起因是商品服务检索接口测试失败,原因是Elasticsearch索引未找到。文章详细描述了解决过程中遇到的几个关键问题:分词器的安装、Elasticsearch内存溢出的处理,以及最终成功创建`gulimall_product`索引的步骤。作者还分享了使用Postman测试接口的经历,并强调了问题解决过程中遇到的挑战和所花费的时间。