建造者模式

GitHub代码链接
建造者模式(Builder Pattern)使用多个简单的对象一步一步构建成一个复杂的对象。

什么是建造者模式

一个Builder类会一步一步构造最终的对象,该Builder类是独立于其他对象的。

解决的问题

主要解决在软件系统中,有时候面临一个复杂对象的创建工作,通常这个复杂对象由各个部分的子对象用一定的算法构建成。由于需求的变化,这个复杂对象的各个部分通常会出现巨大的变化,所以,将各个子对象独立出来,容易修改。
例如肯德基的点餐系统,汉堡,薯条,可乐,炸鸡是不变的,变化的是他们组合出的套餐,所以点餐系统使用建造者模式,非常容易拓展出不同的套餐,而且既定的套餐,修改薯条为大份的时候,也非常方便。

优点

  • 将一个系统中的变与不变分离,容易拓展
  • 便于控制细节风险

缺点

  • 产品必须有共同特点,范围有限
  • 如果子类变化复杂,会有很多建造类

代码实现

我们以肯德基的点餐系统为例,实现建造者模式。

1.1 创建所有商品的接口

//Item 所有商品的接口
type Item interface {
    Name() string
    Price() float32
}

1.2 实现食物的接口和类

//Food 食物的接口
type Food interface {
    Kind() string
}

//Staple 主食,食物接口的实例
type Staple struct{}

//Drink 饮料,食物接口的实例
type Drink struct{}

//Snack 小吃,食物接口的实例
type Snack struct{}

//Kind 获取主食类型名称
func (staple Staple) Kind() string {
    return "staple"
}

//Kind 获取饮料类型名称
func (drink Drink) Kind() string {
    return "drink"
}

//Kind 获取小吃类型名称
func (snack Snack) Kind() string {
    return "snack"
}

1.3 实现可口可乐商品

这里我们的饮料继承自1.2中的Drink类,但是Golang没有继承这个概念,我们使用组合来实现继承的效果。

//CocaCola 可口可乐商品类,实现Item接口,并且组合了Drink基类
type CocaCola struct {
    Drink
}

//NewCocaCola 实例化可口可乐
func NewCocaCola() *CocaCola {
    return &CocaCola{}
}

//Name 获取可口可乐商品名
func (co *CocaCola) Name() string {
    return "coca-cola"
}

//Price 获取可口可乐价格
func (co *CocaCola) Price() float32 {
    return 2.5
}

1.4 实现薯条商品类

同样的,薯条类组合了1.2中的Snack类

//Chips 薯条商品类,实现Item接口,组合Snack基类
type Chips struct {
    Snack
}

//NewChips 实例化薯条类
func NewChips() *Chips {
    return &Chips{}
}

//Name 获取薯条的名称
func (ch *Chips) Name() string {
    return "chips"
}

//Price 获取薯条价格
func (ch *Chips) Price() float32 {
    return 9.9
}

1.5 实现汉堡商品类

牛肉汉堡组合了1.2中的Staple类

//BeefBurger 牛肉汉堡,实现Item接口,组合Staple基类
type BeefBurger struct {
    Staple
}

//NewBeefBurger 实例化牛肉汉堡
func NewBeefBurger() *BeefBurger {
    return &BeefBurger{}
}

//Name 获取牛肉汉堡名称
func (bf *BeefBurger) Name() string {
    return "beefBurger"
}

//Price 获取牛肉汉堡价格
func (bf *BeefBurger) Price() float32 {
    return 19.8
}

1.6 实现点单机器

//Meal 点单类
type Meal struct {
    Items *list.List
}

//NewMeal 实例化点单类
func NewMeal() *Meal {
    l := list.New()
    return &Meal{l}
}

//AddItem 添加新的商品
func (m *Meal) AddItem(item Item) {
    m.Items.PushBack(item)
}

//GetCost 获取当前订单总额
func (m *Meal) GetCost() float32 {
    var cost float32 = 0
    for i := m.Items.Front(); i != nil; i = i.Next() {
        item, ok := (i.Value).(Item)
        if ok {
            cost += item.Price()
        }
    }
    return cost
}

//ShowItems 显示当前所有餐品
func (m *Meal) ShowItems() {
    for i := m.Items.Front(); i != nil; i = i.Next() {
        item, ok := (i.Value).(Item)
        if ok {
            fmt.Println(item.Name(), ": ", item.Price(), "元")
        }
    }
}
上一篇 下一篇