设计模式之【建造者模式】使用java建造者模式优雅创建对象_java使用builder创建对象-程序员宅基地

技术标签: java  设计模式与开发规范  建造者模式  设计模式  


全网最全最细的【设计模式】总目录,收藏起来慢慢啃,看完不懂砍我

什么是建造者模式

将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

Builder模式是一步一步创建一个复杂对象的创建型模式,它允许使用者在不知道内部建造细节的情况下,可以更精细的控制对象的构造流程。该模式是为了将构建复杂对象的过程和它的部件解耦,是的构建过程和不见得表示隔离开来。

一步一步认清建造者模式

我们此处借用数据源对象举例,让我们一步一步的从零开始认识建造者模式。

1、传统方式创建对象

public class DataSourceConfig {
    

    private String host;
    private Integer port;
    private String userName;
    private String password;
    private String database; // oracle、mysql
    private Integer initialSize;
    private Integer minIdle;
    private Integer maxActive;
	
	// 省略 get  set
   
    public static void main(String[] args) {
    
        DataSourceConfig dataSourceConfig = new DataSourceConfig();
        dataSourceConfig.setHost("");
        dataSourceConfig.setPort(3306);
        dataSourceConfig.setInitialSize(10);
        // ...
    }
}

以上代码相信很多小伙伴都不陌生,new一个对象,然后用set方法赋值,又或者使用超长的构造函数来初始化一个对象,比如说下面:

public DataSourceConfig(String host, Integer port, String userName, String password, String database, Integer initialSize, Integer minIdle, Integer maxActive) {
    
        this.host = host;
        this.port = port;
        this.userName = userName;
        this.password = password;
        this.database = database;
        this.initialSize = initialSize;
        this.minIdle = minIdle;
        this.maxActive = maxActive;
    }

构造函数的参数太长,代码在可读性和易用性上都会变差。在使用构造函数的时候,我们就容易搞错各参数的顺序,传递进错误的参数值,导致非常隐蔽的 bug。

而参数过多,使用set方法很容易漏掉某个参数,导致一些问题。

还有一种问题,这种使用 构造方法创建对象的方式,参数值只能程序员自己确保一定是正确的,否则我们做参数的判断可能就需要这样做:

public DataSourceConfig(String host, Integer port, String userName, String password, String database, Integer initialSize, Integer minIdle, Integer maxActive) {
    
    this.host = host;
    this.port = port;
    this.userName = userName;
    this.password = password;
    this.database = database;
    this.initialSize = initialSize;
    this.minIdle = minIdle;
    this.maxActive = maxActive;
    if(port < 0){
    
        throw new IllegalArgumentException("port不能为负数");
    }
    if(initialSize < 0){
    
        throw new IllegalArgumentException("initialSize不能为负数");
    }
    // ... 其他判断
}

而使用set的方式创建对象,可能需要这样做:

public void setPort(Integer port) {
    
    if(port < 0){
    
        throw new IllegalArgumentException("port不能为负数");
    }
    this.port = port;
}

但是,使用set的方式没法解决属性依赖的问题,比如说maxActive字段与initialSize必须是某个比另一个大,等等。

此时,建造者模式可能会解决这一切困惑与难题。

2、建造者模式创建对象

我们采取“约定大于配置”,将一些基础属性设置默认值,并且有些参数校验可以在最后的build校验,同时,如果不放心的话,也可以在set中校验。

以下实例,创建对象就简单太多了:


public class DataSourceConfig {
    

    private String host;
    private Integer port;
    private String database; // oracle、mysql
    private Integer initialSize;
    private Integer minIdle;
    private Integer maxActive;

    // 构造函数私有化
    private DataSourceConfig(Builder builder) {
    
        this.host = builder.host;
        this.port = builder.port;
        this.database = builder.database;
        this.initialSize = builder.initialSize;
        this.minIdle = builder.minIdle;
        this.maxActive = builder.maxActive;
    }


    // builder静态内部类
    public static class Builder {
    
        private static final int DEFAULT_INITIAL_SIZE = 10;
        private static final int DEFAULT_MIN_IDLE = 30;
        private static final int DEFAULT_MAX_ACTIVE = 200;

        // 设置一样的字段,加上初始值
        private String host;
        private Integer port;
        private String database; // oracle、mysql
        private Integer initialSize = DEFAULT_INITIAL_SIZE;
        private Integer minIdle = DEFAULT_MIN_IDLE;
        private Integer maxActive = DEFAULT_MAX_ACTIVE;

        // 对外暴露build方法
        public DataSourceConfig build(){
    
            // 做参数校验,此处省略其他
            if(StringUtils.isBlank(host)){
    
                throw new IllegalArgumentException("host不能为空");
            }
            
            // 直接使用builder返回想要的对象
            return new DataSourceConfig(this);
        }
        
        public Builder setHost(String host) {
    
            this.host = host;
            // 支持链式调用
            return this;
        }
        
