【建造者模式】

2023-09-19 11:27:54

🏅我是默,一个在CSDN分享笔记的博主。📚📚

🌟在这里,我要推荐给大家我的专栏《20种Java设计模式》。🎯🎯

🚀无论你是编程小白,还是有一定基础的程序员,这个专栏都能满足你的需求。我会用最简单易懂的语言,带你走进Java的世界,让你从零开始,一步步成为JAVA大师。🚀🏆

🌈让我们在Java的世界里畅游吧!🌈

🎁如果感觉还不错的话请记得给我点赞哦!🎁🎁

💖期待你的加入,一起学习,一起进步!💖💖

一.建造者模式的入门介绍

1.什么是建造者模式?

建造者模式(Builder Pattern)是一种创建型设计模式,用于创建复杂对象。它将一个对象的构建过程和表示分离,使得同样的构建过程可以创建不同的表示

建造者模式的核心思想是将对象的构建过程从其表示中解耦出来,使得可以独立地创建不同的产品。通过指挥者来组织构建过程,客户端无需关心具体的构建细节,只需要通过指挥者来获得最终构建完成的产品。

建造者模式适用于需要创建复杂对象,且创建过程具有一定的顺序或步骤的情况。它提供了更好的灵活性和可扩展性,使得可以通过增加或更改具体的建造者来创建不同的产品变种。同时,它也避免了构造函数参数过多的问题,使得代码更加清晰易读。

总结起来,建造者模式能够将复杂对象的构建过程与其表示分离,提供了一种优雅的方式来创建具有不同配置的对象。   

2.建造者模式通常涉及到几的角色!

产品(Product):表示要构建的复杂对象。它通常包含多个部分或属性。

抽象建造者(Abstract Builder):定义了创建产品各个部分的接口,并提供了一个获取产品的方法。典型的方法包括创建部件、组装部件以及返回最终产品。

具体建造者(Concrete Builder):实现了抽象建造者接口,负责实际构建产品的各个部分,并返回最终的产品。

指挥者(Director):负责使用抽象建造者定义的接口来构建产品。它通常不知道具体的构建过程,只负责根据具体的构建者来组织构建过程。

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

创建对象的构造过程比较复杂,包含多个可选参数或具有多个步骤

希望创建不同配置的对象,但避免使用过多的构造函数或参数

需要保证对象在构造过程中是不可变的,即一旦构建完成,就不能再修改

希望提供一种清晰的方式来构建对象,并且可以按顺序设置属性

通过拆分对象的构建过程,可以将共同的构建逻辑抽象出来,提高代码的重用性。

4.建造者模式的优点和不足

4.1建造者模式的优点:

封装了对象的创建过程,使得对象的构建代码与客户端代码分离,提高了代码的可读性和可维护性。

可以灵活地构建不同配置的对象,而无需编写多个构造函数或使用复杂的参数列表。

可以通过建造者类提供额外的方法来设置对象的特定属性,使得代码的可扩展性更好。

可以控制对象的构建过程,例如可以进行参数校验、设置默认值等操作。

可以生成不可变的对象,保证了对象在构造完成后的不可变性。

4.2建造者模式的不足

增加了代码量:使用建造者模式会引入额外的建造者类以及对应的方法,这可能增加代码的复杂性和量,特别是当有多个属性需要设置时。

对象构建过程被分离:建造者模式将对象的构建过程分离到不同的方法中,这导致在客户端代码中无法一目了然地看到对象的完整构建过程。

可能会产生多余的对象:在建造者模式中,通过调用不同的建造者方法来设置对象的不同属性,这可能会导致创建了过多的中间对象,增加了内存的开销。

不适合简单对象的创建:建造者模式适用于创建复杂对象,但对于那些属性较少、构建过程简单的对象,使用建造者模式可能过于繁琐,反而增加了代码的复杂性。

