【设计模式】简单工厂模式,工厂模式,抽象工厂模式,单例,代理,go案例区分总结

工厂模式三种类型:

一、简单工厂模式(Simple Factory)

定义: 用一个工厂类,根据传入的参数决定创建哪一种具体产品类实例。

面试说法: 由一个统一的工厂创建所有对象,增加新产品时需要修改工厂类,不符合OCP开闭原则。

OCP原则:
对扩展开放(Open for extension):软件中的功能应该允许通过新增代码来进行扩展。
对修改关闭(Closed for modification):原有代码不应该被修改,以减少引入 bug 的风险。

Go 实现:

type PayMethod interface {Pay(amount float64)
}type AliPay struct{}
func (a *AliPay) Pay(amount float64) {fmt.Println("使用支付宝支付", amount)
}type WeChatPay struct{}
func (w *WeChatPay) Pay(amount float64) {fmt.Println("使用微信支付", amount)
}func PayFactory(channel string) PayMethod {switch channel {case "alipay":return &AliPay{}case "wechat":return &WeChatPay{}default:return nil}
}

二、工厂方法模式(Factory Method)

定义: 把创建对象的逻辑下放到每个具体工厂类,遵循了开闭原则。

面试说法: 一个抽象工厂接口 + 多个具体工厂类,每种产品由对应工厂类负责创建。符合开闭原则,新增产品时不修改已有代码,只需新增工厂类。

Go 实现:

type PayMethod interface {Pay(amount float64)
}type PayFactory interface {CreatePayMethod() PayMethod
}type AliPay struct{}
func (a *AliPay) Pay(amount float64) {fmt.Println("使用支付宝支付", amount)
}type AliPayFactory struct{}
func (f *AliPayFactory) CreatePayMethod() PayMethod {return &AliPay{}
}type WeChatPay struct{}
func (w *WeChatPay) Pay(amount float64) {fmt.Println("使用微信支付", amount)
}type WeChatPayFactory struct{}
func (f *WeChatPayFactory) CreatePayMethod() PayMethod {return &WeChatPay{}
}

三、抽象工厂模式(Abstract Factory)

定义: 提供一系列产品创建的接口,每个具体工厂创建一整套产品。

面试说法: 用于创建一系列相关产品,适用于产品族的扩展。每个工厂可以返回多个产品实例,如 UI 工厂创建按钮、窗口、滚动条等。

Go 实现:

// 抽象产品
type Pay interface {Pay(amount float64)
}
type Refund interface {Refund(amount float64)
}// 抽象工厂
type PayFactory interface {CreatePay() PayCreateRefund() Refund
}// 支付宝产品实现
type AliPay struct{}
func (a *AliPay) Pay(amount float64) {fmt.Println("支付宝支付", amount)
}
type AliRefund struct{}
func (a *AliRefund) Refund(amount float64) {fmt.Println("支付宝退款", amount)
}// 支付宝工厂
type AliFactory struct{}
func (f *AliFactory) CreatePay() Pay {return &AliPay{}
}
func (f *AliFactory) CreateRefund() Refund {return &AliRefund{}
}

面试中该如何答工厂模式的区别?

  1. 简单工厂模式

    • 优点:结构简单,适用于产品数量较少、需求变化不频繁的场景。
    • 缺点:不符合开闭原则,新增产品需要修改工厂代码。
    • 应用场景:支付渠道种类少的项目、快速原型开发。
  2. 工厂方法模式

    • 优点:符合开闭原则,新增产品时只需新增具体工厂类,扩展性好。
    • 缺点:类的数量增加,结构相对复杂。
    • 应用场景:产品频繁扩展变化的系统,例如支持多个支付渠道,未来还会新增。
  3. 抽象工厂模式

    • 优点:可以创建一整套相关联的产品(产品族),一致性强。
    • 缺点:不方便支持新增产品种类(每个工厂都要改),灵活性略低。
    • 应用场景:跨平台开发、UI 工具包、数据库驱动(同一工厂生产连接器和执行器等)。

