jjzjj

建造者模式

飞天猪皮怪 2023-03-28 原文

定义

将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以建造不同的表示。

建造者模式的结构

Builder(抽象建造者):它为创建一个产品Product对象的各个部件指定抽象接口,在该接口中一般声明两类方法,一类方法是BuildPartX(),他们用于创建复杂对象的各个部件,另一类方法是GetResult(),它们用于返回复杂对象,Builder可以是抽象类,也可以是接口。

ConcreteBuilder(具体建造者):它实现了Builder接口,实现各个部件的具体构造和装配方法,定义并明确所创建的复杂对象,还可以提供一个方法返回创建好的复杂产品对象(该方法可以由抽象建造者实现)。

Product(产品):它是被构建的复杂对象,包含多个组成部件,具体构造者创建该产品内部表示并定义它的装配过程。

Director(指挥者):指挥者又称为导演类,它负责安排复杂对象的构造次序,指挥者与抽象建造者之间存在关联。可以在其Construct()建造方法中调用建造者对象的部件构造和装备方法,完成复杂对象的构造。客户端一般只需要与指挥者进行交互,在客户端确认具体建造者的类型,并实例化具体建造者对象(也可以通过配置文件和反射机制),然后通过指挥者类的构造函数或者Setter方法将对象传入指挥者类中。有些时候Director不是必要的,例如你可以直接在抽象类中完成这个操作。

撸码

汽车通常是由发动机、底盘、车身和电气设备4大部分组成。

今天这个例子就以汽车为例。

Builder:CarBuilder

ConcreteBuilder:BenzCarBuilder,BMWCarBuilder,AudiCarBuilder

Product:汽车

Director:CarController

 

Product

首先好确认车子的组成部分(属性)

 /// <summary>
    /// Product
    /// </summary>
    public class Car
    {

        /// <summary>
        /// 发动机
        /// </summary>
        public string Engine { get; set; }

        /// <summary>
        /// 底盘
        /// </summary>
        public string Chassis { get; set; }

        /// <summary>
        /// 车声
        /// </summary>
        public string Body { get; set; }
        /// <summary>
        /// 电气设备
        /// </summary>
        public string Electrical { get; set; }


    }

 

Builder

抽象建造者类