不支持方法链式调用:在传统的建造者模式中,每个建造者方法返回的是建造者对象自身,无法进行方法链式调用。如果希望实现方法链式调用,需要额外的工作。

总的来说,建造者模式在创建复杂对象、灵活配置对象属性以及保持对象不可变性方面有其优势,但也需要权衡其在代码量、代码可读性以及对象构建过程分离等方面的不足之处。在实际应用中,需要根据具体情况来选择是否使用建造者模式

二.代码实例实现

1.在IDEA中构建Maven项目时,导入依赖(pom.xml)

<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  <modelVersion>4.0.0</modelVersion>

  <groupId>org.example</groupId>
  <artifactId>jzz</artifactId>
  <version>1.0-SNAPSHOT</version>
  <packaging>war</packaging>

  <name>jzz Maven Webapp</name>
  <!-- FIXME change it to the project's website -->
  <url>http://www.example.com</url>

  <properties>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    <maven.compiler.source>1.7</maven.compiler.source>
    <maven.compiler.target>1.7</maven.compiler.target>
  </properties>

  <dependencies>

      <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <version>1.18.20</version>
        <scope>provided</scope>
      </dependency>

    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>4.11</version>
      <scope>test</scope>
    </dependency>
  </dependencies>

  <build>
    <finalName>jzz</finalName>
    <pluginManagement><!-- lock down plugins versions to avoid using Maven defaults (may be moved to parent pom) -->
      <plugins>
        <plugin>
          <artifactId>maven-clean-plugin</artifactId>
          <version>3.1.0</version>
        </plugin>
        <!-- see http://maven.apache.org/ref/current/maven-core/default-bindings.html#Plugin_bindings_for_war_packaging -->
        <plugin>
          <artifactId>maven-resources-plugin</artifactId>
          <version>3.0.2</version>
        </plugin>
        <plugin>
          <artifactId>maven-compiler-plugin</artifactId>
          <version>3.8.0</version>
        </plugin>
        <plugin>
          <artifactId>maven-surefire-plugin</artifactId>
          <version>2.22.1</version>
        </plugin>
        <plugin>
          <artifactId>maven-war-plugin</artifactId>
          <version>3.2.2</version>
        </plugin>
        <plugin>
          <artifactId>maven-install-plugin</artifactId>
          <version>2.5.2</version>
        </plugin>
        <plugin>
          <artifactId>maven-deploy-plugin</artifactId>
          <version>2.8.2</version>
        </plugin>
      </plugins>
    </pluginManagement>
  </build>
</project>

2.Lombok的注解实现的Computer类(是产品,又充当了具体建造)

package com.lz.entity;

import lombok.Builder;
import lombok.Data;

@Data
@Builder
public class Computer {
    private String cpu;
    private String memory;
    private String hardDisk;
    private String graphicsCard;

    public void display() {
        System.out.println("Computer Specs:");
        System.out.println("CPU: " + cpu);
        System.out.println("Memory: " + memory);
        System.out.println("Hard Disk: " + hardDisk);
        System.out.println("Graphics Card: " + graphicsCard);
    }
}

代码分析

  • cpu:表示CPU型号的字符串。
  • memory:表示内存容量的字符串。
  • hardDisk:表示硬盘容量的字符串。
  • graphicsCard:表示显卡型号的字符串。

此外,该类还定义了一个名为display()的方法,用于展示计算机的规格信息

通过使用Lombok的@Builder注解,可以为该类自动生成一个建造者模式的构造器,使得创建对象时可以更加简洁和灵活。同时,@Data注解会自动生成equals()hashCode()toString()等方法。

使用该类可以创建计算机对象,并调用display()方法展示计算机的规格信息。

3.使用Lombok注解的ComputerBuilder类,用于构建Computer对象 (具体建造者)

package com.lz.ComputerBuilder;
import com.lz.entity.Computer;
import lombok.Builder;

@Builder
public class ComputerBuilder {
    private String cpu;
    private String memory;
    private String hardDisk;
    private String graphicsCard;

