当前位置: 首页 > news >正文

建造者模式(Builder Pattern)

建造者模式(Builder Pattern)是一种创建型设计模式,它主要用于将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。这种设计模式的核心思想是将一个复杂对象的构建分解成多个相对简单的步骤,并允许通过不同的方式组合这些步骤来创建不同的对象。以下是关于建造者模式的详细解析:

一、定义与特点

  • 定义: 建造者模式将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。

  • 特点:
    1.封装性: 将复杂对象的创建过程封装在独立的建造者类中,使得对象的创建过程更加清晰和易于理解。
    2.灵活性: 通过使用建造者模式,可以灵活地修改对象的创建过程,而不需要修改对象的具体表示。
    3.可复用性: 建造者模式将对象的创建过程分解为多个步骤,并将这些步骤封装在独立的建造者类中,从而提高了代码的可复用性。

二、角色与结构

在建造者模式中,通常包含以下几个角色:

Product(产品类): 表示被构建的复杂对象。它包含多个组成部件的类,以及这些部件的装配方式。
Builder(抽象建造者): 为创建一个产品对象的各个部件指定抽象接口。它不提供具体的实现,而是由子类来实现具体的构建过程。
ConcreteBuilder(具体建造者): 实现Builder接口,完成复杂产品的各个部件的具体创建方法。在构造过程完成后,提供产品的实例。
Director(指挥者): 构造一个使用Builder接口的对象。它主要负责定义构建过程的执行顺序,并通过Builder接口来指导具体建造者完成构建过程。

三、适用场景

建造者模式适用于以下场景:

  1. 创建的对象较复杂,由多个部件构成,各部件面临着复杂的变化,但构件间的建造顺序是稳定的。
  2. 创建复杂对象的算法独立于该对象的组成部分以及它们的装配方式,即产品的构建过程和最终的表示是独立的。
  3. 当需要生成的产品对象有复杂的内部结构,并且这些产品的内部表示被隐藏起来时,可以使用建造者模式。

四、优缺点

  • 优点
    1.封装性好: 将复杂对象的创建过程封装在独立的建造者类中,使得对象的创建过程更加清晰和易于理解。
    2.灵活性高: 可以通过不同的具体建造者来创建不同的产品表示,而不需要修改产品的内部结构和创建过程。
    3.可复用性强: 建造者模式将对象的创建过程分解为多个步骤,并将这些步骤封装在独立的建造者类中,从而提高了代码的可复用性。
  • 缺点
    1.类的数量增多: 由于将对象的创建过程分解为多个步骤,并将这些步骤封装在独立的建造者类中,因此需要创建多个建造者类,从而增加了类的数量。
    2.建造过程复杂: 如果对象的创建过程过于复杂,可能会导致建造者类的建造过程也变得复杂,从而增加了代码的维护难度。

五、示例

以构建汽车为例,假设汽车由引擎、轮胎和车门等多个部件组成。使用建造者模式,可以分别为每种类型的车(如跑车和SUV)提供不同的建造者。这些建造者负责构建汽车的各个部件,并通过指挥者类来确保构建过程的顺序和正确性。最终,客户端可以通过调用指挥者类的方法来获取构建好的汽车对象。
在C#中实现建造者模式,你需要定义几个关键组件:产品(Product)、抽象建造者(Builder)、具体建造者(ConcreteBuilder)以及指挥者(Director)。以下是一个简单的C#示例,展示了如何使用建造者模式来构建不同类型的车辆(如汽车和卡车)。

  1. 定义产品类(Vehicle)
    首先,定义一个产品类,它包含了一些基本的部件和组装这些部件的方法(尽管在建造者模式中,这些方法的实现可能会留空,因为具体的组装过程将由建造者来完成)。
public class Vehicle  
{  public string Engine { get; set; }  public string Wheels { get; set; }  public string Body { get; set; }  // 通常,这里的方法可能用于设置或获取车辆的状态,但在建造者模式中,  // 这些方法可能会被留空或用于验证构建的车辆。  public void Display()  {  Console.WriteLine($"Vehicle with Engine: {Engine}, Wheels: {Wheels}, Body: {Body}");  }  
}
  1. 定义抽象建造者接口(IVehicleBuilder)
    然后,定义一个抽象建造者接口,它声明了构建产品所需的方法。
public interface IVehicleBuilder  
{  void BuildEngine();  void BuildWheels();  void BuildBody();  Vehicle GetVehicle();  
}
  1. 定义具体建造者类(CarBuilder 和 TruckBuilder)
    接下来,为每个类型的产品定义具体的建造者类。这些类实现了抽象建造者接口,并提供了构建产品的具体步骤。