“简单工厂是最基本的一种模式,用一个工厂创建所有对象,不符合开闭原则。工厂方法将创建逻辑分离到每个工厂类,支持扩展。抽象工厂则用于生产一组相关产品,产品族统一,适合跨平台系统。”

工厂模式
✅ 通常只有一个抽象工厂接口多个具体工厂实现类,每个工厂创建一种产品
抽象工厂模式
✅ 有一个抽象工厂接口多个具体工厂实现类,每个工厂创建一个产品族
✅ 强调产品之间的 “族” 关系(如 Windows 风格的按钮 + 复选框必须来自同一个工厂)。

// 抽象产品族
type Button interface { Click() }
type Checkbox interface { Check() }// 具体产品(按族分类)
type WindowsButton struct{}
type WindowsCheckbox struct{}
type MacButton struct{}
type MacCheckbox struct{}// 抽象工厂
type GUIFactory interface {CreateButton() ButtonCreateCheckbox() Checkbox
}// 具体工厂
type WindowsFactory struct{} // 创建Windows风格的Button和Checkbox
type MacFactory struct{}     // 创建Mac风格的Button和Checkbox

四、单例模式(Singleton Pattern)

定义:

确保一个类只有一个实例,并提供一个全局访问点。

使用场景:

  • 配置管理类
  • 数据库连接池
  • 日志处理类
  • 缓存管理器

特点:

  • 全局唯一
  • 延迟初始化(懒汉)
  • 线程安全

Go 语言实现(懒汉 + 线程安全):

package singletonimport ("sync"
)type Singleton struct {Name string
}var (instance *Singletononce     sync.Once
)func GetInstance() *Singleton {once.Do(func() {instance = &Singleton{Name: "唯一实例"}})return instance
}

单例模式确保某个类在整个系统中只有一个实例。常用 sync.Once 来实现线程安全的延迟初始化。在场景如日志记录器、配置加载器中很常见。它的关键是将构造函数私有化,通过提供一个全局访问方法来获取实例。

饿汉

package singletontype EagerSingleton struct {Name string
}var eagerInstance = &EagerSingleton{Name: "饿汉式单例"}func GetEagerInstance() *EagerSingleton {return eagerInstance
}
  1. 懒汉式在第一次使用时才初始化,需要注意并发下的线程安全,Go 中推荐使用 sync.Once 来保证只初始化一次
  2. 饿汉式在程序启动时就创建实例,线程安全但资源利用不够高效。适合对资源敏感性不强的场景。
    实际开发中,推荐懒汉式 + sync.Once 方式来兼顾延迟加载和线程安全。

五、代理模式(Proxy Pattern)

定义:

为其他对象提供一个“代理”以控制对该对象的访问。

使用场景:

  • 访问控制(权限校验)
  • 延迟加载(虚拟代理)
  • 远程代理(RPC Stub)
  • 缓存代理(避免重复计算)

Go 语言实现(以下载器为例,带缓存功能的代理):

package proxyimport "fmt"type Downloader interface {Download(url string)
}// 真实对象
type RealDownloader struct{}func (r *RealDownloader) Download(url string) {fmt.Println("正在下载:", url)
}// 代理对象
type ProxyDownloader struct {real     *RealDownloadercacheMap map[string]bool
}func NewProxyDownloader() *ProxyDownloader {return &ProxyDownloader{real:     &RealDownloader{},cacheMap: make(map[string]bool),}
}func (p *ProxyDownloader) Download(url string) {if p.cacheMap[url] {fmt.Println("缓存命中:", url)} else {p.real.Download(url)p.cacheMap[url] = true}
}

代理模式通过引入一个代理对象来控制对目标对象的访问。比如在缓存代理中,代理会先判断是否命中缓存,如果没有才真正访问目标对象。这样可以在不改动原有逻辑的情况下,增强对象的功能(控制、缓存、权限等)。


https://github.com/0voice

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

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

相关文章

