Python设计模式 - 建造者模式

定义

建造者模式是一种创建型设计模式,主要用于构建包含多个组成部分的复杂对象。它将对象的构建过程与表示分离,使得同样的构建过程可以创建不同的对象表示。

结构

在这里插入图片描述

  • 抽象建造者(Builder):声明创建产品的各个部件的方法,以及一个获取产品对象的方法。
  • 具体建造者(ConcreteBuilder):实现抽象建造者的方法,这些方法用来构建产品的各个部件。
  • 产品(Product):被构建的复杂对象。
  • 指挥者(Director):负责安排产品各部件的建造顺序,调用抽象建造者的方法构建产品的各个部分,最后返回创建好的产品对象。
  • 客户端(Client):使用指挥者和建造者来构建产品。

应用场景

  1. 对象结构复杂,创建过程涉及多个步骤:当对象由多个组件组成,且这些组件的初始化顺序或组合方式较灵活时,建造者模式可以将创建过程拆分为多个步骤,使得对象的构建更加清晰。例如,构建 Computer 对象时,可以逐步配置 CPU、内存、硬盘等组件。
  2. 需要灵活配置对象:当对象的某些部分是可选的,或者可以根据不同需求定制时,建造者模式能够提供流畅的配置方式,而不影响对象的整体创建。例如,创建游戏角色时,可以选择不同的装备、技能组合,而无需修改底层实现。

优缺点

优点:

  1. 创建过程与表示分离:建造者模式将对象的创建过程和最终表示解耦,使得同样的构造过程可以创建不同的对象变体
  2. 封装复杂创建过程:将对象的创建流程封装在指挥者中,客户端无需关心具体的构造细节,而是通过指挥者直接获取完整的对象,从而简化调用逻辑。
  3. 提高代码的可扩展性:新增产品时,只需扩展新的具体建造者,而无需修改现有代码,符合开闭原则。

缺点:

  1. 可能导致冗余代码:对于每种不同的产品类型,都需要创建对应的建造者类,这在产品种类较多但构建逻辑相似的情况下,可能导致大量重复代码,不如使用抽象工厂模式更合适。

代码示例

from abc import ABC, abstractmethodclass Computer:""" 计算机产品 """def __init__(self, brand):self.brand = brandself.cpu = Noneself.memory = Noneself.storage = Noneself.graphics_card = None  # 可选组件(独立显卡)def __str__(self):gpu_info = f", {self.graphics_card} GPU" if self.graphics_card else ", No GPU"return f"{self.brand} Computer with {self.cpu}, {self.memory} RAM, {self.storage} Storage{gpu_info}"class ComputerBuilder(ABC):""" 计算机建造者基类(抽象类) """def __init__(self, brand):self.computer = Computer(brand)@abstractmethoddef set_cpu(self):pass@abstractmethoddef set_memory(self):pass@abstractmethoddef set_storage(self):pass@abstractmethoddef set_graphics_card(self):passdef build(self):return self.computerclass LenovoComputerBuilder(ComputerBuilder):""" 联想电脑建造者(默认带独立显卡) """def __init__(self):super().__init__("Lenovo")def set_cpu(self):self.computer.cpu = "Intel i7"return selfdef set_memory(self):self.computer.memory = "16GB"return selfdef set_storage(self):self.computer.storage = "512GB SSD"return selfdef set_graphics_card(self):self.computer.graphics_card = "NVIDIA RTX 4090"return selfclass MacComputerBuilder(ComputerBuilder):""" Mac 电脑建造者(默认不带独立显卡) """def __init__(self):super().__init__("Mac")def set_cpu(self):self.computer.cpu = "Apple M2"return selfdef set_memory(self):self.computer.memory = "32GB"return selfdef set_storage(self):self.computer.storage = "1TB SSD"return selfdef set_graphics_card(self):return selfclass Director:""" 指挥者,控制建造流程 """@staticmethoddef construct_pc(builder: ComputerBuilder):""" 统一的构造过程 """return (builder.set_cpu().set_memory().set_storage().set_graphics_card().build())# 客户端代码
lenovo_pc = Director.construct_pc(LenovoComputerBuilder())  # 联想默认带显卡
mac_pc = Director.construct_pc(MacComputerBuilder())  # Mac 默认无显卡print(lenovo_pc)  # Lenovo Computer with Intel i7, 16GB RAM, 512GB SSD Storage, NVIDIA RTX 4090 GPU
print(mac_pc)  # Mac Computer with Apple M2, 32GB RAM, 1TB SSD Storage, No GPU