public class CarBuilder : IVehicleBuilder  
{  private Vehicle _vehicle = new Vehicle();  public void BuildEngine()  {  _vehicle.Engine = "V8 Engine";  }  public void BuildWheels()  {  _vehicle.Wheels = "4 Wheels";  }  public void BuildBody()  {  _vehicle.Body = "Sedan Body";  }  public Vehicle GetVehicle()  {  return _vehicle;  }  
}  public class TruckBuilder : IVehicleBuilder  
{  private Vehicle _vehicle = new Vehicle();  public void BuildEngine()  {  _vehicle.Engine = "Diesel Engine";  }  public void BuildWheels()  {  _vehicle.Wheels = "6 Wheels";  }  public void BuildBody()  {  _vehicle.Body = "Truck Body";  }  public Vehicle GetVehicle()  {  return _vehicle;  }  
}
  1. 定义指挥者类(VehicleDirector)
    最后,定义一个指挥者类,它负责控制构建过程。指挥者类持有一个建造者对象的引用,并定义了一个构建产品的步骤列表。
public class VehicleDirector  
{  private IVehicleBuilder _builder;  public VehicleDirector(IVehicleBuilder builder)  {  _builder = builder;  }  public void Construct()  {  _builder.BuildEngine();  _builder.BuildWheels();  _builder.BuildBody();  }  public Vehicle GetVehicle()  {  return _builder.GetVehicle();  }  
}
  1. 客户端代码
    现在,你可以在客户端代码中使用这些类来构建不同类型的车辆了。
class Program  
{  static void Main(string[] args)  {  // 使用CarBuilder构建汽车  IVehicleBuilder carBuilder = new CarBuilder();  VehicleDirector carDirector = new VehicleDirector(carBuilder);  carDirector.Construct();  Vehicle car = carDirector.GetVehicle();  car.Display();  // 使用TruckBuilder构建卡车  IVehicleBuilder truckBuilder = new TruckBuilder();  VehicleDirector truckDirector = new VehicleDirector(truckBuilder);  truckDirector.Construct();  Vehicle truck = truckDirector.GetVehicle();  truck.Display();  }  
}

在这个例子中,建造者模式使得创建不同类型的车辆变得灵活且易于管理。你可以通过添加新的具体建造者类来轻松地扩展系统,以支持更多类型的车辆。

六、总结

建造者模式是一种强大的设计模式,它通过将复杂对象的构建过程与其表示分离,提供了更好的封装性、灵活性和可复用性。在需要创建复杂对象且对象的创建过程需要高度定制化的场景中,建造者模式是一个非常好的选择。

http://www.lryc.cn/news/410857.html

相关文章:

  • 【MySQL】索引 【下】{聚簇索引VS非聚簇索引/创建主键索引/全文索引的创建/索引创建原则}
  • 论文快过(图像配准|Coarse_LoFTR_TRT)|适用于移动端的LoFTR算法的改进分析 1060显卡上45fps
  • 免费发送邮件两种接口方式:SMTP和邮件API
  • 大模型日报 2024-07-30
  • docker 构建 mongodb
  • LeetCode每日练习 | 二分查找 | 数组 |Java | 图解算法
  • 2024年获客新渠道,大数据爬虫获客:技术实现精准抓取数据资源
  • 滑模变结构控制仿真实例(s-function代码详解)
  • MySQL处理引擎
  • HTTP 方法详解:GET、POST 和 PUT
  • 被工信部认可的开源软件治理解决方案
  • 文件包含漏洞--pyload
  • C++包管理之`vcpkg`简介
  • 【机器学习】必会核函数之:高斯核函数
  • 51单片机和STM32区别
  • Python 伪随机数生成器
  • 7.5 grafana上导入模板看图并讲解告警
  • BUG解决(vue3+echart报错):Cannot read properties of undefined (reading ‘type‘)
  • VSCode+git的gitee仓库搭建
  • Golang | Leetcode Golang题解之第297题二叉树的序列化与反序列化
  • 交叉熵和MSE的区别
  • 具身智能又进一步!卡内基梅隆Meta苏黎世联邦实现虚拟人超灵活抓取
  • 嘉盛:货币政策走向与市场预期
  • [C#]基于wpf实现的一百多种音色的Midi键盘软件
  • 关于香橙派系统烧录,1.1.8或者1.1.10两个版本都无法启动Orangepi5
  • 深入解析Python `httpx`源码,探索现代HTTP客户端的秘密!
  • python爬虫【3】—— 爬虫反反爬
  • LIS接入开发
  • Stable Diffusion Windows本地部署超详细教程(手动+自动+整合包三种方式)
  • 【Golang 面试 - 基础题】每日 5 题(七)