「 Java基础-对象 」一篇文章讲清楚Java开发中如何更优雅的创建对象

简介: 本文介绍了java独享创建的6中基本方法,引申出复杂非业务场景如何优雅地创建对象这一问题,我们依次使用的对象构建方法如下:> 1、单一构造函数> 2、多构造函数> 3、JavaBean方式> 4、Builder方式最终,通过比较得出Builder方法最为合适的解决办法。

一 前言

Java是面向对象的编程语言,作为Java开发人员,我们每天都会创建许多对象.只要代码中用到,就需要创建对象。

Java语言中,创建对象基本的方法有6种,下面我们来一一地在回顾一下。

二 基本的方法

(1)使用new关键字

Personp1=newPerson();

(2)反射之ClassnewInstance()

Personp2=Person.class.newInstance();

(3)反射之Constructor类的newInstance()

Personp3=Person.class.getDeclaredConstructor().newInstance();

(4)Object对象的clone方法

Personp4= (Person) p1.clone();

注意事项:

a、注意Object类的clone方法是protected的,在Override的时候,可以改成public,这样让其它所有类都可以调用。

b、注意浅拷贝和深拷贝。

(5)反序列化

ObjectOutputStreamoos=newObjectOutputStream(newFileOutputStream("object.bin"));
oos.writeObject(p1);
oos.close();
ObjectInputStreamois=newObjectInputStream(newFileInputStream("object.bin"));
Personp5= (Person) ois.readObject();
ois.close();

注意事项:

a、必须要实现Serializable接口;

b、需要注意哪些字段可序列化,哪些字段不会被序列化,如何控制;

c、注意serialVersionUID的作用;

(6)使用Unsafe类

Fieldf=Unsafe.class.getDeclaredField("theUnsafe");
f.setAccessible(true);
Unsafeunsafe= (Unsafe) f.get(null);
Personp6= (Person) unsafe.allocateInstance(Person.class);

*注意:此方法一般不常用,只是作为知识框架的补充提及。

基本方法小结

1、正常创建:通过new操作符;

2、反射创建:调用Classjava.lang.reflect.ConstructornewInstance()方法;

3、克隆创建:调用现有对象的clone()方法;

4、反序列化:调用java.io.ObjectInputStreamgetObject()方法反序列化;

5、使用Unsafe类:不常用的方法


三 优雅的方式

Java对象的创建方式是其语法明确规定,用户不可能从外部改变的。本文仍然要使用上面的方式来创建对象,只是方式更高级、更优雅!

假设有这样一个场景,现在要构建一个大型的对象,这个对象包含许多个参数的对象,有些参数有些是必填的,有些则是选填的。那么如何构建优雅、安全地构建这个对象呢?

方式一:单一构造函数

通常,我们第一反应能想到的就是单一构造函数方式。直接new的方式构建,通过构造函数来传递参数,见下面的代码:

/*** 单一构造函数*/publicclassPerson01 {
// 姓名(必填)privateStringname;
// 年龄(必填)privateintage;
// 身高(选填)privateintheight;
// 学校(选填)privateStringschool;
// 爱好(选填)privateStringhobby;
publicPerson01(Stringname, intage, intheight, Stringschool, Stringhobby) {
this.name=name;
this.age=age;
this.height=height;
this.school=school;
this.hobby=hobby;
    }
}

上面的构建方式有下面的缺点:

1、有些参数是可以选填的(如height,school),在构建Person的时候必须要传入可能并不需要的参数。

2、现在上面才5个参数,构造函数就已经非常长了。如果是20个参数,写出来的代码可读性、可维护性就会非常差。

3、构建的这样的对象非常容易出错。客户端必须要对照Javadoc或者参数名来讲实参传入对应的位置。如果参数都是String类型的,一旦传错参数,编译是不会报错的,但是运行结果却是错误的。

方式二:多构造函数

/*** 多构造函数*/publicclassPerson02 {
// 姓名(必填)privateStringname;
// 年龄(必填)privateintage;
// 身高(选填)privateintheight;
// 学校(选填)privateStringschool;
// 爱好(选填)privateStringhobby;
publicPerson02(Stringname, intage) {
this.name=name;
this.age=age;
    }
publicPerson02(Stringname, intage, intheight) {
this.name=name;
this.age=age;
this.height=height;
    }
publicPerson02(Stringname, intage, intheight, Stringschool) {
this.name=name;
this.age=age;
this.height=height;
this.school=school;
    }
publicPerson02(Stringname, intage, intheight, Stringschool, Stringhobby) {
this.name=name;
this.age=age;
this.height=height;
this.school=school;
this.hobby=hobby;
    }
}

上面的方式确实能在一定程度上降低构造函数的长度,但是却有下面的缺陷:

