今天看netty权威指南,第一次听说构建器,百度了几个博客,但是并没有通俗易懂一点儿的,综合别人的博客,总结如下:

1. 构建器是什么?

当创建对象需要传入多个参数的时候我们通常会根据参数的数量写不同的构造器,具体如下

public A(int a){}

public A(int a, int b){}

public A(int a, int b, int c){}

根据不同的参数调用不同的构造器,但是当参数多了的时候,这种方式不够灵活,所以会实现动态传参的方法

public A(){}

public void seta(int a){}

public void setb(int b){}

public void setc(int c){}

这种方式提高了传参的可读性,也提高了传参的灵活性,但是会增加代码行数,同时在多线程异步执行的时候导致奇怪的错误

有没有办法解决呢?既能提高代码可读性,提高参数灵活性,又不会增加代码行数,并保证线程安全呢?

构建器模式登场,先看代码。

这里我们写一个 Person 类,并为这个类加上构建器:

  1. public class Person { 

  2. private final String name; 

  3. private final int age; 


  4. private final String address; 

  5. private final String phone; 


  6. public static class Builder{ 

  7. private final String name; 

  8. private final int age; 


  9. private String address = null; 

  10. private String phone = null; 


  11. public Builder(String name,int age){ 

  12. this.name = name; 

  13. this.age = age; 




  14. public Builder address(String val){ 

  15. address = val; 

  16. return this; 




  17. public Builder phone(String val){ 

  18. phone = val; 

  19. return this; 




  20. public Person builder(){ 

  21. return new Person(this); 






  22. private Person(Builder builder){ 

  23. this.name = builder.name; 

  24. this.age = builder.age; 

  25. this.address = builder.address; 

  26. this.phone = builder.phone; 




  27. @Override 

  28. public String toString() { 

  29. return "name:"+name+" age:"+age+" address:"+address+" phone:"+phone; 






调用这个构建器的方式

public class PersonTest {
public static void main(String[] args) {
Person p = new Person.Builder("tom", 18).address("深圳").phone("110").builder();
System.out.println(p.toString());
}
}

2. 为什么使用构建器?

2.1 参数的限制

静态工厂方法与构造器都有一个共同的局限性,就是它们不能很好的扩展到大量的可选参数。就像我们上面的那个Person 类,在实际中我们会有许多的属性,性别、出生年月、爱好...对与这样的类。

2.2 重叠构造器

我们初学的时候都会选择 重叠构造器(telecoping constructor)模式 。在这种情况下,第一个构造器是实例化对象必须的参数,第二个会多一个参数,就这样叠加,最后是一个有所有参数的构造器

  1. public class Person { 

  2. private final String name; 

  3. private final int age; 


  4. private final String address; 

  5. private final String phone; 


  6. public Person(String name, int age) { 

  7. this(name,age,null); 





  8. public Person(String name, int age, String address) { 

  9. this(name,age,address,null); 




  10. public Person(String name, int age, String address, String phone) { 

  11. super(); 

  12. this.name = name; 

  13. this.age = age; 

  14. this.address = address; 

  15. this.phone = phone; 




  16. @Override 

  17. public String toString() { 

  18. return "name:"+name+" age:"+age+" address:"+address+" phone:"+phone; 






获得对象

public class PersonTest {
public static void main(String[] args) {
Person p = new Person("tom",18,null,"110");
System.out.println(p.toString());
}
}

在这个构造器中也许会有你不想要的参数,如果我们的参数变多了的话,情况就不会很好。
总结一句话:重叠构造器可行,但当有很多的参数的时候,客户端的代码就会很难编写并且不容易阅读我们在使用的时候,必须很仔细的看每一个参数的位置和含义。

2.3 JavaBeans模式

2.3.1 创建JavaBeans模式

这个时候我们还有一种替代的方式,这个就是JavaBeans模式。这种模式下,使用无参的构造方法创建对象,然后调用setter 方法给属性设置值

  1. public class Person { 

  2. private String name; 

  3. private int age; 


  4. private String address; 

  5. private String phone; 


  6. public void setName(String name) { 

  7. this.name = name; 



  8. public void setAge(int age) { 

  9. this.age = age; 



  10. public void setAddress(String address) { 

  11. this.address = address; 



  12. public void setPhone(String phone) { 

  13. this.phone = phone; 




  14. @Override 

  15. public String toString() { 

  16. return "name:"+name+" age:"+age+" address:"+address+" phone:"+phone; 






使用的方式,这个相比与重叠构造器更容易的创建了对象,同时让代码跟容易的阅读。

public class PersonTest {
public static void main(String[] args) {
Person p = new Person();
p.setName("tom");
p.setAge(18);
p.setAddress("深圳");
p.setPhone("110");
System.out.println(p.toString());
}
}

2.3.2 JavaBeans模式的劣势

  • 构造的过程分到了几个调用中,在构造JavaBeans的时候可能会不一致

  • 类无法仅仅通过检验构造器参数的有效性来保证一致性!

  • 对象的不一致会导致失败,JavaBeans模式阻止了把类做为不可变的可能,需要程序员做额外努力来保证它线程安全。

2.4 构建器

  • 构建器的创建对象就比较易于创建与阅读,线程安全

  • 等待所有的参数验证通过才会build()对象。

  • 与构造器相比,builder 的微略优势在于,builder可以有多个可变(varargs)参数。构造器像方法一样,只有一个可变参数。因为builder利用单独的方法来设置每个参数,你想要多少个可变参数,他们就可以有多少个,知道每个setter方法都有一个可变参数。

  • builder模式非常灵活,可以理由单个builder构建多个对象。builder的参数可以在创建对象时进行调整

  • 设置了参数的builder生成一个很好的抽象工厂(Abstract Factory),也就是客户端可以将这样一个builder传给方法,使该方法能为客户端创建一个或者多个对象

  • builder也有自己的不足,就是创建对象就必须创建它的构建器。

  • builder模式还比重叠构造器模式更加的冗长,因此它会在参数多的时候使用。但是我们要知道,我们可能会在设计之后还要添加参数,所以一开始就用构建器还是比较好的。

3 总结

如果类的构造器或者静态工厂中具有多个参数,设计这种类时,Builder模式就是不错的选择,特别是当大多数14:39:30参数都是可选的时候。

  • 与重叠构造器相比,builder牧师的客户端更易与阅读和编写

  • 与JavaBeans相比,更加的安全

Java构建器(多个构造器参数)的更多相关文章

  1. Java 构建器

    假如我们的一个实体类有很多的属性值,但是这些属性值又是可选的.如果我们遇到这样的是类,如何设计出方便的实体类呢? 通常解决办法一: 重叠构造器 public class User { private ...

  2. java拦截器获取请求完整参数

    public class OptLogAspect implements HandlerInterceptor { @Override public boolean preHandle(HttpSer ...

  3. 【读书笔记 - Effective Java】02. 遇到多个构造器参数时要考虑用构建器

    类有多个可选参数的解决方案: 1. 重叠构造器模式可行,但是当有许多参数的时候,客户端代码会很难编写,并且仍然较难以阅读. 2. JavaBeans模式,调用一个无参构造器来创造对象,然后调用sett ...

  4. java构造器和构建器

    本文摘自:https://blog.csdn.net/wh2827991/article/details/79013115 在实例化一个类的过程中,通常会遇到多个参数的构造函数,但如果有些参数是非必需 ...

  5. Java 构造器 遇到多个构造器时要考虑用构建器

    静态工厂和构造器有个共同的局限性:它们都不能很好地扩展到大量的可选参数. 当一个类中有若干个必选属性和多个可选属性时,采用重叠构造器模式.JavaBeans模式或者Builder模式,但各有优劣. 当 ...

  6. Java构造器与构建器的使用

    我们在平常类的构建过程中,可能会面临很多问题,可扩张性.安全性等等.想象一下,这样一个场景,我们现在要创建一个类,其中有6个属性,其中又有4个属性的值是不太确定的(可能某个对象就不需要其中的某个值), ...

  7. 构造器参数过多时考虑使用构建器(Builder)

    一.静态工厂和构造器的局限性 面对需要大量可选参数才能构建对象时,静态工厂和构造器并不能随着可选参数的增加而合理扩展. 假设创建一个类Person需要使用大量的可选参数,其中两个参数是必填的,剩下的都 ...

  8. Effective JAVA 创建和销毁对象 遇到多参构造器考虑使用构建器

    //构建器抽象类,为不同类的实现提供 public interface Builder<T> { public T build(); } /** * @描述: 营养表 */ public ...

  9. Item 2---遇到构造器具有多个参数时,要考虑用构建器;Builder模式

    问题,面对这种一个构造器具备多个参数的问题,现有的做法是使用重叠构造器的方式,该方式存在的问题: public class NutritionFacts { private final int ser ...

随机推荐

  1. 为什么说基于TCP的移动端IM仍然需要心跳保活?(转)

    源:https://segmentfault.com/a/1190000006832547 为什么说基于TCP的移动端IM仍然需要心跳保活?

  2. shell编程系列15--文本处理三剑客之awk格式化输出printf

    shell编程系列15--文本处理三剑客之awk格式化输出printf printf的格式说明符 格式符 含义 %s 打印字符串 %d 打印十进制数 %f 打印一个浮点数 %x 打印十六进制数 %o ...

  3. ES6深入浅出-11 ES6新增的API(上)-2.Array新增API

    Array.form 把不是数组的东西变成数组.最常见的就是把伪数组变成数组 那么什么是伪数组 这就是伪数组,因为它不是继承自Array的原型的对象.它只是一个看起来很像数组的数组 只看下面的代码.a ...

  4. 安卓apk反编译

    在win环境反编译安卓APP的.apk文件 安卓apk 反编译为 Java源码图文教程 Android安全攻防战,反编译与混淆技术完全解析(上)

  5. 机器学习中的数学-线性判别分析(LDA)

    前言在之前的一篇博客机器学习中的数学(7)——PCA的数学原理中深入讲解了,PCA的数学原理.谈到PCA就不得不谈LDA,他们就像是一对孪生兄弟,总是被人们放在一起学习,比较.这这篇博客中我们就来谈谈 ...

  6. keepalived+lvs tcp check 引起的后端服务报Connection reset by peer

    方法一: 取消LVS方式进行tcp转发,进而改为http方式反向代理,问题即可解决. 当然,这是在业务允许使用http的情况下,如果必须使用tcp协议,那就得使用下面的方法了. 方法二: 修改keep ...

  7. 使用mysql连接django时,需要的步骤以及错误解决办法

    django默认使用的sqlite3,更改为SQL时需要按照如下操作进行 1.在settings.py中的78行进行更改 DATABASES = { 'default': { 'ENGINE': 'd ...

  8. [bzoj5483][Usaco2018 Dec]Balance Beam_凸包_概率期望

    bzoj5483 Usaco2018Dec Balance Beam 题目链接:https://lydsy.com/JudgeOnline/problem.php?id=5483 数据范围:略. 题解 ...

  9. 安装AWX

    1.安装最新版python 2.安装最新版docker 设置国内docker镜像源 curl -sSL https://get.daocloud.io/daotools/set_mirror.sh | ...

  10. pytorch1.0神经网络保存、提取、加载

    pytorch1.0网络保存.提取.加载 import torch import torch.nn.functional as F # 包含激励函数 import matplotlib.pyplot ...