【再探】设计模式—抽象工厂及建造者模式

devtools/2024/10/21 9:57:11/

 抽象工厂模式建造者模式都属于创建型模式。两者都能创建对应的对象,而创建者模式更侧重于创建复杂对象,将对象的创建过程封装起来,让客户端不需要知道对象的内部细节。

1 抽象工厂模式

需求:

  1. 在使用工厂方法模式时,当增加新的产品类型时,需要创建对应的Factory类,这导致类的数量增加,让系统变得臃肿。
  2. 需要保证同风格下生成一致的组件。比如在写页面时,有dark及light模式,我们希望button、text等组件能在不同模式下保持一致的风格。抽象工厂模式介绍

1.1 抽象工厂模式介绍

为创建一组对象提供一组解决方案。与工厂模式相比,抽象工厂模式中的具体工厂不只是创建一种产品,而是创建一族产品。

抽象工厂模式UML

public class Button {private final String color;public Button(String color) {this.color = color;}@Overridepublic String toString() {return getClass().getSimpleName() + "{" +"color='" + color + '\'' +'}';}
}public class Text {private final String color;public Text(String color) {this.color = color;}@Overridepublic String toString() {return getClass().getSimpleName() + "{" +"color='" + color + '\'' +'}';}
}public class DarkButton extends Button{public DarkButton(String color) {super(color);}
}public class DarkText extends Text{public DarkText(String color) {super(color);}
}public class LightButton extends Button{public LightButton(String color) {super(color);}
}public class LightText extends Text{public LightText(String color) {super(color);}
}
public interface ComponentsFactory {Button buildButton();Text buildText();}public class DarkComponentFactory implements ComponentsFactory{private final String color = "dark";@Overridepublic Button buildButton() {return new DarkButton(color);}@Overridepublic Text buildText() {return new DarkText(color);}}public class LightComponentFactory implements ComponentsFactory{private final String color = "light";@Overridepublic Button buildButton() {return new LightButton(color);}@Overridepublic Text buildText() {return new LightText(color);}
}

 

public class HtmlWeb {public static void main(String[] args) {ComponentsFactory componentsFactory = new DarkComponentFactory();System.out.println("-----------dark模式-----------");System.out.println(componentsFactory.buildButton());System.out.println(componentsFactory.buildText());System.out.println("-----------light模式-----------");componentsFactory = new LightComponentFactory();System.out.println(componentsFactory.buildButton());System.out.println(componentsFactory.buildText());}
}

1.1.1 优缺点

优点:

  1. 隔离类的实例化过程,使得客户端并不需要知道如何创建对象。
  2. 当不同产品族多个对象被设计成一起工作时,能保证客户端始终只使用同一个产品族的对象。
  3. 减少了工厂类的数量。
  4. 增加新的产品族方便,符合开闭原则

缺点:

  1. 不符合单一职责原则,一个类创建了多个对象。
  2. 族群增加新的种类时,不符合开闭原则,需要修改全部的工厂类。

2 建造者模式

需求:

  1. 需要创建一个复杂的对象,隔离类的实例化过程,使得客户端不需要知道对象的内部细节。
  2. 创建对象时,对执行顺序有要求。

2.1 建造者模式介绍

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

图 构建模式UML

Director 为指挥者,用来控制Buider 的执行顺序,在实际开发中,这个角色经常会被省略,它的功能集成到Builder上;而Builder中的builderPart方法的返回值为它本身,这样就可以链式调用了。

