Java 是面向对象的语言,不可避免的,“对象”这个概念是 Java 语言的核心部分,这里来简单讨论一下在 Java 中创建一般对象的方法。

  总结下来有以下4种创建对象的方法:

  • 使用 new 关键字调用对象的构造器;
  • 使用 Java 反射的 newInstance() 方法;
  • 使用 Object 类的 clone() 方法;
  • 使用对象流 ObjectInputStream 的 readObject() 方法读取序列化对象;

1.      使用 new 关键字

  最常见的 Java 对象的构造方法,通过调用类提供的构造器创建对象。

2.      使用 newInstance() 方法

  Java 反射中有一个 newInstance() 方法,可以创建对象,步骤如下:

  • 获取要创建的类的 Class 对象。
  • 如果只需要调用这个类的访问权限为 public 无参构造器,直接使用 Class 类的实例方法 newInstance()。
  • 获取 Class 对象的构造器对象,通过调用 Class 类的实例方法 getDeclaredConstractors() 来获取构造器对象的数组。(获取所有构造器,无视访问权限的限制,数组顺序按照代码中的顺序决定)
  • 如果调用的构造器是 private 的,需要调用 Constractor 类的父类 AccessibleObject 类的实例方法 setAccessible(true) 来打破访问限制。
  • 使用 Constractor 类的实例方法 newInstance()。

  示例代码:

 public class MethodNewInstance {

     public static void main(String[] args) throws Exception {

         // 得到类对象
Class<?> clazz = Class.forName("com.gerrard.create.method_newInstance.ObjectToCreate");
// 类对象的 newInstance() 方法,只能调用公有的无参构造器
clazz.newInstance(); // 得到构造器对象数组(不管是私有还是公有的构造器)
Constructor<?>[] cons = clazz.getDeclaredConstructors();
cons[1].newInstance();
cons[2].newInstance("Gerrard");
// 先打破私有构造器不可访问的限制
cons[0].setAccessible(true);
cons[0].newInstance("Gerrard", "Info");
}
}

MethodNewInstance

  备注:

  • 获取 Class 对象的方法有3个,此处不多赘述。
  • 获取 Constractor 对象的方法有4个,此处不多赘述。

3.      使用 clone() 方法

  Object 类是所有类的直接或间接父类,Object 类中提供了 实例方法 clone(),在给定对象的基础上,创建一个完全相同的对象。步骤如下:

  • 想要使用 clone() 方法创建对象的类,实现 Cloneable 接口。
  • 在类的内部,重写 Object 类的 clone() 方法。

  示例代码:

 public class ObjectToCreate implements Cloneable {

     // 重写 Object 类的 clone() 方法(native 方法)
public ObjectToCreate clone() {
ObjectToCreate obj = null;
try {
obj = (ObjectToCreate) super.clone();
} catch (CloneNotSupportedException e) {
// 没有实现 Cloneable 接口就会抛出这个异常
e.printStackTrace();
}
return obj;
}
}

ObjectToCreate

  备注:

  • 没有实现 Cloneable 接口,会抛出 CloneNotSupportedException 异常。
  • Object 类提供的 clone() 方法,是浅复制。
  • Object 类的 clone() 方法,是 native 方法。

4.      使用反序列化的 readObject() 方法

  这个方法一共分两步:

  • 将对象序列化,存储到一个文件中。
  • 从文件中反序列化,得到类对象。

  序列化:

  • 想要序列化对象的类,实现 Serializable 接口。
  • 使用文件输出流 FileOutputStream 创建存储序列化之后对象的文件。
  • 使用对象输出流 ObjectOutputStream 的实例方法 writeObject(obj)。
  • 判断类中是否存在,名为writeReplace(),返回类型为 Object 的方法,若有,写入这个方法的返回值;否则,写入 obj 对象。

  反序列化:

  • 使用文件输入流 FileInputStream 找到存储序列化对象的文件。
  • 使用对象输入流 ObjectInputStream 的实例方法 readObject()。
  • 判断类中是否存在,名为readResolve(),返回类型为 Object 的方法,若有读取这个对象;否则,反序列化文件中的对象流。

  示例代码:

 public class ObjectToCreate implements Serializable {

     private static final long serialVersionUID = 1L;

     private Object writeReplace(){
return new Integer(1);
} private Object readResolve(){
return new Double(2);
}
}