1、导致类过长:这种方式会使得Person类的构造函数成阶乘级增长。按理来说,应该要写的构造函数数是可选成员变量的组合数。如果让其他人调用这样的类,绝对会在心里默念xx!!

2、有些参数组合无法重构。因为Java中重载是有限制的,相同方法签名的方法不能构成重载,编译时无法通过。譬如包含(name,age,school)和(name,age,hobby)的构造函数是不能重载的,因为shcoolhobby同为String类型。Java只认变量的类型,不认变量的含义。

方式三:JavaBean方式

直接上代码:

/*** javaBean 方式创建对象*/publicclassPerson03 {
// 姓名(必填)privateStringname;
// 年龄(必填)privateintage;
// 身高(选填)privateintheight;
// 学校(选填)privateStringschool;
// 爱好(选填)privateStringhobby;
publicPerson03(Stringname, intage) {
this.name=name;
this.age=age;
    }
publicvoidsetHeight(intheight) {
this.height=height;
    }
publicvoidsetSchool(Stringschool) {
this.school=school;
    }
publicvoidsetHobby(Stringhobby) {
this.hobby=hobby;
    }
}

 

客户端使用这个对象的代码如下:

/*** 使用Person03 对象的客户端*/publicclassClient03 {
publicstaticvoidmain(String[] args) {
Person03person03=newPerson03("xiaoliu",12);
person03.setHeight(170);
person03.setSchool("北京大学");
person03.setHobby("读书");
    }
}

这样看起来完美的解决了Person对象构建的问题,使用起来非常优雅便捷。确实,在单一线程的环境中这确实是一个非常好的构建对象的方法,但是如果是在多线程环境中仍有其致命缺陷。

在多线程环境中,这个对象不能安全地被构建,因为它不是不可变对象。一旦Person对象被构建,我们随时可通过setXXX()方法改变对象的内部状态。假设有一个线程正在执行与Person对象相关的业务方法,另外一个线程改变了其内部状态,这样得到莫名其妙的结果。由于线程运行的无规律性,使得这问题有可能不能重现.

方式四:Builder 方式

为了完美地解决这个问题,我们使用构建器(Builder)来优雅、安全地构建Person对象。直接上代码:

