Java中对象的创建和销毁过程详析

2025-02-24 17:50

本文主要是介绍Java中对象的创建和销毁过程详析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《Java中对象的创建和销毁过程详析》:本文主要介绍Java中对象的创建和销毁过程,对象的创建过程包括类加载检查、内存分配、初始化零值内存、设置对象头和执行init方法,对象的销毁过程由垃圾回收机...

前言

Java 编程里,对象的创建和销毁是基础且关键的操作,深刻理解这一过程有助于编写出高效、稳定的代码。下面将详细阐述 Java 中对象的创建和销毁过程。

对象的创建过程

1. 类加载检查

当代码中使用 nphpew 关键字创建对象时,Java 虚拟机(JVM)首先会检查该对象对应的类是否已经被加载到内存中。如果尚未加载,JVM 会通过类加载器将该类的字节码文件加载到内存,并对其进行验证、准备和解析等操作,最终完成类的初始化。例如,当执行 Person person = new Person(); 时,JVM 会先确认 Person 类是否已加载。

2. 分配内存

类加载完成后,JVM 会为新对象分配内存空间。分配内存的方式主要有两种:

  • 指针碰撞:假设 Java 堆中的内存是规整的,所有用过的内存放在一边,空闲的内存放在另一边,中间放着一个指针作为分界点的指示器,那所分配内存就仅仅是把那个指针向空闲空间那边挪动一段与对象大小相等的距离,这种分配方式称为“指针碰撞”(Bump the Pointer)。
  • 空闲列表:如果 Java 堆中的内存并不是规整的,已使用的内存和空闲的内存相互交错,那就没有办法简单地进行指针碰撞了,虚拟机就必须维护一个列表,记录上哪些内存块是可用的,在分配的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表上的记录,这种分配方式称为“空闲列表”(Free List)。

选择哪种分配方式由 Java 堆是否规整决定,而 Java 堆是否规整又由所采用的垃圾收集器是否带有空间压缩整理(Compact)的能力决定。

3. 初始化零值

内存分配完成后,JVM 会将分配到的内存空间都初始化为零值(不包括对象头)。这一步操作保证了对象的实例字段在 Java 代码中可以不赋初始值就直接使用,程序能访问到这些字段的数据类型所对应的零值。例如,int 类型的字段初始值为 0,boolean 类型的字段初始值为 false

4. 设置对象头

JVM 会对对象进行必要的设置,例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码、对象的 GC 分代年龄等信息。这些信息存放在对象的对象头(ObCVlFDject Header)之中。

5. 执行 init 方法

在上述工作都完成之后,从 Java 程序的视角看来,对象已经产生了,但从 JVM 的视角来看,对象创建才编程刚刚开始——<init> 方法还没有执行,所有的字段都还为零。所以一般来说(由字节码中是否跟随有 invokespecial 指令所决定),执行 new 指令之后会接着执行 <init> 方法,把对象按照程序员的意愿进行初始化,这样一个真正可用的对象才算完全被构造出来。

对象的销毁过程

1. 可达性分析

在 Java 中,对象的销毁主要由垃圾回收机制(GC)负责。JVM 会通过可达性分析算法来判断对象是否存活。该算法以一系列被称为“GC Roots”的对象为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到 GC Roots 没有任何引用链相连时,则证明此对象是不可达的,就有可能被回收。常见的 GC Roots 包括:

  • 虚拟机栈(栈帧中的本地变量表)中引用的对象。
  • 方法区中类静态属性引用的对象。
  • 方法区中常量引用的对象。
  • 本地方法栈中 JNI(即一般说的 Native 方法)引用的对象。

2. 第一次标记

当对象被判定为不可达时,它会被第一次标记。但此时对象还不会被立即回收,而是会被放入一个名为 F - Queue 的队列中,并由一个低优先级的线程去执行队列中对象的 finalize() 方法(如果对象重写了该方法)。

3. finalize() 方法执行

finalize() 方法是对象逃脱死亡命运的最后一次机会。在该方法中,对象可以重新与引用链上的任何一个对象建立关联,例如把自己(this 关键字)赋值给某个类变量或者对象的成员变量。如果对象在 finalize() 方法中成功拯救了自己,那在第二次标记时它将被移除出“即将回收”的集合;如果对象没有逃脱,那基本上它就真的要被回收了。不过需要注意的是,finalize() 方法的执行javascript是不可靠的,JVM 并不保证该方法一定会被执行。

4. 第二次标记

如果对象在 finalize() 方法执行后仍然没有与 GC Roots 建立引用关系,它会被进行第二次标记。经过第二次标记的对象,就会被真正地列入可回收对象的集合。

5. 垃圾回收

当垃圾回收器执行垃圾回收操作时,会回收那些经过第二次标记的对象所占用的内存空间,将其释放回 Java 堆中,供后续新对象的分配使用。不同的垃圾回收器采用不同的算法和策略来执行垃圾回收,例如标记 - 清除算法、标记 - 整理算法、复制算法等。

知识补充:

在 Java 虚拟机的堆区,每个对象都可能处于以下三种状态之一。

1)可触及状态:当一个对象被创建后,只要程序中还有引用变量引用它,那么它就始终处于可触及状态。

2)可复活状态:当程序不再有任何引用变量引用该对象时,该对象就进入可复活状态。在这个状态下,垃圾回收器会准备释放它所占用的内存,在释放之前,会调用它及其他处于可复活状态的对象的 finalize() 方法,这些 finalize() 方法有可能使该对象重新转到可触及状态。

3)不可触及状态:当 Java 虚拟机执行完所有可复活对象的 finalize() 方法后,如果这些方法都没有使该对象转到可触及状态,垃圾回收器才会真正回收它占用的内存。

总结

到此这篇关于Java中对象的创建和销毁过程的文章就介绍到这了,更多相关Java对象的创建和销毁内容请搜索China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持China编程(www.chinasem.cn)!

这篇关于Java中对象的创建和销毁过程详析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1153524

相关文章

SpringBoot整合easy-es的详细过程

《SpringBoot整合easy-es的详细过程》本文介绍了EasyES,一个基于Elasticsearch的ORM框架,旨在简化开发流程并提高效率,EasyES支持SpringBoot框架,并提供... 目录一、easy-es简介二、实现基于Spring Boot框架的应用程序代码1.添加相关依赖2.添

通俗易懂的Java常见限流算法具体实现

《通俗易懂的Java常见限流算法具体实现》:本文主要介绍Java常见限流算法具体实现的相关资料,包括漏桶算法、令牌桶算法、Nginx限流和Redis+Lua限流的实现原理和具体步骤,并比较了它们的... 目录一、漏桶算法1.漏桶算法的思想和原理2.具体实现二、令牌桶算法1.令牌桶算法流程:2.具体实现2.1

SpringBoot中整合RabbitMQ(测试+部署上线最新完整)的过程

《SpringBoot中整合RabbitMQ(测试+部署上线最新完整)的过程》本文详细介绍了如何在虚拟机和宝塔面板中安装RabbitMQ,并使用Java代码实现消息的发送和接收,通过异步通讯,可以优化... 目录一、RabbitMQ安装二、启动RabbitMQ三、javascript编写Java代码1、引入

spring-boot-starter-thymeleaf加载外部html文件方式

《spring-boot-starter-thymeleaf加载外部html文件方式》本文介绍了在SpringMVC中使用Thymeleaf模板引擎加载外部HTML文件的方法,以及在SpringBoo... 目录1.Thymeleaf介绍2.springboot使用thymeleaf2.1.引入spring

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程