    public Computer build() {
        return Computer.builder()
                .cpu(cpu)
                .memory(memory)
                .hardDisk(hardDisk)
                .graphicsCard(graphicsCard)
                .build();
    }
}

代码分析

  • 建造者类ComputerBuilder.Builder
  • 设置属性的方法,例如cpu(String cpu)memory(String memory)
  • 构建方法ComputerBuilder.Builder.build()

ComputerBuilder类本身具有以下属性:

  • cpu:表示计算机的CPU型号。
  • memory:表示计算机的内存大小。
  • hardDisk:表示计算机的硬盘容量。
  • graphicsCard:表示计算机的显卡型号。

build()方法中,创建了Computer.Builder对象,并设置其属性值。然后通过调用build()方法,创建并返回一个完整的Computer对象。

使用ComputerBuilder类,可以方便地创建不同配置的计算机对象,同时避免了构造函数过于复杂的问题。

4.演示如何使用建造者模式创建不同配置的计算机对象,并展示它们的详细信息

package com.lz.test;

import com.lz.entity.Computer;

public class Main {
    public static void main(String[] args) {
        Computer gamingComputer = Computer.builder()
                .cpu("Intel i7")
                .memory("16GB DDR4")
                .hardDisk("1TB SSD")
                .graphicsCard("NVIDIA GeForce RTX 3080")
                .build();
        gamingComputer.display();

        Computer officeComputer = Computer.builder()
                .cpu("Intel i5")
                .memory("8GB DDR4")
                .hardDisk("500GB HDD")
                .graphicsCard("Integrated Graphics")
                .build();
        officeComputer.display();
    }
}

代码分析 

main()方法中,首先使用建造者模式创建了一个名为gamingComputer的游戏电脑对象。通过链式调用方式,设置了游戏电脑的CPU型号、内存容量、硬盘容量和显卡型号,并最后调用build()方法构建了完整的电脑对象。然后调用display()方法来显示游戏电脑的详细信息。

接着,使用同样的方式创建了一个名为officeComputer的办公电脑对象,并设置了对应的属性值。同样地,调用build()方法构建办公电脑对象,并通过display()方法展示详细信息。

通过这个测试类,可以看到通过建造者模式,可以方便地创建不同配置的计算机对象,并灵活地设置其属性值。同时,通过display()方法展示了各个计算机对象的详细信息,验证了建造者模式的使用效果。

测试结果

 

更多推荐

JVM 篇

目录一、知识点汇总二、知识点详解2.1JVM的主要组成部分及其作用2.2JVM内存模型2.3堆与栈的区别2.4JVM加载class文件的原理机制2.5类的生命周期2.6Java对象结构2.7Java对象创建过程2.8指针碰撞2.9空闲列表2.10TLAB+CAS2.11说说对象内存分配2.12什么是标量与聚合量?2.1

OpenCV自学笔记十六:直方图处理

在OpenCV中,直方图(Histogram)是用于表示图像中像素强度分布的一种统计工具。它可以帮助我们了解图像的亮度、对比度、色彩分布等信息。OpenCV提供了一个函数`cv2.calcHist()`,用于计算图像的直方图。该函数接受图像数组和一些参数,例如要计算的通道数、区间的数量等。它返回一个表示直方图的一维数组

智荟康午休课桌椅成为第十届中国慈善展览会公益亮点产品

第十届中国慈善展览会(以下简称“慈展会”)于9月15日至17日在深圳会展中心隆重举办,此次展会为期3天,主要围绕“共建现代化慈善,聚力高质量发展”的主题,重点聚焦聚力民生福祉,将打造“一展多元”的展会新格局。本届慈展会共有2931家机构、2882个项目参展。开幕当天,意向对接重大捐赠项目、消费帮扶产品采购及产业投资等资

网络工程师的爬虫技术之路:跨界电商与游戏领域的探索

