首页 > 其他分享 >Golang - 选项模式 vs 构建器模式

Golang - 选项模式 vs 构建器模式

时间:2023-06-02 17:34:06浏览次数:39  
标签:string 对象 模式 Golang vs func pizza Pizza

image.png

在使用Golang创建复杂对象时,常用的两种模式是选项模式(Options pattern)和构建器模式(Builder pattern)。这两种模式各有优缺点,选择适合项目需求的模式取决于具体情况。

问题

假设我们想创建一个具有许多可选参数的复杂对象。一种方法是创建一个构造函数,该构造函数接受所有参数,并为可选参数提供默认值。然而,这种方法有一些缺点:

  1. 很难记住参数的顺序。
  2. 很难知道哪些参数是可选的,哪些是必需的。
  3. 构造函数可能会变得很长,难以阅读。

解决方案:选项模式

选项模式可以用于创建具有许多可选参数的对象。在这种模式下,我们定义一个包含可选参数的结构体,并提供设置这些参数的方法。相比构建器模式,这种模式可以更简洁,并且适用于参数较少的对象。

示例

在Golang中,可以使用函数选项(functional options)来实现选项模式。函数选项是以结构体作为参数,并返回修改后的结构体的函数。以下是使用选项模式创建披萨对象的示例代码:

type Pizza struct {
    dough     string
    sauce     string
    cheese    string
    toppings  []string
}

type PizzaOptions struct {
    Dough     string
    Sauce     string
    Cheese    string
    Toppings  []string
}

type PizzaOption func(*PizzaOptions)

func WithDough(dough string) PizzaOption {
    return func(po *PizzaOptions) {
        po.Dough = dough
    }
}

func WithSauce(sauce string) PizzaOption {
    return func(po *PizzaOptions) {
        po.Sauce = sauce
    }
}

func WithCheese(cheese string) PizzaOption {
    return func(po *PizzaOptions) {
        po.Cheese = cheese
    }
}

func WithToppings(toppings []string) PizzaOption {
    return func(po *PizzaOptions) {
        po.Toppings = toppings
    }
}

func NewPizza(options ...PizzaOption) *Pizza {
    opts := &PizzaOptions{}
    for _, option := range options {
        option(opts)
    }
    pizza := &Pizza{
        dough: opts.Dough,
        sauce: opts.Sauce,
        cheese: opts.Cheese,
        toppings: opts.Toppings,
    }
    return pizza
}

在这个例子中,我们定义了Pizza结构体和PizzaOptions结构体,其中PizzaOptions是一个包含可选参数的结构体。然后,我们定义了一系列函数来设置每个选项,比如WithDoughWithSauceWithToppings。这些函数返回一个PizzaOption,用于设置PizzaOptions结构体上对应的字段。最后,我们定义了一个NewPizza函数,它接受任意数量的PizzaOptions参数,并构造一个Pizza对象。

func main() {
   pizza := NewPizza(
      WithDough("Regular"),
      WithSauce("Tomato"),
      WithCheese("Mozzarella"),
      WithToppings([]string{"Pepperoni", "Olives", "Mushrooms"}),
   )

   println(pizza.dough)
   println(pizza.sauce)
   println(pizza.cheese)
   println(pizza.toppings)
}

Options模式可以是Builder模式的一个很好替代方案,用于创建具有许多可选参数的对象,特别是当对象的参数较少时。然而,对于具有许多参数的对象来说,Options模式可能变得笨拙,因为需要设置所有选项的函数数量可能会很大。

在Golang标准库中的使用

Options模式在Golang标准库中被用于创建诸如http.Request对象之类的对象,该对象具有许多可选参数。http.NewRequest函数接受方法、URL和可选的headers和body等参数,返回一个新的http.Request对象。headers和body是可选参数,可以使用函数选项来设置。

替代方案:Builder模式