        public Builder setPort(Integer port) {
    
            this.port = port;
            // 支持链式调用
            return this;
        }

        public Builder setDatabase(String database) {
    
            this.database = database;
            // 支持链式调用
            return this;
        }
        
        // ... 省略其他set方法
        
    }


    public static void main(String[] args) {
    
        // 约定大于配置
        DataSourceConfig config = new DataSourceConfig.Builder()
                .setHost("localhost")
                .setPort(3306)
                .setDatabase("mysql")
                // 省略其他参数
                .build();
    }
}

建造者模式优缺点

优点:

  • 产品的建造和表示分离,实现了解耦,可以使用相同的创建过程得到不同的产品。
  • 将复杂对象的创建步骤分解在不同的方法中,使得创建过程更加清晰。(良好的封装性,不必知道内部组成的细节)
  • 增加新的具体建造者无需修改原有类库的代码,易于拓展,符合“开闭原则“。

缺点:

  • 对象必须有共同点,限制了使用范围。建造者模式创造出来的对象,其组成部分基本相同。如果对象之间的差异较大,则不适用这个模式。
  • 产生多余的builder对象以及director对象,消耗内存。
  • 如内部变化复杂,会有很多的建造类,难以维护。

建造者模式适用环境

  • 相同的方法,不同的执行顺序,产生不同的事件结果。
  • 对各部件或零件,都可以配到一个对象中,但是产生的运行结果又不相同时。
  • 产品类非常复杂,或者产品类中的调用顺序不同产生了不同的结果,这个时候使用创建者模式非常合适。
  • 当初始化一个对象特别复杂,如参数多,且很多参数都有默认值时(一个类有多个构造函数的时候,可以考虑使用建造者模式)
  • 需要生成的产品对象有复杂的内部结构,这些产品对象具备共性;
  • 隔离复杂对象的创建和使用,并使得相同的创建过程可以创建不同的产品。
  • 类的属性之间有一定的依赖关系或者约束条件。
  • 如果我们希望创建不可变对象,也就是说,对象在创建好之后,就不能再修改内部的属性值

与工厂模式的区别

工厂模式是用来创建不同但是相关类型的对象(继承同一父类或者接口的一组子类),由给定的参数来决定创建哪种类型的对象。建造者模式是用来创建一种类型的复杂对象,通过设置不同的可选参数,“定制化”地创建不同的对象。

参考资料

https://www.kongzid.com/archives/design3
王争老师《设计模式之美》

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/A_art_xiang/article/details/127792525

智能推荐

分布式光纤传感器的全球与中国市场2022-2028年:技术、参与者、趋势、市场规模及占有率研究报告_预计2026年中国分布式传感器市场规模有多大-程序员宅基地

文章浏览阅读3.2k次。本文研究全球与中国市场分布式光纤传感器的发展现状及未来发展趋势,分别从生产和消费的角度分析分布式光纤传感器的主要生产地区、主要消费地区以及主要的生产商。重点分析全球与中国市场的主要厂商产品特点、产品规格、不同规格产品的价格、产量、产值及全球和中国市场主要生产商的市场份额。主要生产商包括:FISO TechnologiesBrugg KabelSensor HighwayOmnisensAFL GlobalQinetiQ GroupLockheed MartinOSENSA Innovati_预计2026年中国分布式传感器市场规模有多大

07_08 常用组合逻辑电路结构——为IC设计的延时估计铺垫_基4布斯算法代码-程序员宅基地

文章浏览阅读1.1k次,点赞2次,收藏12次。常用组合逻辑电路结构——为IC设计的延时估计铺垫学习目的:估计模块间的delay,确保写的代码的timing 综合能给到多少HZ,以满足需求!_基4布斯算法代码

OpenAI Manager助手(基于SpringBoot和Vue)_chatgpt网页版-程序员宅基地

文章浏览阅读3.3k次,点赞3次,收藏5次。OpenAI Manager助手(基于SpringBoot和Vue)_chatgpt网页版

关于美国计算机奥赛USACO,你想知道的都在这_usaco可以多次提交吗-程序员宅基地

文章浏览阅读2.2k次。USACO自1992年举办,到目前为止已经举办了27届,目的是为了帮助美国信息学国家队选拔IOI的队员,目前逐渐发展为全球热门的线上赛事,成为美国大学申请条件下,含金量相当高的官方竞赛。USACO的比赛成绩可以助力计算机专业留学,越来越多的学生进入了康奈尔,麻省理工,普林斯顿,哈佛和耶鲁等大学,这些同学的共同点是他们都参加了美国计算机科学竞赛(USACO),并且取得过非常好的成绩。适合参赛人群USACO适合国内在读学生有意向申请美国大学的或者想锻炼自己编程能力的同学,高三学生也可以参加12月的第_usaco可以多次提交吗

MySQL存储过程和自定义函数_mysql自定义函数和存储过程-程序员宅基地

