JAVA实例化对象的五种方式

面向对象的编程语言JAVA,万物皆为对象,工作中我们常常会将创建(实例化)对象的操作交给框架或者容器(如SpringIoC),但依旧需要了解JAVA实例化对象的多种方式

所属分类 JAVA

相关标签 对象实例化构造工厂序列化

前言

最为JAVA开发者,无时无刻不在与JAVA对象打交道。
我们在日常工作中常常将创建(实例化)对象的任务交给依赖框架,如Spring。
但依旧需要了解JAVA实例化对象的比较常见的一些形式,他们分别是:

  1. 使用new关键字(构造函数会被调用)
  2. 使用Class类的newInstance()(构造函数会被调用)
  3. 使用Constructor类的newInstance()(构造函数会被调用)
  4. 使用clone方法(无构造函数调用)
  5. 使用deserialization(无构造函数调用)

相关实例

最通用、常见、普遍、简单的实例化方式,默认调用无参构造,支持调用有参构造。

/**无参构造创建对象*/
User jack= new User();
/**有参构造创建对象 有参构造的入参与自定义的构造函数一致*/
User tom = new User("mebugs","man");

使用反射的Class类的newInstance()来实例化对象,newInstance()方法将调用无参构造方法去实例化一个对象。

//方法将调用类的无参构造
User classIns = User.class.newInstance();

使用反射的Constructor类的newInstance()方法,与Class类中的newInstance()方法相似。

Constructor类的newInstance()方法我们能够调用有参构造函数和私有构造函数。

//默认调用形式
User consIns = User.class.getConstructor().newInstance();
//调用有参构造
User consInsTom = User.class.getConstructor().newInstance("mebugs","man");
//调用私有无参构造
User consInsPrive = User.class.getDeclaredConstructor().newInstance();

调用对象的clone()方法时,JVM都会为我们创建一个新对象,并将前一个对象的所有内容复制到其中。

使用clone()方法创建对象不会调用任何构造函数,但对象的类需要实现Cloneable接口并定义clone()方法。

clone()方法是比较有争议的话题,确实有它的缺点,但其是创建任何对象副本的最流行和最简单的方法。

//默认调用形式
User jack = tom.clone();

使用反序列化创建对象的前提是已经进行序列化操作。

当进行序列化和反序列化对象时,JVM会为我们创建了一个独立的对象。

在反序列化过程中,JVM不使用任何构造函数来创建对象,需要在类中实现Serializable接口。

//序列化
User tom = new User("mebugs","man");
ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("com.mebugs"));
out.writeObject(tom);
out.close();
//反序列化
ObjectInputStream in = new ObjectInputStream(new FileInputStream("com.mebugs"));
User tomBack = (User) in.readObject();
in.close();

其他说明

Class类的newInstance()方法和Constructor类的newInstance()方法都被称为创建对象的反射方法。

实际上,Class类的newInstance()方法内部使用Constructor类的newInstance()方法。

这就是为什么后者更受欢迎,并且也被Spring、Hibernate等不同的经典框架所使用的原因。

上文中我们提到getConstructor()和getDeclaredConstructor()来获取Constructor类。

  • getDeclaredConstructor()返回指定参数类型、所有声明的(包括private)构造函数
  • getConstructor()返回指定参数类型、具有public访问权限的构造函数

因此getConstructor()是getDeclaredConstructor()的子集。

米虫

做一个有理想的米虫,伪全栈程序猿,乐观主义者,坚信一切都是最好的安排!

本站由个人原创、收集或整理,如涉及侵权请联系删除

本站内容支持转发,希望贵方携带转载信息和原文链接

本站具有时效性,不提供有效、可用和准确等相关保证

本站不提供免费技术支持,暂不推荐您使用案例商业化

发表观点

提示

昵称

邮箱

QQ

网址

当前还没有观点发布,欢迎您留下足迹!

同类其他

JAVA

listener、filter、servlet的加载次序

在web.xml中经常会看到listener,filter,servlet的相关标签配置,它们分别是监听器、过滤器、容器,都是在项目启动的时候就可以进行初始化的加载动作

JAVA中创建线程的三种方式的使用与区别

JAVA中通过继承Thread类、实现Runnable接口以及实现Callable接口配合Future接口实现创建多线程,三种方式各有优缺点,而第三种则具备更多的增强能力

SpringBoot配置文件生效优先级

SpringBoot 可以通过 spring.profiles.active 属性指定生效不同配置文件来满足多环境要求,多环境更为复杂的场景,就需要理解配置文件生效优先级,考虑直接引入外部配置项和配置文件

自定义filter过滤器拦截未登录(非法)请求

在JAVA的WEB工程中我们可以将JSP页面文件放在WEB-INFO中限制用户进行URL直接访问,但静态资源如js、css文件却是需要被外部直接访问的,直接对外暴露又不太安全,可以通过自定义过滤器处理

Maven的pom配置文件的scope属性

scope属性主要用于控制依赖范围,主要分为编译、打包、运行、测试、依赖传递等各个常见,scope不同于optional提供了更多可选择的配置参数用于应对不同的依赖场景。

JAVA语言中的反射机制的作用原理及使用

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;动态获取的信息以及动态调用对象的方法的功能。

选择个人头像

昵称

邮箱

QQ

网址

评论提示

  • 头像:系统为您提供了12个头像自由选择,初次打开随机为你选择一个
  • 邮箱:可选提交邮箱,该信息不会外泄,或将上线管理员回复邮件通知
  • 网址:可选提交网址,评论区该地址将以外链的形式展示在您的昵称上
  • 记忆:浏览器将记忆您已选择或填写过得信息,下次评论无需重复输入
  • 审核:提供一个和谐友善的评论环境,本站所有评论需要经过人工审核