省略Director

在有些情况下,为了简化系统结构,可以将Director和抽象建造者Builder进行合并,在Builder中提供逐步构建复杂产品对象的construct()方法。

class ComputerBuilder(ABC):""" 计算机建造者基类(抽象类) """def __init__(self, brand):self.computer = Computer(brand)@abstractmethoddef set_cpu(self):pass@abstractmethoddef set_memory(self):pass@abstractmethoddef set_storage(self):pass@abstractmethoddef set_graphics_card(self):passdef construct(self):""" 统一的构造过程 """return (self.set_cpu().set_memory().set_storage().set_graphics_card().computer)# 客户端代码:直接选择 Builder 并调用 construct()
lenovo_pc = LenovoComputerBuilder().construct()  # 联想默认带显卡
mac_pc = MacComputerBuilder().construct()  # Mac 默认无显卡print(lenovo_pc)  # Lenovo Computer with Intel i7, 16GB RAM, 512GB SSD Storage, NVIDIA RTX 4090 GPU
print(mac_pc)  # Mac Computer with Apple M2, 32GB RAM, 1TB SSD Storage, No GPU

这种方式不影响系统的灵活性和可扩展性,同时还简化了系统结构,但是加重了抽象建造者类的职责。如果construct()方法较为复杂,待构建产品的组成部分较多,建议还是将construct()方法单独封装在Director中,这样做更符合单一职责原则。

参考

《设计模式的艺术》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/31248.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

音乐API

https://neteasecloudmusicapi.vercel.app/docs/#/https://neteasecloudmusicapi.vercel.app/docs/#/ 使用实例 所有榜单内容摘要 说明 : 调用此接口,可获取所有榜单内容摘要 接口地址 : /toplist/detail 调用例子 : /toplist/detail 获取歌单所有歌曲 说明 : 由于网易云…

Jetpack Compose — 入门实践

一、项目中使用 Jetpack Compose 从此节开始,为方便起见,如无特殊说明,Compose 均指代 Jetpack Compose。 开发工具: Android Studio 1.1 创建支持 Compose 新应用 新版 Android Studio 默认创建新项目即为 Compose 项目。 注意:在 Language 下拉菜单中,Kotlin 是唯一可…

【day12】进程切换与调度:linux系统的幕后操控术

【Day12】进程切换与调度:linux系统的幕后操控术 进程优先级进程属性:UID进程属性:PRI和NI进程饥饿 竞争/独立/并行/并发进程切换进程调度(O(1)调度算法) 进程优先级 进程优先级的本质:衡量进程得到CPU资源…

STM32之BKP

VBAT备用电源。接的时候和主电源共地,正极接在一起,中间连接一个100nf的电容。BKP是RAM存储器。 四组VDD都要接到3.3V的电源上,要使用备用电池,就把电池正极接到VBAT,负极跟主电源共地。 TEMPER引脚先加一个默认的上拉…

mapbox高阶,结合threejs(threebox)添加管道

👨‍⚕️ 主页: gis分享者 👨‍⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍⚕️ 收录于专栏:mapbox 从入门到精通 文章目录 一、🍀前言1.1 ☘️mapboxgl.Map 地图对象1.2 ☘️mapboxgl.Map style属性1.3 ☘️threebox Tube静态对象二、🍀使用thr…

Android15使用FFmpeg解码并播放MP4视频完整示例

效果: 1.编译FFmpeg库: 下载FFmpeg-kit的源码并编译生成安装平台库 2.复制生成的FFmpeg库so文件与包含目录到自己的Android下 如果没有prebuiltLibs目录,创建一个,然后复制 包含目录只复制arm64-v8a下

利用FatJar彻底解决Jar包冲突(三)

利用FatJar彻底解决Jar包冲突 Spring 容器的加载与隔离⽀持注解配置⽂件定位与容器初始化嵌套Spring容器的加载 隔离优化EagleEye traceId不⼀致问题原因解决 Spring 容器的加载与隔离 ⽀持注解 这个⽐较容易,主要是我们之前的应⽤不⽀持⼆⽅包内部的注解&#xf…

ThinkPHP8.0+MySQL8.0搭建简单实用电子证书查询系统

