美文网首页
Go语言设计模式(3)抽象工厂模式

Go语言设计模式(3)抽象工厂模式

作者: NoelleMu | 来源:发表于2021-09-19 15:34 被阅读0次

    Go语言设计模式(3)抽象工厂模式

    抽象工厂模式的定义

    抽象工厂模式的定义如下:

    Provide an interface for creating families of related or dependent objects without specifying their concrete classes.

    为创建一组相关或者相互依赖的对象提供一个接口,而且无须指定它们的具体类。

    ——《设计模式之禅》

    好吧,就跟它的名字一样,它的定义也很抽象,很让人摸不着头脑。但其实这种设计模式还是很好理解的,只要明白了它和工厂模式有什么区别,再看一些实例代码,理解这种设计模式并不是难事。我们接下来就来具体讲讲它。

    抽象工厂模式和工厂模式的区别

    首先考虑一个问题:抽象工厂模式和工厂模式的区别在哪?

    抽象工厂模式最大的特征就是定义中提到的“相关或相互依赖的对象”,通俗点说,就是抽象工厂模式相比工厂模式多了两个概念:“产品族”和“等级结构”。工厂模式中的每个工厂只能生产一种产品,而抽象工厂模式将某些相关、等级结构不同的产品组成了一个“产品族”,然后用一个工厂来统一生产。

    还是觉得很难理解?好的,Show you code!

    抽象工厂模式的简单例子

    这里我没有使用《设计模式之禅》的例子,因为它比较复杂,我们来举个更简单更好理解的例子:

    我们可以把Web应用简单地看做是由前端(客户端)和后端(服务端)组成的,不知道在看我的文章的同学中有没有知道“前端架构师”这个职业的同学,我们一直以为做后端开发更容易成为架构师,但是当前端变得越来越复杂时,前端架构师这种职位也就产生了。所以,在这个例子中,我们把架构师分为“前端架构师”和“后端架构师”两类。当然,要开发一个完整的Web应用,光有架构师还不行,还得有程序员来把架构师给出的设计用代码实现,而我们都知道程序员有做前端开发的,也有做后端开发的,这样也可以把程序员分为“前端程序员”和“后端程序员”两类。

    在这个场景中,我们有四种产品:前端程序员、前端架构师、后端程序员、后端架构师。我们可以把“前端”和“后端”分为两个不同的等级结构,我们可以为这两个等级结构创建两个工厂类FrontEndFactoryBackEndFactory,我们同时也可以把“程序员”和“架构师”分为另外两个不同的产品族,在每个工厂类里添加CreateProgrammer()CreateArchitect()两个方法用于创建程序员和架构师的实例——也就是说,有多少个产品族,在抽象工厂类里就有多少个创建方法

    根据这个场景我们可以画出下面这张UML类图(稍微有点不太标准,不过能通过这张图把上面描述的场景搞懂就OK了):

    image

    接下来用代码实现它:

    abstract_factory/programmer_factory.go

    <pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="go" cid="n39" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: var(--font-monospace); font-size: 0.85rem; display: block; break-inside: avoid; text-align: left; white-space: normal; background-image: inherit; background-size: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(248, 248, 248) !important; position: relative !important; width: inherit; border: 1px solid rgb(244, 244, 244); -webkit-font-smoothing: initial; line-height: 1.43rem; border-top-left-radius: 2px; border-top-right-radius: 2px; border-bottom-right-radius: 2px; border-bottom-left-radius: 2px; word-wrap: normal; margin: 0.8rem 0px !important; padding: 0.3rem 0px !important; caret-color: rgb(52, 73, 94); color: rgb(52, 73, 94); font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-indent: 0px; text-transform: none; widows: auto; word-spacing: 0px; -webkit-text-size-adjust: auto; -webkit-text-stroke-width: 0px; text-decoration: none; background-position: inherit; background-repeat: inherit;">package abstract_factory

    import "fmt"

    // Programmer 程序员总称
    type Programmer interface {
    Work() // 程序员都会工作
    }

    // Architect 架构师总称
    type Architect interface {
    Design() // 架构师都会做架构设计
    }

    // FrontEndArchitect 前端架构师
    type FrontEndArchitect struct{}

    func (a *FrontEndArchitect) Design() {
    fmt.Println("前端架构师做了一个页面秒开的设计")
    }

    // FrontEndProgrammer 前端程序员
    type FrontEndProgrammer struct{}

    func (p *FrontEndProgrammer) Work() {
    fmt.Println("前端程序员在用WebStorm写TypeScript代码")
    }

    // BackEndArchitect 后端架构师
    type BackEndArchitect struct{}

    func (a *BackEndArchitect) Design() {
    fmt.Println("后端架构师做了一个可以抗住上万并发的设计")
    }

    // BackEndProgrammer 后端程序员
    type BackEndProgrammer struct{}

    func (p *BackEndProgrammer) Work() {
    fmt.Println("后端程序员在用GoLand写Golang代码")
    }

    // AbstractFactory 抽象工厂
    type AbstractFactory interface {
    CreateProgrammer() Programmer // 创建程序员
    CreateArchitect() Architect // 创建架构师
    }

    // FrontEndFactory 前端工厂
    type FrontEndFactory struct{}

    func (f *FrontEndFactory) CreateProgrammer() Programmer {
    return &FrontEndProgrammer{}
    }

    func (f *FrontEndFactory) CreateArchitect() Architect {
    return &FrontEndArchitect{}
    }

    // BackEndFactory 后端工厂
    type BackEndFactory struct{}

    func (f *BackEndFactory) CreateProgrammer() Programmer {
    return &BackEndProgrammer{}
    }

    func (f *BackEndFactory) CreateArchitect() Architect {
    return &BackEndArchitect{}
    }
    </pre>

    代码比较长,可以对照着类图来看。

    写个场景类(测试)来调用一下:

    abstract_factory/programmer_factory_test.go

    <pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="go" cid="n48" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: var(--font-monospace); font-size: 0.85rem; display: block; break-inside: avoid; text-align: left; white-space: normal; background-image: inherit; background-size: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(248, 248, 248) !important; position: relative !important; width: inherit; border: 1px solid rgb(244, 244, 244); -webkit-font-smoothing: initial; line-height: 1.43rem; border-top-left-radius: 2px; border-top-right-radius: 2px; border-bottom-right-radius: 2px; border-bottom-left-radius: 2px; word-wrap: normal; margin: 0.8rem 0px !important; padding: 0.3rem 0px !important; caret-color: rgb(52, 73, 94); color: rgb(52, 73, 94); font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-indent: 0px; text-transform: none; widows: auto; word-spacing: 0px; -webkit-text-size-adjust: auto; -webkit-text-stroke-width: 0px; text-decoration: none; background-position: inherit; background-repeat: inherit;">package abstract_factory

    import (
    "fmt"
    "testing"
    )

    func TestCreateProgrammerAndArchitect(t *testing.T) {
    fmt.Println("前端组招到了一个程序员和一个架构师")
    ff := FrontEndFactory{}
    fa := ff.CreateArchitect()
    fp := ff.CreateProgrammer()
    fmt.Println("前端组接到任务,开始工作...")
    fa.Design()
    fp.Work()
    fmt.Println("后端组招到了一个程序员和一个架构师")
    bf := BackEndFactory{}
    ba := bf.CreateArchitect()
    bp := bf.CreateProgrammer()
    fmt.Println("后端组接到任务,开始工作...")
    ba.Design()
    bp.Work()
    }
    </pre>

    运行结果:

    <pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="" cid="n53" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: var(--font-monospace); font-size: 0.85rem; display: block; break-inside: avoid; text-align: left; white-space: normal; background-image: inherit; background-size: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(248, 248, 248) !important; position: relative !important; width: inherit; border: 1px solid rgb(244, 244, 244); -webkit-font-smoothing: initial; line-height: 1.43rem; border-top-left-radius: 2px; border-top-right-radius: 2px; border-bottom-right-radius: 2px; border-bottom-left-radius: 2px; word-wrap: normal; margin: 0.8rem 0px !important; padding: 0.3rem 0px !important; caret-color: rgb(52, 73, 94); color: rgb(52, 73, 94); font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-indent: 0px; text-transform: none; widows: auto; word-spacing: 0px; -webkit-text-size-adjust: auto; -webkit-text-stroke-width: 0px; text-decoration: none; background-position: inherit; background-repeat: inherit;">=== RUN TestCreateProgrammerAndArchitect
    前端组招到了一个程序员和一个架构师
    前端组接到任务,开始工作...
    前端架构师做了一个页面秒开的设计
    前端程序员在用WebStorm写TypeScript代码
    后端组招到了一个程序员和一个架构师
    后端组接到任务,开始工作...
    后端架构师做了一个可以抗住上万并发的设计
    后端程序员在用GoLand写Golang代码
    --- PASS: TestCreateProgrammerAndArchitect (0.00s)
    PASS</pre>

    看完这个例子之后,是不是彻底明白抽象工厂模式是怎么一回事了呢。

    我们也可以看出,在抽象工厂模式中,添加等级结构很方便——假如我们要添加一个新的等级结构“基础架构(BasicArchitecture)”,那么直接新建对应的程序员类BasicArchitectureProgrammer和架构师类BasicArchitectureArchitect继承ProgrammerArchitect,再新建一个工厂类BasicArchitectureFactory继承AbstractFactory就可以完成扩展。

    但是,我们也很容易发现添加产品族很难,且违背了开闭原则——假如我们要添加一个新的工种“测试工程师(TestEngineer)”,则需要建立新的抽象类TestEngineer(成员方法可以叫Test()),并且在所有的工厂类里面都添加、实现CreateTestEngineer()用于创建不同等级结构(前端、后端)的测试工程师,这样才可以完成扩展。这种方法显而易见地违背了开闭原则,增大了维护难度,所以在使用抽象工厂模式时最好一开始就把所有的产品族设计好,尽量减少对产品族的添加和删除。

    抽象工厂模式的通用代码

    abstract_factory/abstract_factory.go

    <pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="go" cid="n66" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: var(--font-monospace); font-size: 0.85rem; display: block; break-inside: avoid; text-align: left; white-space: normal; background-image: inherit; background-size: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(248, 248, 248) !important; position: relative !important; width: inherit; border: 1px solid rgb(244, 244, 244); -webkit-font-smoothing: initial; line-height: 1.43rem; border-top-left-radius: 2px; border-top-right-radius: 2px; border-bottom-right-radius: 2px; border-bottom-left-radius: 2px; word-wrap: normal; margin: 0.8rem 0px !important; padding: 0.3rem 0px !important; caret-color: rgb(52, 73, 94); color: rgb(52, 73, 94); font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-indent: 0px; text-transform: none; widows: auto; word-spacing: 0px; -webkit-text-size-adjust: auto; -webkit-text-stroke-width: 0px; text-decoration: none; background-position: inherit; background-repeat: inherit;">package abstract_factory

    import "fmt"

    // AbstractProductA 抽象产品类A
    type AbstractProductA interface {
    DoSomethingA()
    }

    // AbstractProductB 抽象产品类B
    type AbstractProductB interface {
    DoSomethingB()
    }

    // ProductA1 产品A1的实现类
    type ProductA1 struct{}

    func (p *ProductA1) DoSomethingA() {
    fmt.Println("产品A1的实现方法")
    }

    // ProductA2 产品A2的实现类
    type ProductA2 struct{}

    func (p *ProductA2) DoSomethingA() {
    fmt.Println("产品A2的实现方法")
    }

    // ProductB1 产品B1的实现类
    type ProductB1 struct{}

    func (p *ProductB1) DoSomethingB() {
    fmt.Println("产品B1的实现方法")
    }

    // ProductB2 产品B2的实现类
    type ProductB2 struct{}

    func (p *ProductB2) DoSomethingB() {
    fmt.Println("产品B2的实现方法")
    }

    // AbstractCreator 抽象工厂类
    type AbstractCreator interface {
    CreateProductA() AbstractProductA
    CreateProductB() AbstractProductB
    }

    // Creator1 产品等级1的工厂实现类
    type Creator1 struct{}

    func (c *Creator1) CreateProductA() AbstractProductA {
    return &ProductA1{}
    }

    func (c *Creator1) CreateProductB() AbstractProductB {
    return &ProductB1{}
    }

    // Creator2 产品等级2的工厂实现类
    type Creator2 struct{}

    func (c *Creator2) CreateProductA() AbstractProductA {
    return &ProductA2{}
    }

    func (c *Creator2) CreateProductB() AbstractProductB {
    return &ProductB2{}
    }
    </pre>

    abstract_factory/abstract_factory_test.go

    <pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="go" cid="n71" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: var(--font-monospace); font-size: 0.85rem; display: block; break-inside: avoid; text-align: left; white-space: normal; background-image: inherit; background-size: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(248, 248, 248) !important; position: relative !important; width: inherit; border: 1px solid rgb(244, 244, 244); -webkit-font-smoothing: initial; line-height: 1.43rem; border-top-left-radius: 2px; border-top-right-radius: 2px; border-bottom-right-radius: 2px; border-bottom-left-radius: 2px; word-wrap: normal; margin: 0.8rem 0px !important; padding: 0.3rem 0px !important; caret-color: rgb(52, 73, 94); color: rgb(52, 73, 94); font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-indent: 0px; text-transform: none; widows: auto; word-spacing: 0px; -webkit-text-size-adjust: auto; -webkit-text-stroke-width: 0px; text-decoration: none; background-position: inherit; background-repeat: inherit;">package abstract_factory

    import "testing"

    func TestAbstractFactory(t *testing.T) {
    c1 := Creator1{}
    c2 := Creator2{}
    a1 := c1.CreateProductA()
    a2 := c2.CreateProductA()
    b1 := c1.CreateProductB()
    b2 := c2.CreateProductB()
    a1.DoSomethingA()
    a2.DoSomethingA()
    b1.DoSomethingB()
    b2.DoSomethingB()
    }
    </pre>

    运行结果:

    <pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="" cid="n76" mdtype="fences" style="box-sizing: border-box; overflow: visible; font-family: var(--font-monospace); font-size: 0.85rem; display: block; break-inside: avoid; text-align: left; white-space: normal; background-image: inherit; background-size: inherit; background-attachment: inherit; background-origin: inherit; background-clip: inherit; background-color: rgb(248, 248, 248) !important; position: relative !important; width: inherit; border: 1px solid rgb(244, 244, 244); -webkit-font-smoothing: initial; line-height: 1.43rem; border-top-left-radius: 2px; border-top-right-radius: 2px; border-bottom-right-radius: 2px; border-bottom-left-radius: 2px; word-wrap: normal; margin: 0.8rem 0px !important; padding: 0.3rem 0px !important; caret-color: rgb(52, 73, 94); color: rgb(52, 73, 94); font-style: normal; font-variant-caps: normal; font-weight: normal; letter-spacing: normal; orphans: auto; text-indent: 0px; text-transform: none; widows: auto; word-spacing: 0px; -webkit-text-size-adjust: auto; -webkit-text-stroke-width: 0px; text-decoration: none; background-position: inherit; background-repeat: inherit;">=== RUN TestAbstractFactory
    产品A1的实现方法
    产品A2的实现方法
    产品B1的实现方法
    产品B2的实现方法
    --- PASS: TestAbstractFactory (0.00s)
    PASS</pre>

    总结

    抽象工厂模式看似很难理解,实际上只要理解了“产品族”和“等级结构”这两个概念,抽象工厂模式就非常简单了。那么它有什么用呢?一个很典型的用途就是适配不同数据库——不同的数据库所提供的操作应该是相同的(增、删、改、查、事务等),但是每种数据库的底层实现又不一样,这时就可以使用抽象工厂模式来生成不同数据库的操作对象;另外,Java的AWT也运用了抽象工厂模式来实现不同操作系统下应用程序界面的统一。

    相关文章

      网友评论

          本文标题:Go语言设计模式(3)抽象工厂模式

          本文链接:https://www.haomeiwen.com/subject/mbxpgltx.html