文章浏览阅读394次。1.1 存储程序1.2 创建存储过程1.3 创建自定义函数1.3.1 示例1.4 自定义函数和存储过程的区别1.5 变量的使用1.6 定义条件和处理程序1.6.1 定义条件1.6.1.1 示例1.6.2 定义处理程序1.6.2.1 示例1.7 光标的使用1.7.1 声明光标1.7.2 打开光标1.7.3 使用光标1.7.4 关闭光标1.8 流程控制的使用1.8.1 IF语句1.8.2 CASE语句1.8.3 LOOP语句1.8.4 LEAVE语句1.8.5 ITERATE语句1.8.6 REPEAT语句。_mysql自定义函数和存储过程

半导体基础知识与PN结_本征半导体电流为0-程序员宅基地

文章浏览阅读188次。半导体二极管——集成电路最小组成单元。_本征半导体电流为0

随便推点

【Unity3d Shader】水面和岩浆效果_unity 岩浆shader-程序员宅基地

文章浏览阅读2.8k次,点赞3次,收藏18次。游戏水面特效实现方式太多。咱们这边介绍的是一最简单的UV动画(无顶点位移),整个mesh由4个顶点构成。实现了水面效果(左图),不动代码稍微修改下参数和贴图可以实现岩浆效果(右图)。有要思路是1,uv按时间去做正弦波移动2,在1的基础上加个凹凸图混合uv3,在1、2的基础上加个水流方向4,加上对雾效的支持,如没必要请自行删除雾效代码(把包含fog的几行代码删除)S..._unity 岩浆shader

广义线性模型——Logistic回归模型(1)_广义线性回归模型-程序员宅基地

文章浏览阅读5k次。广义线性模型是线性模型的扩展,它通过连接函数建立响应变量的数学期望值与线性组合的预测变量之间的关系。广义线性模型拟合的形式为:其中g(μY)是条件均值的函数(称为连接函数)。另外,你可放松Y为正态分布的假设,改为Y 服从指数分布族中的一种分布即可。设定好连接函数和概率分布后,便可以通过最大似然估计的多次迭代推导出各参数值。在大部分情况下,线性模型就可以通过一系列连续型或类别型预测变量来预测正态分布的响应变量的工作。但是,有时候我们要进行非正态因变量的分析,例如:(1)类别型.._广义线性回归模型

HTML+CSS大作业 环境网页设计与实现(垃圾分类) web前端开发技术 web课程设计 网页规划与设计_垃圾分类网页设计目标怎么写-程序员宅基地

文章浏览阅读69次。环境保护、 保护地球、 校园环保、垃圾分类、绿色家园、等网站的设计与制作。 总结了一些学生网页制作的经验:一般的网页需要融入以下知识点:div+css布局、浮动、定位、高级css、表格、表单及验证、js轮播图、音频 视频 Flash的应用、ul li、下拉导航栏、鼠标划过效果等知识点,网页的风格主题也很全面:如爱好、风景、校园、美食、动漫、游戏、咖啡、音乐、家乡、电影、名人、商城以及个人主页等主题,学生、新手可参考下方页面的布局和设计和HTML源码(有用点赞△) 一套A+的网_垃圾分类网页设计目标怎么写

C# .Net 发布后,把dll全部放在一个文件夹中,让软件目录更整洁_.net dll 全局目录-程序员宅基地

文章浏览阅读614次,点赞7次,收藏11次。之前找到一个修改 exe 中 DLL地址 的方法, 不太好使,虽然能正确启动, 但无法改变 exe 的工作目录,这就影响了.Net 中很多获取 exe 执行目录来拼接的地址 ( 相对路径 ),比如 wwwroot 和 代码中相对目录还有一些复制到目录的普通文件 等等,它们的地址都会指向原来 exe 的目录, 而不是自定义的 “lib” 目录,根本原因就是没有修改 exe 的工作目录这次来搞一个启动程序,把 .net 的所有东西都放在一个文件夹,在文件夹同级的目录制作一个 exe._.net dll 全局目录

BRIEF特征点描述算法_breif description calculation 特征点-程序员宅基地

文章浏览阅读1.5k次。本文为转载,原博客地址:http://blog.csdn.net/hujingshuang/article/details/46910259简介 BRIEF是2010年的一篇名为《BRIEF:Binary Robust Independent Elementary Features》的文章中提出,BRIEF是对已检测到的特征点进行描述,它是一种二进制编码的描述子,摈弃了利用区域灰度..._breif description calculation 特征点

房屋租赁管理系统的设计和实现,SpringBoot计算机毕业设计论文_基于spring boot的房屋租赁系统论文-程序员宅基地

文章浏览阅读4.1k次,点赞21次,收藏79次。本文是《基于SpringBoot的房屋租赁管理系统》的配套原创说明文档,可以给应届毕业生提供格式撰写参考,也可以给开发类似系统的朋友们提供功能业务设计思路。_基于spring boot的房屋租赁系统论文