某标杆房企BI平台2.0升级实践

当房地产行业从“规模竞赛”转向“精益运营”,数字化转型成为破局关键。某千亿房企携手亿信华辰,以“用数据重构业务价值链”为目标,历经6个月完成BI平台战略性升级。在这场从“数据可视化”到“决策智能化”的跃迁中,亿信华辰ABI…

Lua 脚本在 Redis 中的运用-24 (使用 Lua 脚本实现原子计数器)

实践练习:使用 Lua 脚本实现原子计数器 实现原子计数器是许多应用程序中的常见需求,例如跟踪网站访问量、限制 API 请求或管理库存。虽然 Redis 提供了 INCR 命令用于递增整数,但在复杂场景或与其他操作结合时直接使用它可能并不足够。本课程探讨了如何在 Redis 中利用 Lua…

Rust 学习笔记:使用迭代器改进 minigrep

Rust 学习笔记:使用迭代器改进 minigrep Rust 学习笔记:使用迭代器改进 minigrep不使用 clone,而使用迭代器使用迭代器适配器使代码更清晰在循环或迭代器之间进行选择 Rust 学习笔记:使用迭代器改进 minigrep 前情提要&#xff1…

el-table配置表头固定而且高度变化

根据官网提示只要在 el-table 元素中定义了 height 属性,即可实现固定表头的表格,而不需要额外的代码。 如果你想既要固定表头,又要下方表格高度自适应,可以设置为 height"100%" : 然后外层设置scroll:

弱光环境下如何手持相机拍摄静物:摄影曝光之等效曝光认知

写在前面 博文内容为一次博物馆静物拍摄笔记的简单总结内容涉及:弱光环境拍摄静物如何选择,以及等效曝光的认知理解不足小伙伴帮忙指正 😃,生活加油 我看远山,远山悲悯 持续分享技术干货,感兴趣小伙伴可以关注下 _ 采…

ARM笔记-ARM伪指令及编程基础

第四章 ARM伪指令及编程基础 4.1 伪指令概述 4.1.1 伪指令定义 人们设计了一些专门用于指导汇编器进行汇编工作的指令,由于这些指令不形成机器码指令,它们只是在汇编器进行汇编工作的过程中起作用,所以被叫做伪指令。 4.1.2 伪指令特征 …

智能手表怎么申请欧盟EN 18031认证

智能手表申请欧盟 EN 18031 认证(针对消费类物联网设备的网络安全标准)的流程与智能门锁类似,但需结合手表的功能特性(如数据交互、定位、支付等)调整合规重点。以下是具体流程和关键要点: 一、标准适配与…

算法-全排列

1、全排列函数的使用 举例&#xff1a;{1,2,3}的全排列 #include<iostream> #include<bits/stdc.h> using namespace std; typedef long long ll; int main(){ll a[3] {1, 2, 3};do{for (ll i 0; i < 3;i){cout << a[i] << " ";}cout…

面试加分秘籍:校招数据倾斜场景下的SQL优化方案

校招面试经常会问大家有没有过调优的经验&#xff0c;相信大家的回答基本都是往数据倾斜和小文件问题这两方面回答&#xff0c;对于数据倾斜相信大部分同学对热key打散或null值引发的倾斜已经非常熟悉&#xff0c;但这些内容面试官也是听腻了&#xff0c;希望大家在面试时候讲一…

Elasticsearch索引机制与Lucene段合并策略深度解析

引言 在现代分布式搜索引擎Elasticsearch中&#xff0c;文档的索引、更新和删除操作不仅是用户交互的核心入口&#xff0c;更是底层存储架构设计的关键挑战。本文围绕以下核心链路展开&#xff1a; 文档生命周期管理&#xff1a;从客户端请求路由到分片定位&#xff0c;从内存…

git提交更改

第一步&#xff1a;添加改动 git add . 第二步&#xff1a;提交改动 git commit -m “替换了 SerialPort 库文件” 第三步&#xff1a;推送到远程 git push 为什么git 的UI界面做的远不如SVN

