GO设计模式——5、建造者模式(创建型)

目录

[建造者模式(Builder Pattern)](#建造者模式(Builder Pattern))

建造者模式的核心角色

优缺点

使用场景

注意事项

代码实现


建造者模式(Builder Pattern)

建造者模式(Builder Pattern)是将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。建造者模式是一步一步创建一个复杂的对象,它允许用户只通过指定复杂对象的类型和内容就可以构建它们,用户不需要知道内部的具体构建细节。

建造者模式的核心角色

1、抽象建造者(Builder):是为创建一个Product对象的各个部件指定的抽象接口。

2、具体建造者(ConcreteBuilder):实现Builder接口,构造和装配各个部件。

3、指挥者(Director):是构建一个实用Builder接口的对象。一方面它隔离了客户与生产过程;另一方面它负责控制产品的生成过程。

4、产品角色(Product):具体的产品

优缺点

(1)优点:

  • 分离构建过程和表示,使得构建过程更加灵活,可以构建不同的表示。
  • 可以更好地控制构建过程,隐藏具体构建细节。
  • 代码复用性高,可以在不同的构建过程中重复使用相同的建造者。

(2)缺点:

  • 如果产品的属性较少,建造者模式可能会导致代码冗余。
  • 建造者模式增加了系统的类和对象数量。

使用场景

建造者模式在创建复杂对象时非常有用,特别是当对象的构建过程涉及多个步骤或参数时。它可以提供更好的灵活性和可维护性,同时使得代码更加清晰可读。

  • 需要生成的对象具有复杂的内部结构。
  • 需要生成的对象内部属性本身相互依赖

注意事项

与工厂模式的区别是:建造者模式更加关注与零件装配的顺序。

代码实现

Go 复制代码
package main

import (
    "errors"
    "fmt"
)

// Product内的参数
type ResourceParams struct {
    name     string
    maxTotal int64
    maxIdle  int64
    minIdle  int64
}

// Product接口
type ResourceProduct interface {
    show()
}

// 实际Product,有show函数
type RedisResourceProduct struct {
    resourceParams ResourceParams
}

// show成员函数,用于显示product的参数内容
func (p *RedisResourceProduct) show() {
    fmt.Printf("Product的数据为 %+v ", p.resourceParams)
}

// 资源类创建接口
type ResourceBuilder interface {
    setName(name string) ResourceBuilder
    setMaxTotal(maxTotal int64) ResourceBuilder
    setMaxIdle(maxIdle int64) ResourceBuilder
    setMinIdle(minIdle int64) ResourceBuilder
    getError() error
    build() (p ResourceProduct)
}

// 实际建造者
type RedisResourceBuilder struct {
    resourceParams ResourceParams
    err            error
}

// 设置名称
func (r *RedisResourceBuilder) setName(name string) ResourceBuilder {
    if name == "" {
       r.err = errors.New("name为空")
       return r
    }
    r.resourceParams.name = name
    fmt.Println("RedisResourceBuilder setName ", name)
    return r
}

// 设置maxTotal值,值不能小于0
func (r *RedisResourceBuilder) setMaxTotal(maxTotal int64) ResourceBuilder {
    if maxTotal <= 0 {
       r.err = errors.New("maxTotal小于0")
       return r
    }
    r.resourceParams.maxTotal = maxTotal
    fmt.Println("RedisResourceBuilder setMaxTotal ", maxTotal)
    return r
}

// 设置maxIdle值,值不能小于0
func (r *RedisResourceBuilder) setMaxIdle(maxIdle int64) ResourceBuilder {
    if maxIdle <= 0 {
       r.err = errors.New("maxIdle小于0")
       return r
    }
    r.resourceParams.maxIdle = maxIdle
    fmt.Println("RedisResourceBuilder setMaxIdle ", maxIdle)
    return r
}

// 设置minIdle值,值不能小于0
func (r *RedisResourceBuilder) setMinIdle(minIdle int64) ResourceBuilder {
    if minIdle <= 0 {
       r.err = errors.New("minIdle小于0")
       return r
    }
    r.resourceParams.minIdle = minIdle
    fmt.Println("RedisResourceBuilder setMinIdle ", minIdle)
    return r
}

// 获取错误信息
func (r *RedisResourceBuilder) getError() error {
    return r.err
}

// 构建product: 1. 做参数校验  2. 根据参数生成product
func (r *RedisResourceBuilder) build() (p ResourceProduct) {
    // 校验逻辑放到这里来做,包括必填项校验、依赖关系校验、约束条件校验等
    if r.resourceParams.name == "" {
       r.err = errors.New("name为空")
       return
    }

    if !((r.resourceParams.maxIdle == 0 && r.resourceParams.minIdle == 0 && r.resourceParams.maxTotal == 0) ||
       (r.resourceParams.maxIdle != 0 && r.resourceParams.minIdle != 0 && r.resourceParams.maxTotal != 0)) {
       r.err = errors.New("数据需要保持一致")
       return
    }

    if r.resourceParams.maxIdle > r.resourceParams.maxTotal {
       r.err = errors.New("maxIdle > maxTotal")
       return
    }
    if r.resourceParams.minIdle > r.resourceParams.maxTotal || r.resourceParams.minIdle > r.resourceParams.maxIdle {
       r.err = errors.New("minIdle > maxTotal|maxIdle")
       return
    }
    fmt.Println("RedisResourceBuilder build")
    product := &RedisResourceProduct{
       resourceParams: r.resourceParams,
    }
    return product
}

// 指挥者
type Director struct {
}

// 指挥者控制建造过程
func (d *Director) construct(builder ResourceBuilder) ResourceProduct {
    resourceProduct := builder.setName("redis").
       setMinIdle(10).
       setMaxIdle(10).
       setMaxTotal(20).
       build()

    err := builder.getError()
    if err != nil {
       fmt.Println("构建失败,原因为" + err.Error())
       return nil
    }
    return resourceProduct
}

func main() {
    builder := &RedisResourceBuilder{}

    director := &Director{}
    product := director.construct(builder)

    if product == nil {
       return
    }

    product.show()
}
相关推荐
c1s2d3n4cs10 分钟前
Qt使用QZipWriter和QZipReader来解压、压缩文件
开发语言·qt
huipeng92610 分钟前
第三章线性表+第四章ArrayList与顺序表
java·开发语言
sin220114 分钟前
springboot测试类里注入不成功且运行报错
spring boot·后端·sqlserver
酷讯网络_24087016023 分钟前
【全开源】Java多语言tiktok跨境商城TikTok内嵌商城送搭建教程
java·开发语言·开源
思忖小下25 分钟前
梳理你的思路(从OOP到架构设计)_设计模式Observer模式
观察者模式·设计模式·eit
海威的技术博客1 小时前
JS中的原型与原型链
开发语言·javascript·原型模式
WPG大大通1 小时前
基于DIODES AP43781+PI3USB31531+PI3DPX1207C的USB-C PD& Video 之全功能显示器连接端口方案
c语言·开发语言·计算机外设·开发板·电源·大大通
从以前1 小时前
【算法题解】Bindian 山丘信号问题(E. Bindian Signaling)
开发语言·python·算法
kirito学长-Java1 小时前
springboot/ssm网上宠物店系统Java代码编写web宠物用品商城项目
java·spring boot·后端