public class JdbcConnector {private JdbcConnection connection;public void initConnection(URI uri) {connection = new JdbcConnection(uri);}public JdbcPreparedStatement getPreparedStatement(String query) {if (connection == null) throw new RuntimeException("连接器还未初始化");return connection.getPreparedStatement(query);}public static class JdbcConnection {public JdbcConnection(URI uri){System.out.println(uri);}public JdbcPreparedStatement getPreparedStatement(String query) {return new JdbcPreparedStatement(query);}}public static class JdbcPreparedStatement {String query;public JdbcPreparedStatement(String query) {this.query = query;}public String getResult() {return this.query;}}}public class JdbcConnectorBuilder {private String scheme;private String host;private String port;private String database;private String username;private String password;public JdbcConnectorBuilder setScheme(String scheme) {this.scheme = scheme;return this;}public JdbcConnectorBuilder setHost(String host) {this.host = host;return this;}public JdbcConnectorBuilder setPort(String port) {this.port = port;return this;}public JdbcConnectorBuilder setDatabase(String database) {this.database = database;return this;}public JdbcConnectorBuilder setUsername(String username) {this.username = username;return this;}public JdbcConnectorBuilder setPassword(String password) {this.password = password;return this;}/*** 产品*/private JdbcConnector jdbcConnector = null;public JdbcConnectorBuilder() {jdbcConnector = new JdbcConnector();}public JdbcConnector build() {StringBuilder sb = new StringBuilder("jdbc:");sb.append(scheme == null ? "mysql" : scheme).append("://");if (host == null) throw new RuntimeException("host不能为空");sb.append(host);sb.append(":").append(port == null ? "3306" : port);if (database == null) throw new RuntimeException("数据库不能为空");sb.append("/").append(database);try {URI uri = new URI(sb.toString());if (username == null || password == null) throw new RuntimeException("账号或密码错误");jdbcConnector.initConnection(uri);} catch (URISyntaxException e) {throw new RuntimeException("连接失败,连接信息有误");}return jdbcConnector;}}public class JdbcConnectionTest {public static void main(String[] args) {JdbcConnectorBuilder builder = new JdbcConnectorBuilder();builder.setScheme("mysql").setHost("localhost").setPort("3307").setDatabase("my-database").setUsername("root").setPassword("123456");JdbcConnector jdbcConnector = builder.build();JdbcConnector.JdbcPreparedStatement statement = jdbcConnector.getPreparedStatement("select * from student");System.out.println(statement.getResult());}
}

JDK 中的StringBuilder及Apache httpclient 的 URIBuilder 应用了建造者模式

自定义StringBuilder

public class CustomStringBuilder {public static void main(String[] args) {CustomStringBuilder customStringBuilder = new CustomStringBuilder("hello customStringBuilder");for (int i =0; i < 100; i++) {customStringBuilder.append("---建造者模式---").append(i + "");}System.out.println(customStringBuilder);}char[] value;int count = 0;public CustomStringBuilder(int capacity) {value = new char[capacity];}public CustomStringBuilder(String str) {this(str.length() + 16);append(str);}CustomStringBuilder append(String str) {if (str == null) {throw new RuntimeException("字符串不能为空");}ensureCapacity(count + str.length());str.getChars(0,str.length(),value,count);count += str.length();return this;}private void ensureCapacity(int minCapacity) {if (minCapacity >= value.length) {value = Arrays.copyOf(value,value.length << 1);}}@Overridepublic String toString() {return new String(value,0,count);}
}

自定义URIBuilder

public class CustomURIBuilder {public static void main(String[] args) throws URISyntaxException {CustomURIBuilder customURIBuilder = new CustomURIBuilder();customURIBuilder.setHost("localhost").setPort("8080").setEncodedPath("/userinfo");customURIBuilder.appendQueryParam("username","hmf");customURIBuilder.appendQueryParam("status","1");customURIBuilder.appendQueryParam("createDate","2024/04/20");URI uri = customURIBuilder.build();System.out.println(uri);}private String scheme;private String host;private String port;private String encodedPath;private List<NameValuePair> queryParams;public CustomURIBuilder setScheme(String scheme) {this.scheme = scheme;return this;}public CustomURIBuilder setHost(String host) {this.host = host;return this;}public CustomURIBuilder setPort(String port) {this.port = port;return this;}public CustomURIBuilder setEncodedPath(String encodedPath) {this.encodedPath = encodedPath;return this;}public CustomURIBuilder appendQueryParam(String name,String value) {if (queryParams == null) queryParams = new ArrayList<>();queryParams.add(new NameValuePair(name,value));return this;}public URI build() throws URISyntaxException {return new URI(buildStr());}public String buildStr() {StringBuilder sb = new StringBuilder();if (this.scheme != null) sb.append(this.scheme).append("://");if (this.host == null) throw new RuntimeException("host不能为空");sb.append(this.host);if (this.port != null) sb.append(":").append(this.port);if (this.encodedPath != null) sb.append(this.encodedPath);if (this.queryParams != null) {for (int i = 0; i < this.queryParams.size(); i++) {sb.append(i == 0 ? "?" : "&").append(this.queryParams.get(i));}}return sb.toString();}private static class NameValuePair {String name;String value;public NameValuePair(String name, String value) {this.name = name;this.value = value;}@Overridepublic String toString() {if (this.value == null) return name;return name + "=" + value;}}}