WPF的基础控件:布局控件(StackPanel DockPanel)

布局控件&#xff08;StackPanel & DockPanel&#xff09; 1 StackPanel的Orientation属性2 DockPanel的LastChildFill3 嵌套布局示例4 性能优化建议5 常见问题排查 在WPF开发中&#xff0c;布局控件是构建用户界面的基石。StackPanel和DockPanel作为两种最基础的布局容器&…

互联网大厂Java求职面试:AI大模型推理服务性能优化与向量数据库分布式检索

互联网大厂Java求职面试&#xff1a;AI大模型推理服务性能优化与向量数据库分布式检索 面试现场&#xff1a;技术总监的连环追问 技术总监&#xff1a;&#xff08;翻看着简历&#xff09;郑薪苦&#xff0c;你在上一家公司参与过LLM推理服务的性能优化项目&#xff1f;说说你…

如何解决网站服务器的异常问题?

当网站服务器出现异常情况&#xff0c;导致用户无法正常访问网页信息的时候&#xff0c;该如何解决这一问题呢&#xff1f;小编下面就带领大家共同探讨一下这一问题。 企业在面对网站服务器异常时&#xff0c;首先要对服务器硬件设备进行详细的检查&#xff0c;可以使用硬盘检测…

Day 35

模型可视化与推理 知识点回顾&#xff1a; 三种不同的模型可视化方法&#xff1a;推荐torchinfo打印summary权重分布可视化 进度条功能&#xff1a;手动和自动写法&#xff0c;让打印结果更加美观 推理的写法&#xff1a;评估模式 模型结构可视化 理解一个深度学习网络最重要的…

[yolov11改进系列]基于yolov11引入自注意力与卷积混合模块ACmix提高FPS+检测效率python源码+训练源码

[ACmix的框架原理] 1.1 ACMix的基本原理 ACmix是一种混合模型&#xff0c;结合了自注意力机制和卷积运算的优势。它的核心思想是&#xff0c;传统卷积操作和自注意力模块的大部分计算都可以通过1x1的卷积来实现。ACmix首先使用1x1卷积对输入特征图进行投影&#xff0c;生成一组…

[DS]使用 Python 库中自带的数据集来实现上述 50 个数据分析和数据可视化程序的示例代码

使用 Python 库中自带的数据集来实现上述 50 个数据分析和数据可视化程序的示例代码 摘要&#xff1a;由于 sample_data.csv 是一个占位符文件&#xff0c;用于代表任意数据集&#xff0c;我将使用 Python 库中自带的数据集来实现上述 50 个数据分析和数据可视化程序的示例代码…

【Python 中 lambda、map、filter 和 reduce】详细功能介绍及用法总结

以下是 Python 中 lambda、map、filter 和 reduce 的详细功能介绍及用法总结&#xff0c;涵盖基础语法、高频场景和示例代码。 一、lambda 匿名函数 功能 用于快速定义一次性使用的匿名函数。不需要显式命名&#xff0c;适合简化小规模逻辑。 语法 lambda 参数1, 参数2, ..…

贪心算法——分数背包问题

一、背景介绍 给定&#x1d45b;个物品&#xff0c;第&#x1d456;个物品的重量为&#x1d464;&#x1d454;&#x1d461;[&#x1d456;−1]、价值为&#x1d463;&#x1d44e;&#x1d459;[&#x1d456;−1]&#xff0c;和一个容量为&#x1d450;&#x1d44e;&#…

《软件工程》第 5 章 - 需求分析模型的表示

目录 5.1需求分析与验证 5.1.1 顺序图 5.1.2 通信图 5.1.3 状态图 5.1.4 扩充机制 5.2 需求分析的过程模型 5.3 需求优先级分析 5.3.1 确定需求项优先级 5.3.2 排定用例分析的优先顺序 5.4 用例分析 5.4.1 精化领域概念模型 5.4.2 设置分析类 5.4.3 构思分析类之间…