ObjectToCreate

 public class MethodSerialable {

     public static void main(String[] args) {

         // 默认路径是项目的根路径
final String fileName = "./file/serialable.txt"; ObjectToCreate o1 = new ObjectToCreate(); // 序列化
try (FileOutputStream fos = new FileOutputStream(fileName);
ObjectOutputStream oos = new ObjectOutputStream(fos);) {
oos.writeObject(o1);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}

MethodSerialable

 public class MethodAntiSerialable {

     public static void main(String[] args) {
// 默认路径是项目的根路径
final String fileName = "./file/serialable.txt";
Object o2 = null;
// 反序列化
try (FileInputStream fio = new FileInputStream(fileName); ObjectInputStream ois = new ObjectInputStream(fio);) {
o2 = ois.readObject();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
System.out.println(o2);
}
}

MethodAntiSerialable

  备注:

  • 在类中,writeReplace() 和 readResoleve() 是两个非常特殊的方法,其特征签名需要严格限制:方法名限定,参数个数限定为0,返回类型必须是 Object,不能为 Object 的子类,但是可以抛出不同的异常。访问修饰符没有限制,但一般推荐为 private,防止误操作。其特殊的地方还在于将其设为 private 方法,没有其他方法调用的情况下,编译器不会发出警告。

5.      总结

  Java 创建对象的4种方法:第一种是最常用的;第二种方法深入至源码会指向 sun.reflect.ConstructorAccessor 类,JDK 中似乎没有提供继续深入下去的源码,但是既然是调用构造器的方法,那么与第一种方法一样,创建的对象是存储在堆(Heap)中的;第三种方法是要实现特定的接口才可以使用,而且是通过调用 native 方法,也就是非 Java 代码(很大可能是 C)实现的,也就是说,这个方法产生的对象,可能不会被 GC 回收(个人的想法),因为 GC 是用来回收 Java 代码创造的对象,所以要慎用;第四种方法在序列化的时候,需要实现特定的接口,而在反序列化时就不关心这一点了,它是将对象暂存于其他媒介中,在反序列化的时候将对象存于堆中。

第001弹:Java 中创建对象的4种方式的更多相关文章

  1. Java中创建对象的几种方式

    Java中创建对象的五种方式: 作为java开发者,我们每天创建很多对象,但是我们通常使用依赖注入的方式管理系统,比如:Spring去创建对象,然而这里有很多创建对象的方法:使用New关键字.使用Cl ...

  2. Java中创建对象的五种方式

    我们总是讨论没有对象就去new一个对象,创建对象的方式在我这里变成了根深蒂固的new方式创建,但是其实创建对象的方式还是有很多种的,不单单有new方式创建对象,还有使用反射机制创建对象,使用clone ...

  3. 【转】Java中创建对象的5种方式

    Java中创建对象的5种方式   作为Java开发者,我们每天创建很多对象,但我们通常使用依赖管理系统,比如Spring去创建对象.然而这里有很多创建对象的方法,我们会在这篇文章中学到. Java中有 ...

  4. 第一弹:Java 中创建对象的4种方式

    Java 是面向对象的语言,不可避免的,"对象"这个概念是 Java 语言的核心部分,这里来简单讨论一下在 Java 中创建一般对象的方法. 总结下来有以下4种创建对象的方法: 使 ...

  5. Java中创建对象的5种方式

    作为Java开发者,我们每天创建很多对象,但我们通常使用依赖管理系统,比如Spring去创建对象.然而这里有很多创建对象的方法,我们会在这篇文章中学到. Java中有5种创建对象的方式,下面给出它们的 ...

  6. Java中创建对象的5种方式 &&new关键字和newInstance()方法的区别

    转载:http://www.kuqin.com/shuoit/20160719/352659.html 用最简单的描述来区分new关键字和newInstance()方法的区别:newInstance: ...

  7. Java技术——Java中创建对象的5种方式

    此文为译文 原文连接:https://dzone.com/articles/5-different-ways-to-create-objects-in-java-with-ex 0. 前言 作为Jav ...

  8. Java 中创建对象的 5 种方式!

    Java中有5种创建对象的方式,下面给出它们的例子还有它们的字节码 Employee类: class Employee implements Cloneable, Serializable { pri ...

  9. Java中创建对象的5种方法

    将会列举5种方法去创建 Java 对象,以及他们如何与构造函数交互,并且会有介绍如何去使用这些方法的示例. 作为一个 Java 开发人员,我们每天都会创建大量的 Java 对象,但是我们通常会使用依赖 ...

随机推荐

  1. Mongodb之failed to create service entry worker thread

    Mongodb "failed to create service entry worker thread" 错误. 系统:CentOS release 6.8 mongod.lo ...

  2. [VC]char 和 wchar_t相互转化

    #include <windows.h> #include <stdio.h> //function: charTowchar //purpose:char to WCHAR ...

  3. Oracle RAC/Clusterware 多种心跳heartbeat机制介绍 RAC超时机制分析

    ORACLE RAC中最主要存在2种clusterware集群件心跳 &  RAC超时机制分析: 1.Network Heartbeat 网络心跳 每秒发生一次: 10.2.0.4以后网络心跳 ...

  4. [神经网络]一步一步使用Mobile-Net完成视觉识别(四)

    1.环境配置 2.数据集获取 3.训练集获取 4.训练 5.调用测试训练结果 6.代码讲解 本文是第四篇,下载预训练模型并训练自己的数据集. 前面我们配置好了labelmap,下面我们开始下载训练好的 ...

  5. BCB:AnsiString BSTR WideString

    WideString wstr;AnsiString astr;wchar_t *wp;//或者 BSTR wp; wp=wstr.c_bstr(); //WideString转化为BSTRwstr= ...

  6. Dojo的on函数(以前的dojo.connect)

    ​同jQuery的on函数: require(["esri/map", "dojo/on"], function(Map, on) { // ... on(my ...

  7. Vue路由跳转到新页面时 默认在页面最底部 而不是最顶部 的解决

    今天碰到一个问题   vue路由跳转到新的页面时会直接显示页面最底部  正常情况下是显示的最顶部的  而且好多路由中不是全部都是这种情况  折腾好长时间也没解决  最后在网上找到了解决办法 其实原理很 ...

  8. CentOS7练习

    为编译安装的httpd服务,实现service unit文件破解centos7 口令修改默认的启动内核为新编译内核启动时临时禁用SELinux启动时进入emergency模式卸载编译安装的新内核

  9. SSH框架面试总结----1

    1:struts2的工作流程 1)客户端浏览器发出HTTP请求. 2)根据web.xml配置,HTTP请求会被FilterDispatcher接收. 3)根据struts.xml,找到对应的Actio ...

  10. ThinkPHP5 高级查询之构建分组条件

    ThinkPHP5 高级查询之构建分组条件 一.在tp5中通过where方法如何构建分组条件, 例如:where user_id=$this->user_id and (status in (4 ...