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

[设计模式] 建造者模式

一、引言

起因是学习okhttp过程中遇到的这段代码

        Request request = original.newBuilder().url(original.url()).header("Authorization", "Bearer " + BearerTokenUtils.getToken(configuration.getApiKey(), configuration.getApiSecret())).header("Content-Type", Configuration.JSON_CONTENT_TYPE).header("User-Agent", Configuration.DEFAULT_USER_AGENT).header("Accept", Configuration.SSE_CONTENT_TYPE).method(original.method(), original.body()).build();

newBuilder().build()创建对象的方式以前没注意过,搜了一下,这种方法属于广义上的建造者模式,于是进行学习。

二、介绍

我自己概括建造者模式的思想:把一个整体A,分割成多个部分A1,A2,A3,A4。每个部分分开建造,最后组装成一个整体。

设想这样的场景,装修一个房子,需要考虑吊顶、涂料、地板、瓷砖等部分。每个部分都有不同的品牌选择。

装修公司提供一些成套的、整体装修方案,可以看作是这些部分的排列组合。

三、代码

我们用代码进行描述。以下是代码结构

模块builder-pattern-01中,Matter是材料的父接口,定义了材料的各种属性。

public interface Matter {/*** 场景:ceiling、coat、floor、tile* @return 吊顶、涂料、地板、瓷砖*/String scene();/*** 品牌* @return 自定字符串*/String brand();/*** 型号* @return 自定义字符串*/String model();/*** 单价(平米报价)* @return BigDecimal*/BigDecimal price();/*** 描述* @return 自定义字符串*/String desc();
}

 ceiling、coat、floor、tile分别表示四个部分:吊顶、涂料、地板、瓷砖。

这四个部分有不同的材料。这些材料组合装修,成为一个整体方案。代码略。

builder-pattern-02中,IMenu是装修包接口,用于添加材料;DecoratiuonPackageMenu是其实现。
 

/*** 装修包接口* 用于添加材料*/
public interface IMenu {/*** 添加吊顶* @param matter 材料* @return 装修包*/IMenu appendCeiling(Matter matter);/*** 添加涂料* @param matter 材料* @return 装修包*/IMenu appendCoat(Matter matter);/*** 添加地板* @param matter 材料* @return 装修包*/IMenu appendFloor(Matter matter);/*** 添加瓷砖* @param matter 材料* @return 装修包*/IMenu appendTile(Matter matter);/*** 获取装修包信息* @return 装修包detail*/String getDetail();
}
public class DecorationPackageMenu implements IMenu{/** 材料清单 */private List<Matter> list = new ArrayList<>(16);/** 总价格 */private BigDecimal price = BigDecimal.ZERO;/** 面积 */private BigDecimal area;/** 装修等级 */private String grade;public DecorationPackageMenu(){}public DecorationPackageMenu(double area, String grade) {this.area = new BigDecimal(area);this.grade = grade;}@Overridepublic IMenu appendCeiling(Matter matter) {list.add(matter);// 注意这个price=赋值操作,单纯的add等运算不会改变原来的值// price = price + ( area * (unitPrice * matter.price()) )price = price.add(area.multiply(matter.price()));return this;}@Overridepublic IMenu appendCoat(Matter matter) {list.add(matter);price = price.add(area.multiply(matter.price()));return this;}@Overridepublic IMenu appendFloor(Matter matter) {list.add(matter);price = price.add(area.multiply(matter.price()));return this;}@Overridepublic IMenu appendTile(Matter matter) {list.add(matter);price = price.add(area.multiply(matter.price()));return this;}@Overridepublic String getDetail() {StringBuilder detail = new StringBuilder("\r\n-------------------------------------------------------\r\n" +"装修清单: " + "\r\n" +"套餐等级: " + grade + "\r\n" +"套餐价格: " + price.setScale(2, BigDecimal.ROUND_HALF_UP) + "元\r\n" +"房屋面积: " + area.doubleValue() + "平方米\r\n" +"材料清单: " + "\r\n" );for (Matter matter : list) {detail.append("场景:").append(matter.scene()).append("、").append("品牌:").append(matter.brand()).append("、").append("类别:").append(matter.model()).append("、").append("平米价格:").append(matter.price()).append("元\r\n");}return detail.toString();}
}

builder是建造者,使用IMenu装修包对各种材料进行组装

public class Builder {public IMenu levelOne(Double area) {return new DecorationPackageMenu(area, "豪华欧式").appendCeiling(new LevelOneCeiling()).appendCoat(new DuluxCoat()).appendFloor(new DerFloor()).appendTile(new DongPengTile());}public IMenu levelTwo(Double area) {return new DecorationPackageMenu(area, "轻奢田园").appendCeiling(new LevelTwoCeiling()).appendCoat(new NipponCoat()).appendFloor(new ShengXiangFloor()).appendTile(new MarcoPoloTile());}public IMenu levelThree(Double area) {return new DecorationPackageMenu(area, "现代简约").appendCeiling(new LevelOneCeiling()).appendCoat(new NipponCoat()).appendFloor(new ShengXiangFloor()).appendTile(new DongPengTile());}
}

我们使用单元测试进行测试一下

