解析java创建对象的全过程

 更新时间:2025年12月16日 09:01:44   作者:IT云烟.  
文章总结了Java创建对象的几种方式,包括使用new关键字、反射、clone和反序列化,此外,还详细描述了创建对象的过程,包括类加载、内存分配和对象初始化,最后,强调了并发情况下的操作原子性问题

一、java创建对象的几种方式

1.1、使用new关键字

调用类的构造方法创建对象

1.2、反射创建对象

1.2.1、Class.newInstance创建对象

1.2.2、调用构造器再去创建对象Constructor.newInstance

先通过反射获取类中无参构造器,然后通过newInstance()获取对象

1.3、clone实现

通过Clone创建对象,首先实体类中必须先实现Cloneable接口并复写Object的clone方法(因为Object的这个方法是protected的)

1.4、反序列化

序列化:指把 Java 对象转换为字节序列的过程;

反序列化:指把字节序列恢复为 Java 对象的过程;

此方式需要类先实现Serializable接口

public class TestStack {
    public static void main(String[] args) throws Exception {
        File file =new File("M:/Serializable.txt");
        FileOutputStream fileOutputStream = new FileOutputStream(file);
        ObjectOutputStream outputStream = new ObjectOutputStream(fileOutputStream);
        UserParam userParam =new UserParam("hello");
        outputStream.writeObject(userParam);
        FileInputStream fileInputStream  = new FileInputStream(file);
        ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);
        UserParam userParam1 = (UserParam)objectInputStream.readObject();
        userParam1.setNickName("world");
        System.out.println(userParam1);
    }
}

二、创建对象的过程

当Java虚拟机遇到一条字节码new指令时:

1、检查类是否已经被加载

去常量池中查找该引用所指向的类有没有被虚拟机加载,如果没有被加载,那么会进行类的加载过程。类的加载过程需要经历:加载、链接、初始化三个阶段。对象的大小,在类加载完成时确定。(jdk1.8中,运行时常量池、类常量池存在于方法区中。)

2、 为对象分配内存空间

JVM为对象分配空间,即把一块确定大小的内存块从Java堆中划分出来。

2.1、分配空间的方式

1、指针碰撞

假设Java堆中内存是绝对规整的,所有被使用过的内存都被放在一边,空闲的内存被放在另一边,中间放着一个指针作为分界点的指示器,那所分配内存就仅仅是把那个指针向空闲空间方向挪动一段与对象大小相等的距离。

①正常情况

②给对象分配内存后

这种方式的优点是工作简单,效率高,只需要移动指针就可以分配内存空间。

缺点也很明显:由于用指针碰撞分配内存空间分为两步:

  • 1、读取指针当前的位置。
  • 2、根据自身大小移动指针,不是原子操作,对象创建在虚拟机中是非常频繁的操作,在并发情况下,会导致执行读操作或执行写操作的结果与预设的结果不一致(指针划分不一致)。

例如:线程A要给对象分配8kb,读取到指针当前的位置,时间片用完,切换到线程B,线程B要给它的对象分配16kb,也读取到指针当前的位置(和线程A读取到的一样),将指针向空闲内存方向移动16kb大小,线程B时间片用完,切换到线程A继续执行,由于线程A使用的指针位置还是之前读到的。(线程不安全问题)

③针对指针碰撞线程不安全,有两种方案:

1、同步处理(加锁)分配内存空间行为

采用 CAS 分配重试的方式来保证更新操作的原子性

2、把内存分配行为按照线程,划分在不同的内存空间进行

  • 即每个线程在Java堆中预先分配一小块内存,称为本地线程分配缓冲(Thread Local Allocation Buffer,TLAB),哪个线程要分配内存,就在哪个线程的本地缓冲区中分配,只有本地缓冲区用完了,分配新的缓存区时才需要同步锁定
  • 虚拟机是否使用TLAB,可以通过-XX:+/-UseTLAB参数来设定。

2、空闲列表

如果Java堆中的内存并不是规整的,已被使用的内存和空闲的内存相互交错在一起,那就没有办法简单地进行指针碰撞了,虚拟机就必须维护一个列表,记录上哪些内存块是可用的,在分配的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表上的记录。

3、怎么选择分配方式

两种方式的选择由 Java 堆是否规整决定,Java 堆是否规整是由选择的垃圾收集器是否具有压缩整理能力决定的。