Builder模式通过将复杂对象的构建与其表示分离,提供了对这些问题的解决方案。Builder模式涉及以下组件:

  1. Builder接口,定义构建对象的步骤。
  2. ConcreteBuilder结构体,实现Builder接口并提供构建对象的方法。
  3. Director结构体,使用Builder来构建对象。

示例

以下是在Golang中使用Builder模式实现的示例,使用了文章中提到的pizza对象:

type Pizza struct {
    dough     string
    sauce     string
    cheese    string
    toppings  []string
}

type PizzaBuilder interface {
    SetDough(string) PizzaBuilder
    SetSauce(string) PizzaBuilder
    SetCheese(string) PizzaBuilder
    SetToppings([]string) PizzaBuilder
    Build() *Pizza
}

type ConcretePizzaBuilder struct {
    pizza *Pizza
}

func NewConcretePizzaBuilder() *ConcretePizzaBuilder {
    return &ConcretePizzaBuilder{pizza: &Pizza{}}
}

func (cpb *ConcretePizzaBuilder) SetDough(dough string) PizzaBuilder {
    cpb.pizza.dough = dough
    return cpb
}

func (cpb *ConcretePizzaBuilder) SetSauce(sauce string) PizzaBuilder {
    cpb.pizza.sauce = sauce
    return cpb
}

func (cpb *ConcretePizzaBuilder) SetCheese(cheese string) PizzaBuilder {
    cpb.pizza.cheese = cheese
    return cpb
}

func (cpb *ConcretePizzaBuilder) SetToppings(toppings []string) PizzaBuilder {
    cpb.pizza.toppings = toppings
    return cpb
}

func (cpb *ConcretePizzaBuilder) Build() *Pizza {
    return cpb.pizza
}

type Director struct {
    builder PizzaBuilder
}

func NewDirector(builder PizzaBuilder) *Director {
    return &Director{builder: builder}
}

func (d *Director) Construct() *Pizza {
    return d.builder.SetDough("Thin Crust").SetSauce("Tomato").SetCheese("Mozzarella").SetToppings([]string{"Mushrooms", "Olives", "Onions"}).Build()
}

在这个示例中,我们定义了Pizza结构体和PizzaBuilder接口。ConcretePizzaBuilder结构体实现了PizzaBuilder接口,并提供了构建Pizza对象的方法。Director结构体使用PizzaBuilder来构建Pizza对象。Director结构体并不是严格必需的,但它提供了一种简化构建Pizza对象过程的方式。

我们可以使用DirectorConcretePizzaBuilder来创建一个Pizza对象,如下所示:

builder := NewConcretePizzaBuilder()
director := NewDirector(builder)
pizza := director.Construct()

这将创建一个具有以下属性的Pizza对象:

  • Dough: Thin Crust
  • Sauce: Tomato
  • Cheese: Mozzarella
  • Toppings: Mushrooms, Olives, Onions

请注意,我们只需要指定要更改的属性。所有其他属性都被设置为默认值。这使得创建具有许多可选参数的复杂对象变得更容易,无需记住参数的顺序以及哪些参数是可选的,哪些是必需的。

在 Golang 标准库中的使用

构建器模式并未在 Golang 标准库中使用,但在 Golang 应用程序中,它是一种常用的模式,用于创建具有许多可选参数的复杂对象。Options 模式也被用作 Golang 应用程序中的替代方案,用于创建具有许多可选参数的对象。

结论

Options 模式是 Builder 模式的一种替代方案,可用于创建具有许多可选参数的对象。它比 Builder 模式更简洁,但对于具有许多参数的对象可能会变得笨重。在 Golang 中,可以使用函数选项来实现 Options 模式。

Builder 模式是一种强大的模式,可以用于创建具有许多可选参数的复杂对象。它将对象的构建与表示分离,并提供了一种使用相同构建过程创建同一对象的不同表示的方式。在 Golang 中,Builder 模式可用于轻松创建复杂对象。

参考资料

  • "Design Patterns: Elements of Reusable Object-Oriented Software"(《设计模式:可复用面向对象软件的基础》)Erich Gamma、John Vlissides、Ralph Johnson、Richard Helm著