/*** Builder 方式*/publicclassPerson04 {
// 姓名(必填)privatefinalStringname;
// 年龄(必填)privatefinalintage;
// 身高(选填)privatefinalintheight;
// 学校(选填)privatefinalStringschool;
// 爱好(选填)privatefinalStringhobby;
/*** 这个私有构造函数的作用:* 1、成员变量的私有化,final 类型的变量必须进行初始化,否则无法编译成功* 2、私有构造函数能够保证该对象无法从外部创建,并且person类无法被继承* @param name* @param age* @param height* @param school* @param hobby*/privatePerson04(Stringname, intage, intheight, Stringschool, Stringhobby) {
this.name=name;
this.age=age;
this.height=height;
this.school=school;
this.hobby=hobby;
    }
publicvoiddoSomeThing(){
//todo do what you want!    }
/*** 构建器* 为什么Builder是内部静态类?* 1、必须是person的内部类,否则,由于Person的构造函数私有,不能通过new的方式创建对象;* 2、必须是静态类,由于Person对象无法从外部创建,如果不是静态类,则外部无法引用Builder对象;* 注意:Builder的内部成员变量要与Person的成员变量保持一致*/publicstaticclassBuilder{
// 【这里成员变量不能是final的】// 姓名(必填)privateStringname;
// 年龄(必填)privateintage;
// 身高(选填)privateintheight;
// 学校(选填)privateStringschool;
// 爱好(选填)privateStringhobby;
publicBuilder(Stringname,intage){
this.name=name;
this.age=age;
        }
publicBuildersetHeight(intheight) {
this.height=height;
returnthis;
        }
publicBuildersetSchool(Stringschool) {
this.school=school;
returnthis;
        }
publicBuildersetHobby(Stringhobby) {
this.hobby=hobby;
returnthis;
        }
/*** 构建对象* 返回待够贱的对象本身* @return*/publicPerson04build(){
returnnewPerson04(name,age,height,school,hobby);
        }
    }
}

客户端使用这个对象的代码如下:

/*** 使用 Person04 对象的客户端*/publicclassClient04 {
publicstaticvoidmain(String[] args) {
/*** 通过链式调用的方式创建Person对象,非常优雅!*/Person04person04=newPerson04                .Builder("xiaoliu",25)
                .setHeight(175)
                .setSchool("北京大学")
                .setHobby("reading")
                .build();
person04.doSomeThing();
    }
}

通过privatePerson(..)使得Person类不可被继承;

通过将Person类的成员变量设置为final类型,使得其不可变;

通过Person内部的static Builder类来构建Person对象;

通过将Builder类内部的setXXX()方法返回Builder类型本身,实现链式调用构建Person对象;

至此,我们就相对完美地解决这一类型的对象创建问题!总结待创建的对象特点:

1、需要用户手动的传入多个参数,并且有多个参数是可选的、顺序任意;

2、对象不可变;

3、对象所属的类不是为了继承而设计的,即类不能被继承;

四 总结

本文介绍了java独享创建的6中基本方法,引申出复杂非业务场景如何优雅地创建对象这一问题,我们依次使用的对象构建方法如下:

1、单一构造函数

2、多构造函数

3、JavaBean方式

4、Builder方式

最终,通过比较得出Builder方法最为合适的解决办法。

五 参考

Java创建对象的六种方法-权当记录一下

感谢前人的经验、分享和付出,让我们可以有机会站在巨人的肩膀上眺望星辰大海!

相关文章
|
2月前
|
监控 Java API
如何使用Java语言快速开发一套智慧工地系统
使用Java开发智慧工地系统,采用Spring Cloud微服务架构和前后端分离设计,结合MySQL、MongoDB数据库及RESTful API,集成人脸识别、视频监控、设备与环境监测等功能模块,运用Spark/Flink处理大数据,ECharts/AntV G2实现数据可视化,确保系统安全与性能,采用敏捷开发模式,提供详尽文档与用户培训,支持云部署与容器化管理,快速构建高效、灵活的智慧工地解决方案。
|
10天前
|
移动开发 前端开发 Java
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
JavaFX是Java的下一代图形用户界面工具包。JavaFX是一组图形和媒体API,我们可以用它们来创建和部署富客户端应用程序。 JavaFX允许开发人员快速构建丰富的跨平台应用程序,允许开发人员在单个编程接口中组合图形,动画和UI控件。本文详细介绍了JavaFx的常见用法,相信读完本教程你一定有所收获!
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
|
1月前
|
Java 开发者 微服务
Spring Boot 入门:简化 Java Web 开发的强大工具
Spring Boot 是一个开源的 Java 基础框架,用于创建独立、生产级别的基于Spring框架的应用程序。它旨在简化Spring应用的初始搭建以及开发过程。
58 6
Spring Boot 入门:简化 Java Web 开发的强大工具
|
21天前
|
存储 JavaScript 前端开发
基于 SpringBoot 和 Vue 开发校园点餐订餐外卖跑腿Java源码
一个非常实用的校园外卖系统,基于 SpringBoot 和 Vue 的开发。这一系统源于黑马的外卖案例项目 经过站长的进一步改进和优化,提供了更丰富的功能和更高的可用性。 这个项目的架构设计非常有趣。虽然它采用了SpringBoot和Vue的组合,但并不是一个完全分离的项目。 前端视图通过JS的方式引入了Vue和Element UI,既能利用Vue的快速开发优势,
103 13
|
26天前
|
算法 Java API
如何使用Java开发获得淘宝商品描述API接口?
本文详细介绍如何使用Java开发调用淘宝商品描述API接口,涵盖从注册淘宝开放平台账号、阅读平台规则、创建应用并申请接口权限,到安装开发工具、配置开发环境、获取访问令牌,以及具体的Java代码实现和注意事项。通过遵循这些步骤,开发者可以高效地获取商品详情、描述及图片等信息,为项目和业务增添价值。
57 10
|
19天前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
55 2
|
28天前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
4月前
|
存储 Java
Java编程中的对象和类
【8月更文挑战第55天】在Java的世界中,“对象”与“类”是构建一切的基础。就像乐高积木一样,类定义了形状和结构,而对象则是根据这些设计拼装出来的具体作品。本篇文章将通过一个简单的例子,展示如何从零开始创建一个类,并利用它来制作我们的第一个Java对象。准备好让你的编程之旅起飞了吗?让我们一起来探索这个神奇的过程!
39 10
|
4月前
|
存储 Java
Java的对象和类的相同之处和不同之处
在 Java 中,对象和类是面向对象编程的核心。
70 18
|
4月前
|
Java
Java 对象和类
在Java中,**类**(Class)和**对象**(Object)是面向对象编程的基础。类是创建对象的模板,定义了属性和方法;对象是类的实例,通过`new`关键字创建,具有类定义的属性和行为。例如,`Animal`类定义了`name`和`age`属性及`eat()`、`sleep()`方法;通过`new Animal()`创建的`myAnimal`对象即可调用这些方法。面向对象编程通过类和对象模拟现实世界的实体及其关系,实现问题的结构化解决。
35 4