2.1.1 优缺点

优点:

  1. 封装性好,将产品的内部表示与产品的生成过程分割开。客户端只需知道所需的产品类型,而不需要知道产品内部的具体结构和实现细节。
  2. 扩展性好,各个具体建造者相互独立,如果需要添加新的产品,只需创建对应的建造者即可,符合开闭原则
  3. 便于构建发展对象,可以简化对象的创建过程,提高代码的可读性和可维护性。

缺点:

  1. 可维护性差,如果产品的内部发生变化,则对应的创建者可能需要修改的地方会比较多。
  2. 类的数量增加。

http://www.ppmy.cn/devtools/32237.html

相关文章

2013NOIP普及组真题 4. 车站分级

线上OJ&#xff1a; 一本通&#xff1a;http://ybt.ssoier.cn:8088/problem_show.php?pid1964 核心思想&#xff1a; 1、原文中提到 “如果这趟车次停靠了火车站 x&#xff0c;则始发站、终点站之间所有级别大于等于火车站 x 的都必须停靠”&#xff0c;如果设停靠站为A&…

《Beginning C++20 From Novice to Professional》第七章Working with Strings

字符串处理是非常令人关注的领域&#xff0c;因为大部分情况下我们的程序不是在处理数字而是在处理字符串&#xff0c;对于字符串的表示和操作成为编程语言中非常重要的一部分 书里也强调C中对于字符串的处理要好过C风格的char数组&#xff0c;更高效也更安全 本章我们可以学…

Matlab : unique函数的用法

unique是MATLAB中一个非常实用的函数&#xff0c;用于找出数组中的唯一值。它有几种不同的用法&#xff0c;可以根据需要选择合适的参数来控制输出的行为。以下是unique函数的基本用法和一些常见参数的介绍&#xff1a; 基本语法&#xff1a; C unique(A) [C,IA,IC] unique…

ATTCK的优缺点分别是什么

ATT&CK(Adversarial Tactics, Techniques, and Common Knowledge)框架是一个广泛使用的资源,它提供了对网络威胁的深入洞察,特别是关于攻击者可能采取的战术、技术和程序(TTPs)。以下是ATT&CK框架的优缺点: 优点: 全面的威胁情报:ATT&CK框架详细描述了各种…

redis集群的创建安装与配置,以及维护

一、redis集群配置 port 7000 cluster-enabled yes cluster-config-file nodes.conf cluster-node-timeout 5000 appendonly yes 二、安装ruby&#xff08;ruby可以充当redis客户端的角色&#xff09; yum -y install ruby ruby-devel rubygems rpm-build gem install …

Big Data 平障录

Hive Hive 生成带压缩的格式&#xff0c;需要如此设置 SET parquet.compressionSNAPPY;yarn.scheduler.fair.assignmultiple 相关jira&#xff1a;https://issues.apache.org/jira/browse/YARN-5035 yarn.scheduler.fair.assignmultiple是YARN Fair Scheduler的一个配置参数…

人大金仓数据库的数据库系统的技术架构

人大金仓数据库作为中国社会科学院经济研究所主办的重要数据库平台&#xff0c;其数据库系统技术架构设计合理、稳定高效&#xff0c;为用户提供了丰富的数据资源和强大的数据分析功能。下面我将详细介绍人大金仓数据库的数据库系统技术架构。 1. 数据库系统概述 人大金仓数据…

Altium Designer入门基础操作

软件下载环境搭建&#xff1a;pan.baidu.com/s/1HshgKTmkkBpbIRa-9Wq9cQ 密码&#xff1a;ckck 工程建立&#xff1a; 创建 库绘制 为什么管脚要100mil 元素10mil 原理图库得正确性报告 原理图页设置大小&#xff0c;标准自定义&#xff0c;格点为100mil 使用库画原理图&a…