• 设计模式之工厂模式


    工厂模式是一种创建型设计模式,它提供了一个用于创建对象的接口,但允许子类决定实例化哪个类。工厂方法让一个类的实例化延迟到其子类。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

    在软件开发中,对象的创建和使用是常见的操作。然而,对象的创建过程常常会涉及到复杂的逻辑和多变的需求。为了解决这个问题,我们可以使用工厂模式来将对象的创建和使用分离,提高代码的可维护性和灵活性。

    1. 工厂模式的定义

    工厂模式是一种创建型设计模式,它定义了一个用于创建对象的接口,但允许子类决定实例化哪个类。工厂方法让一个类的实例化延迟到其子类。

    2. 工厂模式的结构

    工厂模式主要包括以下几个角色:

    1. 抽象产品(Abstract Product):定义了产品的通用方法,是所有具体产品的公共父类。
    2. 具体产品(Concrete Product):实现了抽象产品定义的通用方法,是具体的产品实现。
    3. 抽象工厂(Abstract Factory):定义了创建产品的接口,是所有具体工厂的公共父类。
    4. 具体工厂(Concrete Factory):实现了抽象工厂定义的创建产品的方法,用于创建具体的产品实例。

    3. 工厂模式的优点

    1. 封装变化:工厂模式将对象的创建过程封装起来,使得调用者无需关心产品的实例化过程,只需依赖工厂即可得到自己想要的产品。这样可以有效地封装变化,使得代码更加灵活和可维护。
    2. 代码结构清晰:工厂模式将对象的创建和使用分离,使得代码结构更加清晰。调用者只需要知道产品的接口即可,无需关心具体的实现细节。
    3. 扩展性:工厂模式使得添加新产品变得非常容易。只需要添加新的具体产品类和相应的具体工厂类即可,无需修改原有的代码结构。

    4. 工厂模式的分类

    工厂模式可以分为三类:简单工厂模式、工厂方法模式和抽象工厂模式。简单工厂模式不是一个标准的设计模式,它可以看为工厂方法模式的一种特例。工厂方法模式和抽象工厂模式都是标准的设计模式。这三种工厂模式在设计模式的分类中都属于创建型模式,三种模式从上到下逐步抽象。

    1. 简单工厂模式:简单工厂模式是最简单的工厂模式,它通过一个静态方法来创建对象。这种方法虽然简单,但是不够灵活,无法满足复杂多变的需求。
    2. 工厂方法模式:工厂方法模式是标准的设计模式之一,它定义了一个用于创建对象的接口,但是让子类决定实例化哪个类。这样可以实现更加灵活的创建过程,满足不同的需求。
    3. 抽象工厂模式:抽象工厂模式是最高级的工厂模式,它定义了一个用于创建一系列相关或互相依赖的对象的接口,而不需要指定它们具体的类。这样可以实现更加复杂的创建过程,满足更加多变的需求。

    5. 使用工厂模式的注意事项

    在使用工厂模式时,需要注意以下几点:

    1. 工厂模式增加了系统的复杂度。在添加新产品时,需要编写新的具体产品类,同时还要对工厂类进行修改,增加了系统的复杂度。因此,需要考虑产品类的数量和生命周期等因素来决定是否使用工厂模式。
    2. 需要考虑产品类的数量。如果产品类的数量较少,那么使用工厂模式可能会过于复杂。此时可以考虑使用其他的设计模式来解决问题。
    3. 需要考虑产品类的生命周期。如果产品类的生命周期很长,那么使用工厂模式可能会增加系统的复杂度。此时可以考虑使用单例模式等其他的设计模式来解决问题。

    6. 示例

    下面是一个使用工厂方法模式的示例,以创建不同类型的电视机产品:

    package main
    import "fmt"
    // TV 接口定义了电视机产品的方法
    type TV interface {
    TurnOn()
    TurnOff()
    }
    // HaixinTV 具体产品类,实现了 TV 接口
    type HaixinTV struct{}
    func (tv *HaixinTV) TurnOn() {
    fmt.Println("Haixin TV is on")
    }
    func (tv *HaixinTV) TurnOff() {
    fmt.Println("Haixin TV is off")
    }
    // HaierTV 具体产品类,实现了 TV 接口
    type HaierTV struct{}
    func (tv *HaierTV) TurnOn() {
    fmt.Println("Haier TV is on")
    }
    func (tv *HaierTV) TurnOff() {
    fmt.Println("Haier TV is off")
    }
    // TVFactory 工厂接口定义了创建 TV 对象的方法
    type TVFactory interface {
    CreateTV() TV
    }
    // HaixinFactory 具体工厂类,用于创建 HaixinTV 产品
    type HaixinFactory struct{}
    func (f *HaixinFactory) CreateTV() TV {
    return &HaixinTV{}
    }
    // HaierFactory 具体工厂类,用于创建 HaierTV 产品
    type HaierFactory struct{}
    func (f *HaierFactory) CreateTV() TV {
    return &HaierTV{}
    }
    func main() {
    // 创建 Haixin 电视机工厂
    HaixinFactory := &HaixinFactory{}
    HaixinTV := HaixinFactory.CreateTV()
    HaixinTV.TurnOn()
    HaixinTV.TurnOff()
    // 创建 Haier 电视机工厂
    HaierFactory := &HaierFactory{}
    HaierTV := HaierFactory.CreateTV()
    HaierTV.TurnOn()
    HaierTV.TurnOff()
    }

    孟斯特

    声明:本作品采用署名-非商业性使用-相同方式共享 4.0 国际 (CC BY-NC-SA 4.0)进行许可,使用时请注明出处。
    Author: mengbin
    blog: mengbin
    Github: mengbin92
    cnblogs: 恋水无意


  • 相关阅读:
    笔者认为所谓的产业互联网,就是一个产业与互联网深度融合的过程
    短视频创业如何开始,有哪些需要注意的问题?
    CAS原理与JUC原子类详解
    力扣练习——48 找到小镇的法官
    day52|300.最长递增子序列、674. 最长连续递增序列、718. 最长重复子数组
    flink中使用异步函数的几个注意事项
    《哥德尔、艾舍尔、巴赫——集异璧之大成》阅读笔记1
    深入了解 useMemo 和 useCallback
    MyBatis学习:使用resultMap或在SQL语句中给列起别名处理查询结果中列名和JAVA对象属性名不一致的问题
    Matlab训练BP神经网络的一般步骤
  • 原文地址:https://www.cnblogs.com/lianshuiwuyi/p/17744626.html