①将内存空间初始化为零值

内存分配完成之后,虚拟机必须将分配到的内存空间(但不包括对象头)都初始化为零值。零值初始化意思就是对对象的字段赋0值,或者null值,这也就解释了为什么这些字段在不需要进程初始化时候就能直接使用。

如果使用了TLAB的话,这一项工作也可以提前至TLAB分配时顺便进行。

②对对象进行必要的设置

例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码、对象的GC分代年龄等信息。这些信息存放在对象的对象头中。

从虚拟机的视角来看,一个新的对象已经产生了。但是从Java程序的视角看来,对象创建才刚刚开始——构造函数,即Class文件中的()方法还没有执行,所有的字段都为默认的零值,对象需要的其他资源和状态信息也还没有按照预定的意图构造好。

③执行实例的初始化方法init

init方法包含成员变量、构造代码块的初始化,按照声明的顺序执行,执行对象的构造方法,并把堆内对象的首地址赋值给引用变量。至此,对象创建成功。

三、注意事项

并发情况下,需要考虑操作的步骤是不是原子性,如果不是,就要加锁

原子性就是动作不能再继续被拆分了,读是原子性,写也是原子性,但是读加上写就不是原子性。

四、总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 浅谈Maven镜像更换为阿里云中央仓库(精)

    浅谈Maven镜像更换为阿里云中央仓库(精)

    本篇文章主要介绍了Maven镜像更换为阿里云中央仓库(精),小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-12-12
  • 基于mybatis中test条件中单引号双引号的问题

    基于mybatis中test条件中单引号双引号的问题

    这篇文章主要介绍了基于mybatis中test条件中单引号双引号的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-01-01
  • Java根据日期计算星期几的四种方法

    Java根据日期计算星期几的四种方法

    在我们日常业务代码中,经常要用到星期几,下面这篇文章主要给大家介绍了关于Java根据日期计算星期几的四种方法,文中通过代码将每种实现的非常详细,需要的朋友可以参考下
    2023-09-09
  • 详解SpringBoot 处理异常的几种常见姿势

    详解SpringBoot 处理异常的几种常见姿势

    这篇文章主要介绍了详解SpringBoot 处理异常的几种常见姿势,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-08-08
  • SpringCache框架加载/拦截原理详解

    SpringCache框架加载/拦截原理详解

    这篇文章主要介绍了SpringCache框架加载/拦截原理详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-04-04
  • Docker与Kubernetes部署Java应用容器化实践指南

    Docker与Kubernetes部署Java应用容器化实践指南

    容器化是一种将应用程序及其依赖打包成一个可移植的单元的方法,容器能够在任何环境中运行,并且具有隔离性、可扩展性和高效性,这篇文章主要介绍了Docker与Kubernetes部署Java应用容器化实践指南的相关资料
    2026-05-05
  • Java如何获取Object中Value

    Java如何获取Object中Value

    在Java中,获取Object中的值需依赖于对象的类型和属性,常用方法包括使用反射、getter方法、接口或抽象类、Map等数据结构,本文给大家介绍Java如何获取Object中Value,感兴趣的朋友跟随小编一起看看吧
    2024-09-09
  • springboot使用redis的详细步骤

    springboot使用redis的详细步骤

    SpringBoot对常用的数据库支持外,对NoSQL 数据库也进行了封装自动化,下面这篇文章主要给大家介绍了关于springboot使用redis的详细步骤,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-06-06
  • jcl与jul log4j1 log4j2 logback日志系统机制及集成原理

    jcl与jul log4j1 log4j2 logback日志系统机制及集成原理

    这篇文章主要介绍了jcl与jul log4j1 log4j2 logback的集成原理,Apache Commons-logging 通用日志框架与日志系统的机制,有需要的朋友可以借鉴参考下
    2022-03-03
  • Spring构造器注入及@Autowired、lombok的@RequiredArgsConstructor的异同点说明

    Spring构造器注入及@Autowired、lombok的@RequiredArgsConstructor的异同点说明

    本文介绍了Spring框架中构造器注入的实现原理,包括Spring如何找到参数、构造器注入与Autowired的区别,以及为什么Spring官方推荐构造器注入,文章还强调了构造器注入中使用final关键字的好处,并提供了代码示例,帮助读者更好地理解和应用Spring的最佳实践
    2026-03-03

最新评论