    @Testpublic void test_builder() {Builder builder = new Builder();System.out.println(builder.levelOne(132.5D).getDetail());System.out.println(builder.levelTwo(150.0D).getDetail());System.out.println(builder.levelThree(170.0D).getDetail());}

结果部分为
 

其余略

四、回归okhttp

建造者模式思想基本领略,那么okhttp的newBuilder().build()创建对象的思想跟上面差不多。

那么对于这段代码,

        Request request = original.newBuilder().url(original.url()).header("Authorization", "Bearer " + BearerTokenUtils.getToken(configuration.getApiKey(), configuration.getApiSecret())).header("Content-Type", Configuration.JSON_CONTENT_TYPE).header("User-Agent", Configuration.DEFAULT_USER_AGENT).header("Accept", Configuration.SSE_CONTENT_TYPE).method(original.method(), original.body()).build();

我们可以进行学习newBuilder().build()这种创建对象的方式,先来看看其源码

public Builder newBuilder() {return new Builder(this);}public Request build() {if (this.url == null) {throw new IllegalStateException("url == null");} else {return new Request(this);}}public Builder url(HttpUrl url) {if (url == null) {throw new NullPointerException("url == null");} else {this.url = url;return this;}}public Builder header(String name, String value) {this.headers.set(name, value);return this;}public Builder method(String method, @Nullable RequestBody body) {if (method == null) {throw new NullPointerException("method == null");} else if (method.length() == 0) {throw new IllegalArgumentException("method.length() == 0");} else if (body != null && !HttpMethod.permitsRequestBody(method)) {throw new IllegalArgumentException("method " + method + " must not have a request body.");} else if (body == null && HttpMethod.requiresRequestBody(method)) {throw new IllegalArgumentException("method " + method + " must have a request body.");} else {this.method = method;this.body = body;return this;}}

通过newBuilder()创建了一个Builder类的对象,然后每次url() header()等方法修改属性,都是在对Builder类的对象进行修改,最后build()方法 通过Builder类的对象创建了Request对象。

这种基于不可变对象(或者为了不改变原对象)而创建新对象的方式也值得我们学习。

基于此思想,创建一个类Message

@Data
public class Message implements Serializable {private String role;private String content;private String name;public Message(){}public Message(Builder builder) {this.role = builder.role;this.content = builder.content;this.name = builder.name;}/*** 注意这里 static方法*/public static Builder builder() {return new Builder();}/*** 建造者模式*/public static final class Builder {private String role;private String content;private String name;public Builder(){}public Builder role(Constants.Role role) {this.role = role.getCode();return this;}public Builder content(String content) {this.content = content;return this;}public Builder name(String name) {this.name = name;return this;}public Message build() {return new Message(this);}}
}

得注意build()方法是在最后使用的,所以应该是Builder的方法,并且返回值类型应该是Message类。

补充,后来才发现lombok有@Builder注解,就是这个方法......

@Builder

相关文章:

[设计模式] 建造者模式

一、引言 起因是学习okhttp过程中遇到的这段代码 Request request original.newBuilder().url(original.url()).header("Authorization", "Bearer " BearerTokenUtils.getToken(configuration.getApiKey(), configuration.getApiSecret())).header(&quo…...

在DDD领域驱动下的微服务数据库的MVC设计思路(高度可行性)

在DDD领域驱动下的微服务架构中使用MVC设计思路来设计数据库是可行的&#xff0c;因为MVC是一种经典的软件架构模式&#xff0c;可以将应用程序分为三个主要部分&#xff1a;模型、视图和控制器。在微服务架构中&#xff0c;每个微服务可以看作是一个模块&#xff0c;可以使用M…...

Leetcode2834. 找出美丽数组的最小和

Every day a Leetcode 题目来源&#xff1a;2834. 找出美丽数组的最小和 解法1&#xff1a;贪心 从最小正整数 1 开始枚举&#xff0c;设当前数为 num&#xff0c;如果 nums 里没有 target - num&#xff0c;就说明可以添加 num&#xff0c;依次填满直到有 n 个数即可。 用…...

acwing算法基础之搜索与图论--kruskal算法

目录 1 基础知识2 模板3 工程化 1 基础知识 kruskal算法的关键步骤为&#xff1a; 将所有边按照权重从小到大排序。定义集合S&#xff0c;表示生成树。枚举每条边(a,b,c)&#xff0c;起点a&#xff0c;终点b&#xff0c;边长c。如果结点a和结点b不连通&#xff08;用并查集来…...

微信H5跳转微信小程序

官方文档&#xff1a;目录 | 微信开放文档 方法一&#xff1a;微信浏览器打开的h5跳转方式 HTML代码 <wx-open-launch-weapp id"launch-btn" username"所需跳转的小程序原始id" path"pages/pay/pay"><script type"text/wxtag…...

Yii2 引入 外部无命名空间的类,Class not found

记一次问题解决 问题描述 支付宝开放平台SDK v2 无命名空间。需 require 引入。 require Yii::$app->vendorPath."/alipay-sdk-php/v2/aop/AopClient.php"; var_dump(new AopClient([]));exit();上述写法会直接报错。 Class temporary\controllers\AopClient …...

设计模式是测试模式咩?

设计模式和测试模式概述 软件的生命周期为什么要进行测试&#xff08;测试的目的&#xff09;&#xff1f;软件的设计模式1. **瀑布模型**3. 增量和迭代模型4. 敏捷模型5. 喷泉模型 测试模型V模型W模型 一个应用程序从出生到“死亡”会经过非常漫长的流程…… 软件的生命周期 …...

Aspose.OCR for .NET 2023Crack

Aspose.OCR for .NET 2023Crack 为.NET在图片上播放OCR使所有用户和程序员都可以从特定的图像片段中提取文本和相关的细节&#xff0c;如字体、设计以及书写位置。这一特定属性为OCR的性能及其在扫描遵循排列的记录时的功能提供了动力。OCR的库使用一条线甚至几条线来处理这些特…...

conda环境中pytorch1.2.0版本安装包安装一直失败解决办法!!!

conda环境中pytorch1.2.0版本安装包安装一直失败解决办法 cuda10.0以及cudnn7.4现在以及安装完成&#xff0c;就差torch的安装了&#xff0c;现在torch我要装的是1.2.0版本的&#xff0c;安装包以及下载好了&#xff0c;安装包都是在这个网站里下载的&#xff08;点此进入&…...

后端面试问题(学习版)

JAVA相关 JAVA语言概述 1. 一个".java"源文件中是否可以包含多个类&#xff1f;有什么限制&#xff1f; 可以。 一个源文件可以声明多个类&#xff0c;但是最多只能有一个类使用public进行声明 且要求声明public的类的类名与源文件相同。 2. Java的优势&#xff…...

数据管理系统-week1-介绍

文章目录 一、数据它是什么&#xff1f;二、电子存储设备三、持久存储设备1、硬盘驱动器&#xff08;HDD&#xff09;、硬盘、硬盘驱动器是一种用于存储和检索数字信息的机电持久存储设备。2、固态硬盘&#xff08;SSD&#xff09;使用非易失性存储器&#xff0c;即使用NAND闪存…...

【SpringBoot】手写模拟SpringBoot核心流程

依赖包 新建一个工程&#xff0c;包含两个 module&#xff1a; springboot 模块&#xff0c;表示 springboot 源码实现&#xff1b;user 模块&#xff0c;表示业务系统&#xff0c;使用 springboot 模块&#xff1b; 依赖包&#xff1a;Spring、SpringMVC、Tomcat 等&#xff…...

应对.locked勒索病毒:恢复、预防全方位攻略

导言&#xff1a; .locked勒索病毒并非简单的数字威胁&#xff0c;它是一场对个人和企业数字资产的精密审判。这种病毒通过各种方式感染系统&#xff0c;从而以瞬间之间将用户的关键文件变成数字拼图&#xff0c;无情地要求赎金以换取解锁的密钥。如果您正在经历勒索病毒数据恢…...

基于DS1302时钟液晶12864显示2路闹钟仿真及源程序

一、系统方案 1、本设计采用51单片机作为主控器。 2、DS1302采集年月日时分秒送到液晶12864显示。 3、按键年月日时分秒&#xff0c;两路闹钟。 二、硬件设计 原理图如下&#xff1a; 三、单片机软件设计 1、首先是系统初始化 uchar clock_time[6] {0X00,0X59,0X23,0X09,0X…...

AGC034E Complete Compress

AGC034E Complete Compress 洛谷[AGC034E] Complete Compress 题目大意 给你一棵有 n n n个节点的树&#xff0c;并用 01 01 01串告诉你哪些节点上有棋子&#xff08;恰好一棵&#xff09;。 你可以进行若干次操作&#xff0c;每次操作可以将两颗距离至少为 2 2 2的棋子向彼…...

python设计模式12:状态模式

什么是状态机&#xff1f; 关键属性&#xff1a; 状态和转换 状态&#xff1a; 系统当前状态 转换&#xff1a;一种状态到另外一种状态的变化。 转换由触发事件或是条件启动。 状态机-状态图 状态机使用场景&#xff1a; 自动售货机 电梯 交通灯 组合锁 停车计时…...

JS对图片尺寸和DPI进行编辑修改(1寸照修改为2寸照)

各种报名都对照片有大小限制&#xff0c;鉴于这种情况&#xff0c;网上搜了后拼凑出了如下代码&#xff0c;用于解决1寸照片修改为2寸照片&#xff0c;同时将DPI修改为300&#xff0c;当然也可以根据自己的情况修改代码&#xff1a; HTML <input type"file" id&…...

EDA实验----四选一多路选择器设计(QuartusII)

目录 一&#xff0e;实验目的 二&#xff0e;实验仪器设备 三&#xff0e;实验原理&#xff1a; 四&#xff0e;实验要求 五&#xff0e;实验内容及步骤 1.实验内容 2.实验步骤 六&#xff0e;实验报告 七.实验过程 1.创建Verilog文件&#xff0c;写代码 2.波形仿真 …...

从windows iso文件中提取install.wim

1、首先从微软官方下载需要的windows镜像 https://www.microsoft.com/zh-cn/software-download/windows10/ 2、在下载的iso文件右键&#xff0c;打开压缩包&#xff0c;在sources文件夹下&#xff0c;应该就可以看到install.wim了。但似乎在最新的win10版本&#xff0c;微软采…...

Python的flask网页编程的GET和POST方法的区别

关于flask网页编程的GET及POST方法之间存在哪些区别问题&#xff0c;我们主要从以下六个关键点予以详细阐述&#xff1a; 首先需要明确的是&#xff0c;GET与POST两种不同类型的HTTP方法所采用的请求模式有所差别。其中&#xff0c;GET方法采用的是基于URL请求的机制&#xff…...

团队任务管理软件哪个好?trello、Worktile、Todoist等10大产品对比

本文将深入对比 10 款团队任务管理软件&#xff1a;PingCode、Worktile、Trello、Todoist、Asana、monday.com、ClickUp、Wrike、Jira Confluence、TAPD。一、任务越来越多&#xff0c;真正难的是“协作不确定”团队任务管理这件事&#xff0c;最开始看起来很简单&#xff1a;…...

构建现代化Vue应用界面:Shadcn-Vue组件化架构设计与实践指南

构建现代化Vue应用界面&#xff1a;Shadcn-Vue组件化架构设计与实践指南 【免费下载链接】shadcn-vue Vue port of shadcn-ui 项目地址: https://gitcode.com/gh_mirrors/sh/shadcn-vue 在Vue开发生态中&#xff0c;构建一致、美观且可维护的UI界面一直是开发团队面临的…...

Obsidian新库配置不同步?3分钟搞定插件和主题迁移(附详细路径)

Obsidian新库配置迁移全指南&#xff1a;一键同步插件与主题设置 刚在Obsidian里新建了一个知识库&#xff0c;却发现所有插件和主题设置都消失了&#xff1f;这种"从零开始"的挫败感我太熟悉了。作为一款以Markdown为核心的笔记工具&#xff0c;Obsidian的插件生态是…...

利用node.forge.js实现前端数据加密传输的最佳实践

1. 为什么前端需要数据加密传输&#xff1f; 在Web开发中&#xff0c;前端与后端的数据交互往往涉及敏感信息&#xff0c;比如用户密码、身份证号、银行卡信息等。这些数据如果以明文形式传输&#xff0c;很容易被中间人攻击&#xff08;MITM&#xff09;截获。想象一下&#x…...

Z-Image-GGUF参数详解:EmptyLatentImage尺寸设置与边缘裁剪规避技巧

Z-Image-GGUF参数详解&#xff1a;EmptyLatentImage尺寸设置与边缘裁剪规避技巧 1. 引言&#xff1a;为什么你的图片总被“切掉”一部分&#xff1f; 如果你用过Z-Image-GGUF生成图片&#xff0c;可能遇到过这样的情况&#xff1a;明明想要一张横屏的风景图&#xff0c;结果生…...

Rust文件I/O操作深度解析

Rust文件I/O操作深度解析作为一名从后端开发转向Rust的开发者&#xff0c;我发现Rust的文件I/O操作与Python的文件操作有很多相似之处&#xff0c;但也有一些不同。Rust的文件I/O操作更加注重安全性和性能&#xff0c;同时保持了Rust的类型安全特性。今天我想分享一下我对Rust文…...

LiuJuan Z-Image Generator部署教程:NVIDIA Jetson Orin边缘设备部署可行性

LiuJuan Z-Image Generator部署教程&#xff1a;NVIDIA Jetson Orin边缘设备部署可行性 想在自己的NVIDIA Jetson Orin设备上跑一个高质量的图片生成工具吗&#xff1f;今天我们来聊聊LiuJuan Z-Image Generator在边缘设备上的部署可能性。 这是一个基于阿里云通义Z-Image扩散…...

SOCD Cleaner:重塑游戏输入体验的键盘映射神器

SOCD Cleaner&#xff1a;重塑游戏输入体验的键盘映射神器 【免费下载链接】socd Key remapper for epic gamers 项目地址: https://gitcode.com/gh_mirrors/so/socd 在竞技游戏的微秒级对决中&#xff0c;一个被忽视的技术细节往往成为胜负的关键——同时按下相反方向键…...

如何避免职业停滞?测试工程师的5年跃迁计划

停滞的陷阱与破局契机在技术迭代加速的2026年&#xff0c;软件测试领域正经历深刻变革&#xff1a;AI测试工具覆盖率突破40%&#xff0c;云原生架构普及率达75%&#xff0c;持续测试成为DevOps核心环节。然而行业调研显示&#xff0c;73%的测试从业者在工作5年后陷入能力平台期…...

Scrapy实战:5sing原创音乐网多页数据爬取(完整可运行,附避坑指南)

Scrapy实战&#xff1a;5sing原创音乐网多页数据爬取&#xff08;完整可运行&#xff0c;附避坑指南&#xff09; 今天给大家带来一个高频实战案例——使用Scrapy框架爬取5sing原创音乐网的多页歌曲数据。作为爬虫领域的经典场景&#xff0c;「列表页多页爬取详情页深度解析」…...