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

news2025/6/1 8:56:23

工厂模式三种类型:

一、简单工厂模式(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() Pay
	CreateRefund() 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() Button
    CreateCheckbox() Checkbox
}

// 具体工厂
type WindowsFactory struct{} // 创建Windows风格的Button和Checkbox
type MacFactory struct{}     // 创建Mac风格的Button和Checkbox

四、单例模式(Singleton Pattern)

定义:

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

使用场景:

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

特点:

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

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

package singleton

import (
	"sync"
)

type Singleton struct {
	Name string
}

var (
	instance *Singleton
	once     sync.Once
)

func GetInstance() *Singleton {
	once.Do(func() {
		instance = &Singleton{Name: "唯一实例"}
	})
	return instance
}

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

饿汉

package singleton

type 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 proxy

import "fmt"

type Downloader interface {
	Download(url string)
}

// 真实对象
type RealDownloader struct{}

func (r *RealDownloader) Download(url string) {
	fmt.Println("正在下载:", url)
}

// 代理对象
type ProxyDownloader struct {
	real     *RealDownloader
	cacheMap 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.coloradmin.cn/o/2392549.html

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

相关文章

Linux_编辑器Vim基本使用

✨✨ 欢迎大家来到小伞的大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:LInux_st 小伞的主页:xiaosan_blog 制作不易!点个赞吧!!谢谢喵!&a…

vue展示修改前后对比,并显示修改标注diff

动态父组件 <template><el-buttontype"primary"size"small"plainclick"showDiffDialog(subItem)">查看修改内容</el-button><TextDiffDialogv-model:visible"diffDialogVisible":before"currentDiffItem?.…

LiveWallpaperMacOS:让你的 Mac 桌面动起来

随着桌面美化需求的不断提升,用户对于桌面壁纸的要求已经不再局限于静态图片。越来越多的 Mac 用户希望桌面能像 Windows 一样,拥有动态壁纸,展现个性、提升体验。LiveWallpaperMacOS 正是这样一款让你的 Mac 桌面焕发活力的开源项目。 本文将详细介绍 LiveWallpaperMacOS …

[预训练]Encoder-only架构的预训练任务核心机制

原创文章1FFN前馈网络与激活函数技术解析&#xff1a;Transformer模型中的关键模块2Transformer掩码技术全解析&#xff1a;分类、原理与应用场景3【大模型技术】Attention注意力机制详解一4Transformer核心技术解析LCPO方法&#xff1a;精准控制推理长度的新突破5Transformer模…

07-后端Web实战(部门管理)

5. 修改部门 对于任何业务的修改功能来说&#xff0c;一般都会分为两步进行&#xff1a;查询回显、修改数据。 5.1 查询回显 5.1.1 需求 当我们点击 "编辑" 的时候&#xff0c;需要根据ID查询部门数据&#xff0c;然后用于页面回显展示。 5.1.2 接口描述 参照参照…

mysql ACID 原理

序言&#xff1a;ACID 是一组数据库设计原则&#xff0c;他是业务数据和关键业务程序的可靠性保障。 1、atomicity&#xff08;原子性&#xff09; 依赖如下能力 autocommit commit rollback2、一致性 2.1 double write buffer 1、定义&#xff1a;double write buffer 是…

[Rust_1] 环境配置 | vs golang | 程序运行 | 包管理

目录 Rust 环境安装 GoLang和Rust 关于Go 关于Rust Rust vs. Go&#xff0c;优缺点 GoLang的优点 GoLang的缺点 Rust的优点 Rust的缺点 数据告诉我们什么&#xff1f; Rust和Go的主要区别 (1) 性能 (2) 并发性 (3) 内存安全性 (4) 开发速度 (5) 开发者体验 Ru…

二十五、面向对象底层逻辑-SpringMVC九大组件之HandlerMapping接口设计

一、引言&#xff1a;MVC架构的交通枢纽 在Spring MVC框架中&#xff0c;HandlerMapping接口扮演着"请求导航仪"的关键角色&#xff0c;它决定了HTTP请求如何被路由到对应的Controller处理器。作为MVC模式的核心组件之一&#xff0c;HandlerMapping在请求处理的生命…

HUAWEI交换机配置镜像口验证(eNSP)

技术术语&#xff1a; 流量观察口&#xff1a;就是我们常说的镜像口&#xff0c;被观察的流量的引流目的端口 流量源端口&#xff1a;企业生产端口&#xff0c;作为观察口观察对象。 命令介绍&#xff1a; [核心交换机]observe-port [观察端口ID或编号&#xff08;数字&am…

前端vue3实现图片懒加载

场景和指令用法 场景:电商网站的首页通常会很长&#xff0c;用户不一定能访问到页面靠下面的图片&#xff0c;这类图片通过懒加载优化手段可以做到只有进入视口区域才发送图片请求 核心原理:图片进入视口才发送资源请求 首先&#xff1a;我们需要定义一个全局的指令&#x…

计算机网络-MPLS VPN应用场景与组网

上一篇文章我们通过一个基础实验实现了企业分支间的MPLS VPN互联&#xff0c;如果还不理解的可以多看几遍前面的文章或者多敲下实验。今天来学习几种常见的MPLS VPN应用场景与这些场景下MPLS VPN的部署方法。 一、MPLS VPN典型应用 目前&#xff0c;MPLS VPN的主要应用包括企…

Linux 的编辑器--vim

1.Linux编辑器-vim使⽤ vi/vim的区别简单点来说&#xff0c;它们都是多模式编辑器&#xff0c;不同的是vim是vi的升级版本&#xff0c;它不仅兼容vi的所有指令&#xff0c;⽽且还有⼀些新的特性在⾥⾯。例如语法加亮&#xff0c;可视化操作不仅可以在终端运⾏&#xff0c;也可以…

[Protobuf] 快速上手:安全高效的序列化指南

标题&#xff1a;[Protobuf] (1)快速上手 水墨不写bug 文章目录 一、什么是protobuf&#xff1f;二、protobuf的特点三、使用protobuf的过程&#xff1f;1、定义消息格式&#xff08;.proto文件&#xff09;(1)指定语法版本(2)package 声明符 2、使用protoc编译器生成代码&…

如何将通话记录从Android传输到Android

“如何将通话记录从 Android 转移到 Android&#xff1f;我换了一部新的 Android 手机&#xff0c;想要将通话记录复制到其中。”您需要将通话记录从 Android 传输到 Android 是一种常见的情况&#xff0c;因为通话记录是手机上最重要的数据之一。幸运的是&#xff0c;如果您从…

数据结构第4章 栈、队列和数组 (竟成)

目录 第 4 章 栈、队列和数组 4.1 栈 4.1.1 栈的基本概念 4.1.2 栈的基本操作 4.1.3 栈的实现 1.顺序栈 2.链式栈 3.共享栈 4.1.4 顺序栈的基本操作实现 1.初始化栈 2.判空 3.判满 4.元素进栈 5.元素出栈 6.获取栈顶元素 4.1.5 链栈的基本操作实现 1.元素进栈 2.元素出栈 4.1.6…

2025年渗透测试面试题总结-匿名[校招]安全研究员(SAST方向)(题目+回答)

安全领域各种资源&#xff0c;学习文档&#xff0c;以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具&#xff0c;欢迎关注。 目录 匿名[校招]安全研究员(SAST方向) 一面问题回答框架 1. 自我介绍 2. 简历深挖&#xff08;漏洞挖掘&#x…

Unity 游戏优化(持续更新中...)

垃圾回收 是什么&#xff1f; 垃圾回收&#xff08;Garbage Collection&#xff09;GC 工作机制 1、Unity 为用户生成的代码和脚本采用了自动内存管理。 2、小块数据&#xff08;如值类型的局部变量&#xff09;分配在栈上。大块数据和长期存储分配在托管堆上。 3、垃圾收集…

20250529-C#知识:索引器

C#知识&#xff1a;索引器 索引器给对象添加了索引访问的功能&#xff0c;实际访问的是对象的成员&#xff0c;感觉不太常用。 1、主要内容及代码示例 索引器中类似属性&#xff0c;也包含get和set方法索引器能够使像访问数组一样访问对象一般当类中有数组类型的成员变量时&am…

【笔记】suna部署之获取 Tavily API key

#工作记录 Tavily 注册 Tavily 账号5&#xff1a; 打开浏览器&#xff0c;访问 Tavily 官网Tavily AI。点击页面上的 “注册” 按钮&#xff0c;按照提示填写注册信息&#xff0c;如邮箱地址、设置密码等&#xff0c;完成注册流程。也可以选择使用 Google 或 GitHub 账号授权登…

06-Web后端基础(java操作数据库)

1. 前言 在前面我们学习MySQL数据库时&#xff0c;都是利用图形化客户端工具(如&#xff1a;idea、datagrip)&#xff0c;来操作数据库的。 我们做为后端程序开发人员&#xff0c;通常会使用Java程序来完成对数据库的操作。Java程序操作数据库的技术呢&#xff0c;有很多啊&a…