当前位置: 首页 > news >正文

【设计模式】创建型设计模式之 建造者模式

文章目录

  • 一、介绍
      • 定义
      • UML 类图
  • 二、用法1 简化复杂对象具体构建过程
      • 省略抽象的 Builder 类
      • 省略 Director 类
  • 三、用法2 控制对象构造方法、限制参数关系
      • Guava 中使用建造者模式构建 cache 来进行参数校验

一、介绍

定义

建造者模式,将一个复杂的对象的构建过程与表示分离,使得同样的构建过程可以构建不同的结果。

UML 类图

  1. 抽象建造者类(Builder):这个接口规定要实现复杂对象的哪些部分的创建,并不涉及具体的部件对象的创建。
  2. 具体建造者类(ConcreteBuilder):实现 Builder 接口,完成复杂产品的各个部件的具体创建方法。在构造过程完成后,提供一个方法,返回创建好的负责产品对象。
  3. 产品类(Product):要创建的复杂对象 (包含多个组成部件).
  4. 指挥者类(Director):调用具体建造者来创建复杂对象的各个部分,在指导者中不涉及具体产品的信息,只负责保证对象各部分完整创建或按某种顺序创建(客户端一般只需要与指挥者进行交互)。

二、用法1 简化复杂对象具体构建过程

public class Bike {//车架private String frame;//座椅private String seat;public String getFrame() {return frame;}public void setFrame(String frame) {this.frame = frame;}public String getSeat() {return seat;}public void setSeat(String seat) {this.seat = seat;}
}
public abstract class Builder {protected Bike mBike = new Bike();public abstract void buildFrame();public abstract void buildSeat();public abstract Bike createBike();
}public class HelloBuilder extends Builder {@Overridepublic void buildFrame() {mBike.setFrame("碳纤维车架");}@Overridepublic void buildSeat() {mBike.setSeat("橡胶车座");}@Overridepublic Bike createBike() {return mBike;}
}public class MobikeBuilder extends Builder {@Overridepublic void buildFrame() {mBike.setFrame("铝合金车架");}@Overridepublic void buildSeat() {mBike.setSeat("真皮车座");}@Overridepublic Bike createBike() {return mBike;}
}
public class Director {private Builder mBuilder;public Director(Builder builder) {this.mBuilder = builder;}public Bike construct() {mBuilder.buildFrame();mBuilder.buildSeat();return mBuilder.createBike();}
}
public class Client {public static void main(String[] args) {showBike(new HelloBuilder());showBike(new MobikeBuilder());}private static void showBike(Builder builder) {Director director = new Director(builder);Bike bike = director.construct();System.out.println(bike.getFrame());System.out.println(bike.getSeat());}
}

省略抽象的 Builder 类

如果系统中只需要一个具体的建造者,可以省略抽象建造者

省略 Director 类

如果只有一个具体建造者,并且简化了抽象建造者,那么可以省略掉指导者。让建造者直接扮演指导者。

三、用法2 控制对象构造方法、限制参数关系

建造者模式除了用来分离复杂对象的构建过程,还可以在构建对象的同时用来控制对象的构造方法,限制对象的构建参数关系。
使用场景

  1. 场景 1 类的构造方法过长,降低了代码可读性容易搞错参数。
  2. 场景 2 使用 setter 方法优化场景 1,但是这样会导致遗漏对象的必填属性并且无法实现前后关联属性的设置。
public class ResourcePoolConfig{private String name;private int maxTotal;private int maxIdle;private int minIdle;//如果builder是内部类,这里就可以使用private修饰public ResourcePoolConfig (Builder builder){this.name=builder.name;this.maxTotal=builder.maxTotal;this.maxIdle=builder.maxIdle;this.minIdle=builder.minIdle;}
}
public class Builder{private static final int DEFAULT_MAX_TOTAL=9;private String name;private int maxTotal;//其他字段省略public ResourcePoolConfig build(){//可以写校验逻辑return new ResourcePoolConfig(this);}public Builder setMaxTotal(int maxTotal){//校验if(maxTotal<0){throw new RuntimeException();}this.maxTotal=maxTotal;}}

Guava 中使用建造者模式构建 cache 来进行参数校验

Guava本地缓存框架是Google的Guava库提供的一种高性能、线程安全的本地缓存实现。它旨在帮助Java开发者有效地管理和存储数据在内存中,从而加速数据访问速度并减轻对底层数据存储系统的压力。Guava Cache具有丰富的特性和灵活性,使其成为处理高并发和高性能需求场景下的理想选择。下面是一些关键特性和使用方法的概述:
主要特性:

  1. 自动加载(Loading Cache): Guava允许你定义一个CacheLoader,当缓存中没有请求的键对应的值时,自动调用加载数据的方法并插入到缓存中。这样可以实现懒加载,并且保持代码的简洁性。
  2. 缓存过期: 支持基于时间(如访问后多久过期、写入后多久过期)或基于大小(如缓存容量达到上限后开始移除旧的条目)的过期策略,自动管理缓存项的有效性,避免内存泄漏。
  3. 统计信息: 提供丰富的统计信息,比如命中率、平均加载时间等,帮助监控和优化缓存性能。
  4. 软引用和弱引用: Guava Cache允许使用软引用或弱引用存储缓存项,这样当JVM内存紧张时,这些引用可以被垃圾收集器回收,以避免内存溢出。
  5. 并发支持: 内部实现高度并发安全,利用锁和其他同步机制确保在多线程环境下的正确性和高效性。
  6. 可自定义的缓存行为: 通过CacheBuilder,你可以自定义缓存的各种行为,比如缓存过期策略、最大容量、加载机制、统计开启与否等。

guava 中就通过建造者模式来解决了构造参数过长的问题,因为如果先构造无参对象再通过 SET 赋值参数则无法实现必要的参数校验;
Google Guava 中构建内存缓存的案例如下;

public class CacheDemo{public static void main(String[] args){Cache<String,String> cache = CacheBuilder.newBuilder().initialCapacity(100).maximumSize(100).expireAfterWrite(10,TimeUnit.MINUTES).build();cache.put("key1","value1");String value = cache.getIfPresent("key1");System.out.println(value);}
}

相关文章:

  • 【机器学习】让计算机变得更加智能
  • IDEA创建Maven项目
  • 【设计模式】创建型设计模式之 工厂模式
  • 我要成为算法高手-双指针篇
  • 34.打印K型
  • Vue10-事件修饰符
  • React@16.x(25)useReducer
  • orbslam2代码解读(4):loopclosing回环检测线程
  • 从票务到游戏:Celestia 首届黑客松亮点项目盘点
  • 笨蛋学算法之LeetCodeHot100_2_字母异位词分组(Java)
  • 【机器学习理论基础】定量变量和定性变量
  • 30岁迷茫?AI赛道,人生新起点
  • 【ArcGIS微课1000例】0115:字段数据类型案例详解
  • 计算广告读书杂记-待整理
  • 【AI大模型】Transformers大模型库(七):单机多卡推理之device_map
  • 【译】JS基础算法脚本:字符串结尾
  • Android 架构优化~MVP 架构改造
  • Angular数据绑定机制
  • github指令
  • HTTP--网络协议分层,http历史(二)
  • Java的Interrupt与线程中断
  • laravel5.5 视图共享数据
  • LeetCode541. Reverse String II -- 按步长反转字符串
  • Median of Two Sorted Arrays
  • python3 使用 asyncio 代替线程
  • Python连接Oracle
  • TiDB 源码阅读系列文章(十)Chunk 和执行框架简介
  • ⭐ Unity 开发bug —— 打包后shader失效或者bug (我这里用Shader做两张图片的合并发现了问题)
  • 阿里云爬虫风险管理产品商业化,为云端流量保驾护航
  • 安装python包到指定虚拟环境
  • 代理模式
  • 官方解决所有 npm 全局安装权限问题
  • 计算机在识别图像时“看到”了什么?
  • 力扣(LeetCode)56
  • 浏览器缓存机制分析
  • 前言-如何学习区块链
  • 原生Ajax
  • 职业生涯 一个六年开发经验的女程序员的心声。
  • ​渐进式Web应用PWA的未来
  • ​力扣解法汇总1802. 有界数组中指定下标处的最大值
  • ​力扣解法汇总946-验证栈序列
  • # 安徽锐锋科技IDMS系统简介
  • ## 临床数据 两两比较 加显著性boxplot加显著性
  • $HTTP_POST_VARS['']和$_POST['']的区别
  • (1/2)敏捷实践指南 Agile Practice Guide ([美] Project Management institute 著)
  • (11)MATLAB PCA+SVM 人脸识别
  • (70min)字节暑假实习二面(已挂)
  • (Spark3.2.0)Spark SQL 初探: 使用大数据分析2000万KF数据
  • (创新)基于VMD-CNN-BiLSTM的电力负荷预测—代码+数据
  • (二)springcloud实战之config配置中心
  • (附源码)springboot“微印象”在线打印预约系统 毕业设计 061642
  • (附源码)springboot人体健康检测微信小程序 毕业设计 012142
  • (免费分享)基于springboot,vue疗养中心管理系统
  • (原創) 博客園正式支援VHDL語法著色功能 (SOC) (VHDL)
  • (转载)利用webkit抓取动态网页和链接