客户花了100元买了一个系统,开始不能导入,到处找人帮忙解决。给解决能导入了,不能修改,满足不了用户的需求。用户一狠心,花200块钱,叫我给他定制了一个电子证书查询系统。还免费给部署到服务器。惭愧惭愧……

越早越好!8 个反直觉的金钱真相|金钱心理学

很多人都追求财富自由,但成功的人少之又少。 这可能是因为,人们往往忽略了一些金钱的真相和常识。 01 金钱常识 & 真相 为了构建健康的金钱观,我读了一本有点反直觉,有点像鸡汤,但都是财富真相的书。 来自 Morg…

文本转语音-音画适时推送rtsp并播放

文本语音 rtsp适时播放叫号系统的底层逻辑 发布Linux, unix socket 和window win32做为音频源的 python10下的(ffmpeg version 7.1) 可运行版本. 这两天在弄这个,前2篇是通过虚拟声卡,达到了最简单的一个逻辑,播放文本就从声卡发声&#xff0…

类和对象(中)

1.类的6个默认成员函数 如果一个类中什么成员都没有,简称为空类。空类中真的什么都没有吗?并不是,任何类在什么都不写时,编译器会自动生成以下6个默认成员函数。默认成员函数:用户没有显式实现,编译器会生成…

搜索插入位置(js实现,LeetCode:35)

给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 示例 1: 输入: nums [1,3,5,6], target 5 输出: 2示例 2: 输入…

【问题记录】如何编译nv_peer_memory模块?依赖OFED的4个目录和2类文件?如何解决没有rdma/peer_mem.h文件?

背景 GDR:GPUDirect RDMA。这项新技术在 GPU 内存之间直接与 NVIDIA HCA/NIC 设备之间提供直接的 P2P(点对点)数据路径。这显着降低了 GPU-GPU 通信延迟,并完全减轻了 CPU 的负担。nv_peer_memory模块是网卡提供给GPU使用GDR技术…

3.6c语言

#define _CRT_SECURE_NO_WARNINGS #include <math.h> #include <stdio.h> int main() {int sum 0,i,j;for (j 1; j < 1000; j){sum 0;for (i 1; i < j; i){if (j % i 0){sum i;} }if (sum j){printf("%d是完数\n", j);}}return 0; }#de…

李彦宏:紧抓AI智能体爆发元年机遇 推动新质生产力加快发展

来源 人民网 3月5日公布的政府工作报告&#xff0c;“智能”一词被提及10次&#xff0c;“大模型”则首次被写入。作为大模型和人工智能领域的领军企业&#xff0c;我们感到非常振奋&#xff0c;同时深感责任重大。 报告在介绍今年政府工作任务时提出&#xff0c;“因地制宜发…

基于Python Django的人脸识别上课考勤系统(附源码,部署)

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;…

5G基本概念

作者:私语茶馆 1. 5G应用场景概述 1.1.5G应用场景 ITU域2015年定义了三大应用场景:eMBB(增强型移动宽带)、uRLLC(低时延高可靠通信)、mMTC(海量物联网通信); emBB:Enhanced Mobile Broadband ,移动互联网应用,是4G MBB(移动宽带)的升级,主要侧重于网络速率、带…

VSCode+AI编程生态实战:从环境配置到智能编码的全栈指南

Python环境的搭建&#xff0c;我们选择了Anaconda环境&#xff0c;有Python环境&#xff0c;接下来要搭建当然是AI辅助编程环境了。在搭建这个之前&#xff0c;或者说大家在打算安装VSCode之前&#xff0c;需要先检查自已电脑是否已经安装了VSCode&#xff0c;因为这个软件是Wi…

Ubuntu工控卫士在制造企业中的应用案例

在当今数字化时代&#xff0c;信息安全已成为各行各业关注的焦点。特别是在工业控制领域&#xff0c;网络安全Linux工控机&#xff08;简称“工控卫士”&#xff09;作为信息安全防护的核心设备&#xff0c;正发挥着越来越重要的作用。本文将深入探讨工控卫士在信息安全防护场景…

Stable Diffusion教程|快速入门SD绘画原理与安装

什么是Stable Diffusion&#xff0c;什么是炼丹师&#xff1f;根据市场研究机构预测&#xff0c;到2025年全球AI绘画市场规模将达到100亿美元&#xff0c;其中Stable Diffusion&#xff08;简称SD&#xff09;作为一种先进的图像生成技术之一&#xff0c;市场份额也在不断增长&…