前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >设计模式 -- 工厂方法模式

设计模式 -- 工厂方法模式

作者头像
xy_ss
发布2023-11-22 09:22:31
1210
发布2023-11-22 09:22:31
举报
文章被收录于专栏:浮躁的喧嚣

表述 (创建型模式)

定义创建对象的接口,让子类决定实例化哪一个类。工厂方法使得一个类的实例化延迟到其子类

工厂方法模式和简单工厂十分类似,大致结构是基本类似的。不同在于工厂方法模式对工厂类进行了进一步的抽象,将之前的一个工厂类抽象成了抽象工厂和工厂子类,抽象工厂定义一个创建抽象子类的接口,抽象工厂的子类实现这些接口并决定实例化哪个抽象子类。工厂子类决定着创建哪个抽象子类,外界决定着创建哪种工厂子类,抽象子类和工厂子类是一一对应的

工厂方法模式类图

工厂方法模式类图

Creator(工厂抽象类):定义创建抽象子类的接口,通过接口返回具体的抽象子类 ConcreteCreator(工厂子类):继承自工厂抽象类,并重写父类的方法来创建对应的抽象子类 Product(抽象类):定义抽象子类所需的属性和方法,子类通过继承自抽象类获取这些方法 ConcreteProduct(抽象子类):继承自抽象类,是具体操作的实现者

优点

  • 创建一个具体产品的细节完全封装在具体工厂内部,符合高内聚,低耦合
  • 扩展性好,当需要扩展时,在代码中新建相应的工厂和产品类即可,前台代码无需更改

使用场景

  • 编译时无法准确预期要创建的对象的类
  • 类想让其子类决定在运行时创建什么
  • 类有若干辅助类为其子类,而你想将返回哪个子类这一信息局部化

示例

需求V1:实现俩个数的加、减运算

代码语言:javascript
复制
//创建一个抽象类,抽象类中包含了参与运算的抽象子类的属性和方法
class Operation {
    private var _numA = 0
    private var _numB = 0
    
    var numA: Int{
        get{
            return _numA
        }
        set{
            _numA = newValue
        }
    }
    var numB: Int {
        get{
            return _numB
        }
        set{
            _numB = newValue
        }
    }
    func getResult() -> Int {
        return 0
    }
}

//创建负责运算的抽象子类
class OperationAdd:Operation {
    override func getResult() -> Int {
        return self.numA + self.numB
    }
}
//创建负责运算的抽象子类
class OperationSub:Operation {
    override func getResult() -> Int {
        return self.numA - self.numB
    }
}

//创建一个抽象工厂类,定义了实例化实际运算的方法
class CreateOperationFactory {
    func createOperationFactory() -> Operation {
        return Operation()
    }
}
//由子类继承并实例化不同的运算类
class OperationAddFactory : CreateOperationFactory {
    override func createOperationFactory() -> Operation {
        return OperationAdd()
    }
}
//由子类继承并实例化不同的运算类
class OperationSubFactory : CreateOperationFactory {
    override func createOperationFactory() -> Operation {
        return OperationSub()
    }
}

//直接实例化某个工厂子类,通过外界实例化某个工厂子类来选择具体的运算
let operAddFac = OperationAddFactory()
let operAdd = operAddFac.createOperationFactory()
operAdd.numA = 8
operAdd.numB = 5
let resAdd = operAdd.getResult()
print(resAdd)

let operSubFac = OperationSubFactory()
let operSub = operSubFac.createOperationFactory()
operSub.numA = 8
operSub.numB = 5
let resSub = operSub.getResult()
print(resSub)

需求V2:要在原有基础上添加乘法(对应新增的抽象子类创建对应的工厂子类即可)

代码语言:javascript
复制
class operationMul:Operation {
    override func getResult() -> Int {
        return self.numA * self.numB
    }
}

class OperationMulFactory : CreateOperationFactory {
    override func createOperationFactory() -> Operation {
        return operationMul()
    }
}

let operMulFac = OperationMulFactory()
let operMul = operMulFac.createOperationFactory()
operMul.numA = 8
operMul.numB = 5
let resMul = operMul.getResult()
print(resMul)

简单工厂模式违背了开放封闭原则,每次添加和删除抽象子类的时候,都需要对工厂类进行操作,这样不仅对工厂类的扩展开放了,还开放了工厂类的修改,这就是违背开放封闭原则的。因为按照开放封闭原则,新增加一个需求,应该是在原有类的基础上进行扩展,而不是对原有类进行修改。这样整个模式在生成新算法类时,只是进行扩展而不对模式中原有的代码进行修改,这就是符合开放封闭原则的

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2019-08-23,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 表述 (创建型模式)
  • 工厂方法模式类图
  • 优点
  • 使用场景
  • 示例
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档