/// <summary>
    /// 抽象建造者
    /// </summary>
    public abstract class CarBuilder
    {
        protected Car _car;

        /*
         抽象方法
         */
        public abstract void BuilderEngine();
        public abstract void BuilderChassis();
        public abstract void BuilderBody();
        public abstract void BuilderElectrical();


        public CarBuilder()
        {
            _car = new Car();
        }


        public Car CreateCar()
        {
            return _car;
        }

 

ConcreteBuilder

具体的实现,这里为了避免代码看起来很多我就只放两个实现的Builder。

Audi

public class AudiCarBuilder : CarBuilder
    {
        public override void BuilderBody()
        {
            _car.Body = "豪华车身";
        }

        public override void BuilderChassis()
        {
            _car.Chassis = "贵族底盘";
        }

        public override void BuilderElectrical()
        {
            _car.Electrical = "史诗级电气设备,100000000w超级喷射";
        }

        public override void BuilderEngine()
        {
            _car.Engine = "行星级发动机";
        }
    }

 

Benz

 public class BenzCarBuilder : CarBuilder
    {
        public override void BuilderBody()
        {
            _car.Body = "镶钻车身";
        }

        public override void BuilderChassis()
        {
            _car.Chassis = "真空底盘";
        }

        public override void BuilderElectrical()
        {
            _car.Electrical = "100000000w超级喷射";
        }

        public override void BuilderEngine()
        {
            _car.Engine = "恒星级发动机";
        }
    }

CarController(指挥者)

 /// <summary>
    /// 指挥者
    /// </summary>
    public class CarController
    {
        /// <summary>
        /// 构建一辆汽车
        /// </summary>
        /// <param name="builder"></param>
        /// <returns></returns>
        public Car Construct(CarBuilder builder)
        {

            //可以调整你想要的顺序
            builder.BuilderBody();
            builder.BuilderChassis();
            builder.BuilderElectrical();
            builder.BuilderEngine();

            return builder.CreateCar();
        }
    }

最后测试一下

                CarController controller = new CarController();

                //造一辆奥迪车
                Car car = controller.Construct(new AudiCarBuilder());

                Console.WriteLine(car.Body);
                Console.WriteLine(car.Chassis);
                Console.WriteLine(car.Electrical);
                Console.WriteLine(car.Engine);

结果

 

 当然如果你的Car很多,就可以用一个枚举或者工厂去处理它。

总结

建造者模式的优点

1.在建造者模式,客户端不必知道产品内部组成的细节,将产品本身和产品的创建过程解耦,使得相同的创建过程可以创建不同的产品对象。

2.每一个具体建造者都相对独立,与其他的具体构造着无关,因此可以很方便的地替换具体建造者或增加新的建造者,用户使用不同的具体建造者即可获得不同的产品对象。由于建造者类针对抽象建造者编程,增加新的建造者无需修改原有类库的代码,系统扩展方便,符合开闭原则。

3.用户可以更加精细地控制产品的创建过程,将复杂产品的创建步骤分解到不同的方法中,使得创建过程更加清晰,也更方便使用程序来控制创建过程。

建造者模式的缺点

1.建造者模式所创建的产品一般具有较多的相同点,其组成方式很相似,如果产品之间的差异性很大,例如很多组成部分不相同,则不适合使用建造者模式,因此其适用范围收到一定的限制。

2.如果产品的内部变化复杂,可能需要定义很多具体建造者来实现这种变化,导致系统变得很庞大,增加系统的理解难度和运行成本。

适用环境

1.需要生成的产品对象有复杂的内部结构,这些产品对象通常包含多个成员变量。

2.需要生成的产品对象属性相互依赖,需要指定生成顺序。

3.对象的创建过程独立于创建该对象的类。在建造者模式中通过引入指挥者类,将创建过程封装到指挥者类中,而不是在建造者类和客户类。

4.隔离复杂对象的创建和使用,并使得相同的创建过程创建不同的产品。

 

有关建造者模式的更多相关文章

  1. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

  2. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  3. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

  4. ruby - 是否有用于序列化和反序列化各种格式的对象层次结构的模式? - 2

    给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最

  5. ruby-on-rails - environment.rb 中设置的常量在开发模式中消失 - 2

    了解Rails缓存如何工作的人可以真正帮助我。这是嵌套在Rails::Initializer.runblock中的代码:config.after_initializedoSomeClass.const_set'SOME_CONST','SOME_VAL'end现在,如果我运行script/server并发出请求,一切都很好。然而,在我的Rails应用程序的第二个请求中,一切都因单元化常量错误而变得糟糕。在生产模式下,我可以成功发出第二个请求,这意味着常量仍然存在。我已通过将以上内容更改为以下内容来解决问题:config.after_initializedorequire'some_cl

  6. Ruby:标准递归模式 - 2

    我经常迷上ruby​​的一件事是递归模式。例如,假设我有一个数组,它可能包含无限深度的数组作为元素。所以,例如:my_array=[1,[2,3,[4,5,[6,7]]]]我想创建一个方法,可以将数组展平为[1,2,3,4,5,6,7]。我知道.flatten可以完成这项工作,但这个问题是作为我经常遇到的递归问题的一个例子-因此我试图找到一个更可重用的解决方案。简而言之-我猜这种事情有一个标准模式,但我想不出任何特别优雅的东西。任何想法表示赞赏 最佳答案 递归是一种方法,它不依赖于语言。您在编写算法时要考虑两种情况:再次调用函数的情

  7. ruby - 在 Ruby 中查找多个正则表达式匹配的模式和位置 - 2

    这应该是一个简单的问题,但我找不到任何相关信息。给定一个Ruby中的正则表达式,对于每个匹配项,我需要检索匹配的模式$1、$2,但我还需要匹配位置。我知道=~运算符为我提供了第一个匹配项的位置,而string.scan(/regex/)为我提供了所有匹配模式。如果可能,我需要在同一步骤中获得两个结果。 最佳答案 MatchDatastring.scan(regex)do$1#Patternatfirstposition$2#Patternatsecondposition$~.offset(1)#Startingandendingpo

  8. ruby - sinatra 框架的 MVC 模式 - 2

    我想开始使用“Sinatra”框架进行编码,但我找不到该框架的“MVC”模式。是“MVC-Sinatra”模式或框架吗? 最佳答案 您可能想查看Padrino这是一个围绕Sinatra构建的框架,可为您的项目提供更“类似Rails”的感觉,但没有那么多隐藏的魔法。这是使用Sinatra可以做什么的一个很好的例子。虽然如果您需要开始使用这很好,但我个人建议您将它用作学习工具,以对您来说最有意义的方式使用Sinatra构建您自己的应用程序。写一些测试/期望,写一些代码,通过测试-重复:)至于ORM,你还应该结帐Sequel其中(imho

  9. ruby-on-rails - Rails 如何创建数据模式种子数据 - 2

    有没有一种方法可以自动生成种子数据文件并创建种子数据,就像您在下面链接中的Laravel中看到的那样?LaravelDatabaseMigrations&Seed我在另一个应用程序上看到在Rails的db文件夹下创建了一些带有时间戳的文件,其中包含种子数据。创建它的好方法是什么? 最佳答案 我建议你使用Fabrication的组合gem和Faker.Fabrication允许您编写一个模式来构建您的对象,而Faker为您提供虚假数据,如姓名、电子邮件、电话号码等。这是制造商的样子:Fabricator(:user)dousernam

  10. ruby-on-rails - Ruby on Rails 应用程序的只读模式 - 2

    我有一个交互式RubyonRails应用程序,我想在特定时间将其置于“只读模式”。这将允许用户读取他们需要的数据,但阻止他们执行写入数据库的操作。执行此操作的一种方法是在数据库中放置一个true/false变量,该变量在进行任何写入之前进行检查。我的问题。有没有更优雅的解决方案来解决这个问题? 最佳答案 如果你真的想阻止任何数据库写入,我能想到的最简单的方法是覆盖readonly?始终返回true的模型方法,无论是在选定模型中还是对于所有ActiveRecord模型。如果模型设置为只读(通常通过调用#readonly!来完成),任何

随机推荐