随着数字化时代的到来,跨界电商和游戏行业成为了网络工程师们充满机遇的领域。这两个领域都依赖于高度复杂的技术来实现商业目标和提供卓越的用户体验。本文将深入探讨网络工程师在跨界电商和游戏领域的技术挑战以及应对这些挑战的方法。突破技术障碍的爬虫应用跨界电商业务通常需要大量的市场数据和竞争情报,而这些信息可能分散在全球各个网站

电机控制器HIL测试系统:实现高效、安全的硬件在环验证

电机控制器作为电动机系统的关键组成部分,其应用前景广泛,涵盖新能源汽车、工业自动化、航空航天、能源、医疗器械等领域。随着对环保、自动化和智能化的需求增加,电机控制器将在这些领域中扮演关键角色,为实现可持续发展和智能化生产提供强有力支持。同时,随着技术的不断进步,电机控制器的性能将得到进一步提升,为各行业提供更高效、可靠

基于SpringBoot的免税商品优选购物商城

目录前言一、技术栈二、系统功能介绍登录界面管理员功能模块商家功能模块用户前台功能模块三、核心代码1、登录模块2、文件上传模块3、代码封装四、文章目录前言随着科学技术的飞速发展,各行各业都在努力与现代先进技术接轨,通过科技手段提高自身的优势,免税商品优选购物商城当然也不能排除在外,随着购物商城的不断成熟,它彻底改变了过去

二叉树链式存储结构

目录1.二叉树链式存储结构2.二叉树的遍历2.1前、中、后序遍历2.2层序遍历3.二叉树的其他递归问题3.1二叉树的结点个数​3.2二叉树的叶子结点个数3.3二叉树第k层结点个数3.4二叉树的深度3.5二叉树查找3.6二叉树销毁4.二叉树的基础OJ题4.1单值二叉树4.2检查两棵树是否相同4.3对称二叉树4.4另一棵树

OpenCV自学笔记十八:模板匹配

模板匹配是一种在图像中寻找指定模式的方法。OpenCV库提供了用于模板匹配的函数,可以帮助我们在图像中定位和识别特定的模式。下面是模板匹配的基础原理和一个示例:模板匹配通过在待匹配图像上滑动一个固定大小的模板图像,并计算模板与图像之间的相似度来寻找匹配位置。相似度通常使用相关性或差异度量来衡量。在滑动过程中,当相似度达

我的创作纪念日-毕业后碰到疫情三年的工作、学习和分享总结

机缘hi,各位朋友好,今天是距离我发布第一篇文章的第1024天,收到了系统私信说分享一下这段时间的收获可以给我一枚勋章。作为一个收集爱好者,决定发布一篇文章获取这枚勋章,同时郑重感谢来查看我的相关文章、给我建议和鼓励我的各位。我开始编辑文章的初衷,是我的记性不太好,学习到的东西很快就会忘记,所以决定将学到的东西详细记录

程序员过不去的坎-算法篇

一个程序员一生中可能会邂逅各种各样的算法,但总有那么几种,是作为一个程序员一定会遇见且大概率需要掌握的算法。今天就来聊聊这些十分重要的“必抓!”算法吧~一:引言算法的重要性和应用场景算法重要性不言而喻,目前的短视频app全部依靠算法进行内容推荐,各大网站也是依靠算法进行内容的推荐。可以说算法已经无处不在。程序本身=算法

【Python爬虫】批量爬取豆瓣电影排行Top250

今天给大家分享下我刚开始接触Python时学习的爬虫程序,代码部分很简单,不过当时刚开始学习时还是走了不少弯路的。这个爬虫程序应该是很多书里面的入门练手程序,主要就是去豆瓣爬取电影评分排行前250。本篇文章只做学习交流使用,不涉及任何商业用途。如果有侵权请联系作者删除。大家在使用Python爬取信息时,请遵守该网站的相

热文推荐