如果你喜欢我的文章,点赞,关注,转发!

标签:string,对象,模式,Golang,vs,func,pizza,Pizza
From: https://blog.51cto.com/slagga/6404305

相关文章

  • 方芳:坡面尺度上混合植被恢复模式对土壤修复的影响
    武汉市江夏区交通局武汉市江夏区公路局  武汉市江夏区公路建筑工程公司武汉市江夏城投集团有限公司武汉江夏路桥工程总公司 武汉工程大学 土木工程与建筑学院    方芳    15927602711坡面尺度上混合植被恢复模式对土壤修复的影响摘要:随着人类活动......
  • 工厂模式配置servlet(servlet升级版)
    1、创建一个类点击查看代码packagecom.bh.controller;importjavax.servlet.ServletException;importjavax.servlet.http.HttpServlet;importjavax.servlet.http.HttpServletRequest;importjavax.servlet.http.HttpServletResponse;importjava.io.IOException;impo......
  • 设计模式
    设计模式介绍设计模式是程序员在面对同类软件工程设计问题所总结出来的有用经验,是某类问题的通用解决方案。作用:使程序(软件)具有更好:代码重用性(即相同功能的代码,不用多次编写)可读性(即编程规范性,便于其他程序员的阅读和理解)可扩展性(即:当需要增加新的功能时,非常的方便,称为可维......
  • USB Type-C端口静电放电浪涌防护,TVS/ESD二极管怎么选型?
    USBType-C是一种USB接口外形标准,体积比Type-A和Type-B小得多。USBType-C接口特点有:最大数据传输速度可达10Gbit/秒,是USB3.1的标准;纤薄设计;支持正反两面插入,没有正反区别,可随意热插拔;配备Type-C连接器的标准规格连接线可通过3A电流;支持超出现有USB供电能力的“USBPD”;提供最大10......
  • 商业模式:收入、价值与过程
    如上,只说了求橙商学院,商业模式课内的一个小练习,这篇稍稍展开到浙大的郭斌教授第一个半天的核心内容。商业的基础是市场经济,市场经济的基本行为是交易,交易的本质是价值交换,所以,我们可以用如下一句话来概括所有的商业模式:为谁提供什么价值,靠什么方式转换为收入,前两者之间做什么商业过......
  • 【网络知识】虚拟机的桥接、NAT、仅主机模式分别是什么?
    在我们安装VMware时,VMware会自动三种3种网络连接模式,分别为VMnet0(桥接模式)、VMnet8(NAT模式)、VMnet1(仅主机模式),当然我们也可以根据需要自行创建更多的虚拟网络。【个人实验环境】pc:win11家庭版虚拟机:VMwareWorkstation16Pro,16.2.1build-18811642iso镜像:CentO......
  • 策略模式
    参考网址:https://www.yuque.com/qinbinbin-nw5ci/kb/aqn1o8?#@ServicepublicclassBankCardForStrategyFactoryimplementsApplicationContextAware{//创建一个map集合ConcurrentHashMap保证线程是安全的Map<String,StrategyBankCardRuteService>map=newConcur......
  • 设计模式小记
    创建型模式(CreationalPatterns):工厂模式(FactoryPattern):通过工厂方法创建对象,隐藏具体实现细节,例如创建不同类型的按钮。//简单工厂模式classButtonFactory{createButton(type){switch(type){case'submit':returnnewSubmitButton();......
  • 配置VsCode的QT工程
    配置VsCode的QT工程VsCode+qmake环境(Environment):......
  • [ASP.NET MVC 小牛之路]01 - 理解MVC模式
    文章出处: https://www.cnblogs.com/willick/p/3195560.html理解一般意义上的MVC模式MVC模式(Model-View-Controller)是软件工程中的一种软件架构模式,把软件系统分为以下三个基本部分:模型(Model):模型用于封装与应用程序的业务逻辑相关的数据以及对数据的处理方法。“模型”有......