设计模式 - 创建型模式_原型模式

news2025/1/11 8:01:39

文章目录

  • 创建型模式
  • 概述
  • Case
  • 场景模拟⼯程
  • Bad Impl
  • Better Impl (原型模式重构代码)

在这里插入图片描述


创建型模式

创建型模式提供创建对象的机制, 能够提升已有代码的灵活性和可复⽤性。

类型实现要点
工厂方法定义⼀个创建对象的接⼝,让其⼦类⾃⼰决定实例化哪⼀个⼯⼚类,⼯⼚模式使其创建过程延迟到⼦类进⾏。
抽象工厂提供⼀个创建⼀系列相关或相互依赖对象的接⼝,⽽⽆需指定它们具体的类。
建造者将⼀个复杂的构建与其表示相分离,使得同样的构建过程可以创建不同的表示
原型⽤原型实例指定创建对象的种类,并且通过拷⻉这些原型创建新的对象。
单例保证⼀个类仅有⼀个实例,并提供⼀个访问它的全局访问点。

概述

在这里插入图片描述
原型模式主要解决的问题就是创建重复对象,⽽这部分 对象 内容本身⽐较复杂,⽣成过程可能从库或者RPC接⼝中获取数据的耗时较⻓,因此采⽤克隆的⽅式节省时间。


Case

需要实现⼀个考试抽题的服务,因此在这⾥建造⼀个题库题⽬的场景类信息,⽤于创建; 选择题 、 问答题


场景模拟⼯程

在这里插入图片描述
模拟了两个试卷题⽬的类; ChoiceQuestion (选择题)、 AnswerQuestion (问答题)。

【选择题】

/**
 * 单选题
 */
public class ChoiceQuestion {

    private String name;                 // 题目
    private Map<String, String> option;  // 选项;A、B、C、D
    private String key;                  // 答案;B

    public ChoiceQuestion() {
    }

    public ChoiceQuestion(String name, Map<String, String> option, String key) {
        this.name = name;
        this.option = option;
        this.key = key;
    }

    // set get 
}

【问答题】

public class AnswerQuestion {

    private String name;  // 问题
    private String key;   // 答案

    public AnswerQuestion() {
    }

    public AnswerQuestion(String name, String key) {
        this.name = name;
        this.key = key;
    }

  // set get 
}


Bad Impl

没有⼀个类解决不了的业务,只要你敢写


public class QuestionBankController {

    public String createPaper(String candidate, String number) {

        List<ChoiceQuestion> choiceQuestionList = new ArrayList<ChoiceQuestion>();

        choiceQuestionList.add(new ChoiceQuestion("JAVA所定义的版本中不包括", new HashMap<String, String>() {{
            put("A", "JAVA2 EE");
            put("B", "JAVA2 Card");
            put("C", "JAVA2 ME");
            put("D", "JAVA2 HE");
            put("E", "JAVA2 SE");
        }}, "D"));

        choiceQuestionList.add(new ChoiceQuestion("下列说法正确的是", new HashMap<String, String>() {{
            put("A", "JAVA程序的main方法必须写在类里面");
            put("B", "JAVA程序中可以有多个main方法");
            put("C", "JAVA程序中类名必须与文件名一样");
            put("D", "JAVA程序的main方法中如果只有一条语句,可以不用{}(大括号)括起来");
        }}, "A"));
        choiceQuestionList.add(new ChoiceQuestion("变量命名规范说法正确的是", new HashMap<String, String>() {{
            put("A", "变量由字母、下划线、数字、$符号随意组成;");
            put("B", "变量不能以数字作为开头;");
            put("C", "A和a在java中是同一个变量;");
            put("D", "不同类型的变量,可以起相同的名字;");
        }}, "B"));
        choiceQuestionList.add(new ChoiceQuestion("以下()不是合法的标识符", new HashMap<String, String>() {{
            put("A", "STRING");
            put("B", "x3x;");
            put("C", "void");
            put("D", "de$f");
        }}, "C"));
        choiceQuestionList.add(new ChoiceQuestion("表达式(11+3*8)/4%3的值是", new HashMap<String, String>() {{
            put("A", "31");
            put("B", "0");
            put("C", "1");
            put("D", "2");
        }}, "D"));

        List<AnswerQuestion> answerQuestionList = new ArrayList<AnswerQuestion>();
        answerQuestionList.add(new AnswerQuestion("小红马和小黑马生的小马几条腿", "4条腿"));
        answerQuestionList.add(new AnswerQuestion("铁棒打头疼还是木棒打头疼", "头最疼"));
        answerQuestionList.add(new AnswerQuestion("什么床不能睡觉", "牙床"));
        answerQuestionList.add(new AnswerQuestion("为什么好马不吃回头草", "后面的草没了"));

        // 输出结果
        StringBuilder detail = new StringBuilder("考生:" + candidate + "\r\n" +
                "考号:" + number + "\r\n" +
                "--------------------------------------------\r\n" +
                "一、选择题" + "\r\n\n");

        for (int idx = 0; idx < choiceQuestionList.size(); idx++) {
            detail.append("第").append(idx + 1).append("题:").append(choiceQuestionList.get(idx).getName()).append("\r\n");
            Map<String, String> option = choiceQuestionList.get(idx).getOption();
            for (String key : option.keySet()) {
                detail.append(key).append(":").append(option.get(key)).append("\r\n");;
            }
            detail.append("答案:").append(choiceQuestionList.get(idx).getKey()).append("\r\n\n");
        }

        detail.append("二、问答题" + "\r\n\n");

        for (int idx = 0; idx < answerQuestionList.size(); idx++) {
            detail.append("第").append(idx + 1).append("题:").append(answerQuestionList.get(idx).getName()).append("\r\n");
            detail.append("答案:").append(answerQuestionList.get(idx).getKey()).append("\r\n\n");
        }

        return detail.toString();
    }

}

  • 这样的代码往往都⾮常易于理解,要什么程序就给什么代码,不⾯向对象,只⾯向过程。不考虑扩展性,能⽤就⾏
  • 以上的代码主要就三部分内容;⾸先创建选择题和问答题到集合中、定义详情字符串包装结果、返回结果内容。
  • 但以上的代码有⼀个没有实现的地⽅就是不能乱序,所有⼈的试卷顺序都是⼀样的。如果需要加乱序也是可以的,但复杂度⼜会增加。这

【单元测试】
通过junit单元测试的⽅式验证接⼝服务,强调⽇常编写好单测可以更好的提⾼系统的健壮度。

    @Test
    public void test_QuestionBankController() {
        QuestionBankController questionBankController = new QuestionBankController();
        System.out.println(questionBankController.createPaper("花花", "1000001921032"));
        System.out.println(questionBankController.createPaper("豆豆", "1000001921051"));
        System.out.println(questionBankController.createPaper("大宝", "1000001921987"));
    }

输出

考生:花花
考号:1000001921032
--------------------------------------------
一、选择题

第1题:JAVA所定义的版本中不包括
AJAVA2 EE
BJAVA2 Card
CJAVA2 ME
DJAVA2 HE
EJAVA2 SE
答案:D2题:下列说法正确的是
AJAVA程序的main方法必须写在类里面
BJAVA程序中可以有多个main方法
CJAVA程序中类名必须与文件名一样
DJAVA程序的main方法中如果只有一条语句,可以不用{}(大括号)括起来
答案:A3题:变量命名规范说法正确的是
A:变量由字母、下划线、数字、$符号随意组成;
B:变量不能以数字作为开头;
CA和a在java中是同一个变量;
D:不同类型的变量,可以起相同的名字;
答案:B4题:以下()不是合法的标识符
ASTRING
B:x3x;
Cvoid
D:de$f
答案:C5题:表达式(11+3*8)/4%3的值是
A31
B0
C1
D2
答案:D

二、问答题

第1题:小红马和小黑马生的小马几条腿
答案:4条腿

第2题:铁棒打头疼还是木棒打头疼
答案:头最疼

第3题:什么床不能睡觉
答案:牙床

第4题:为什么好马不吃回头草
答案:后面的草没了


考生:豆豆
考号:1000001921051
--------------------------------------------
一、选择题

第1题:JAVA所定义的版本中不包括
AJAVA2 EE
BJAVA2 Card
CJAVA2 ME
DJAVA2 HE
EJAVA2 SE
答案:D2题:下列说法正确的是
AJAVA程序的main方法必须写在类里面
BJAVA程序中可以有多个main方法
CJAVA程序中类名必须与文件名一样
DJAVA程序的main方法中如果只有一条语句,可以不用{}(大括号)括起来
答案:A3题:变量命名规范说法正确的是
A:变量由字母、下划线、数字、$符号随意组成;
B:变量不能以数字作为开头;
CA和a在java中是同一个变量;
D:不同类型的变量,可以起相同的名字;
答案:B4题:以下()不是合法的标识符
ASTRING
B:x3x;
Cvoid
D:de$f
答案:C5题:表达式(11+3*8)/4%3的值是
A31
B0
C1
D2
答案:D

二、问答题

第1题:小红马和小黑马生的小马几条腿
答案:4条腿

第2题:铁棒打头疼还是木棒打头疼
答案:头最疼

第3题:什么床不能睡觉
答案:牙床

第4题:为什么好马不吃回头草
答案:后面的草没了


考生:大宝
考号:1000001921987
--------------------------------------------
一、选择题

第1题:JAVA所定义的版本中不包括
AJAVA2 EE
BJAVA2 Card
CJAVA2 ME
DJAVA2 HE
EJAVA2 SE
答案:D2题:下列说法正确的是
AJAVA程序的main方法必须写在类里面
BJAVA程序中可以有多个main方法
CJAVA程序中类名必须与文件名一样
DJAVA程序的main方法中如果只有一条语句,可以不用{}(大括号)括起来
答案:A3题:变量命名规范说法正确的是
A:变量由字母、下划线、数字、$符号随意组成;
B:变量不能以数字作为开头;
CA和a在java中是同一个变量;
D:不同类型的变量,可以起相同的名字;
答案:B4题:以下()不是合法的标识符
ASTRING
B:x3x;
Cvoid
D:de$f
答案:C5题:表达式(11+3*8)/4%3的值是
A31
B0
C1
D2
答案:D

二、问答题

第1题:小红马和小黑马生的小马几条腿
答案:4条腿

第2题:铁棒打头疼还是木棒打头疼
答案:头最疼

第3题:什么床不能睡觉
答案:牙床

第4题:为什么好马不吃回头草
答案:后面的草没了


  • 是三位考试的试卷; 花花 、 ⾖⾖ 、 ⼤宝 ,每个⼈的试卷内容是⼀样的这没问题,但是三个⼈的题⽬以及选项顺序都是⼀样,就没有达到我们说希望的乱序要求。
  • 以上这样的代码⾮常难扩展,随着题⽬的不断的增加以及乱序功能的补充,都会让这段代码变得越来越混乱。

Better Impl (原型模式重构代码)

接下来使⽤原型模式来进⾏代码优化,也算是⼀次很⼩的重构。

原型模式主要解决的问题就是创建⼤量重复的类,⽽我们模拟的场景就需要给不同的⽤户都创建相同的试卷,但这些试卷的题⽬不便于每次都从库中获取,甚⾄有时候需要从远程的RPC中获取。这样都是⾮常耗时的,⽽且随着创建对象的增多将严重影响效率。

在原型模式中所需要的⾮常重要要的⼿段就是克隆,在需要⽤到克隆的类中都需要实现 implements Cloneable 接⼝

【⼯程结构】

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/180782.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

AcWing蓝桥杯AB组辅导课08、数论

文章目录前言一、数论例题例题1&#xff1a;AcWing 1246. 等差数列&#xff08;最大公约数&#xff0c;第十届蓝桥杯省赛CB第7题&#xff09;分析题解&#xff1a;最大公约数例题2&#xff1a;AcWing 1295. X的因子链&#xff08;算数基本定理、欧拉筛选&#xff0c;多重集合排…

打工人必知必会(四)——股票期权属于劳动争议吗

目录 参考 一、核心概览 二、 注意 三、更多案例 参考 案例评析:股票期权纠纷是否属于劳动争议 股票期权&#xff08;限制性股票&#xff09;相关劳动争议问题 北上广深杭案例 一、核心概览 二、 注意 结合双方股权激励协议的签署背景、目的等因素来综合考量并作出相对准…

【数据结构】8.1 排序概述

文章目录排序的基本概念排序方法的分类存储结构排序的基本概念 什么是排序&#xff1f; 排序&#xff1a;将一组杂乱无章的数据按照一定规律顺次排列起来。 即&#xff0c;讲无序序列排成一个有序序列&#xff08;有小到大或由大到小&#xff09;的运算。 如果参加排序的数据…

MicroPython开发ESP8266——环境搭建

MicroPython开发ESP8266——环境搭建0.前言一、固件烧写1.使用乐鑫官方的烧写工具2.使用python中的esptool工具烧写3.使用uPyCraft烧写4.测试二、IDE工具安装1.windows环境搭建2.Linux环境搭建3.测试1&#xff09;软件设置2&#xff09;测试程序3&#xff09;烧录0.前言 最近刷…

NLP 语种检测 API 数据接口

NLP 语种检测 API 数据接口 180 语言检测&#xff0c;语种全称与缩写&#xff0c;返回置信度。 1. 产品功能 基于 NLP 分析文本的语种支持 180 多语种检测&#xff1b;语种缩写遵循 ISO 639-1 标准&#xff1b;包含检测到的语种置信度&#xff1b;毫秒级响应性能&#xff1b;…

Java 代码ccflow 代码分析

流程属性目录概述需求&#xff1a;设计思路实现思路分析1。代码流程参考资料和推荐阅读Survive by day and develop by night. talk for import biz , show your perfect code,full busy&#xff0c;skip hardness,make a better result,wait for change,challenge Survive. ha…

Linux中详解编译原理每一步

引言&#xff1a; 时间过的飞快&#xff0c;可以看出我们已经开始学习Linux了&#xff0c;但是我们学习Linux过程中&#xff0c;依然会涉及到很多的以前的知识&#xff0c;比如编译原理&#xff0c;我们的代码如何从一个源文件逐步变成一个可执行文件&#xff0c;当初我记得我…

流批一体计算引擎-8-[Flink]的Table API连接器

参考官方文档Table API连接器 1 Table API连接器概述 Flink的Table API和SQL程序可以连接到其他外部系统&#xff0c;用于读取和写入批处理表和流式表。source表提供对存储在外部系统&#xff08;如数据库、键值存储、消息队列或文件系统&#xff09;中的数据的访问。sink表将…

Sentienl学习笔记

PS&#xff1a;本文为作者学习黑马程序员Springcould视频笔记实际技术参考价值不大&#xff0c;文章将持续更新。 文章目录一. 什么是Sentienl1. 介绍2. 与Hystrix对比3. 主要特性二. Sentienl安装配置1. 下载安装包2. 启动三. Sentienl的使用1. Sentienl的整合2. 簇点链路四. …

每日学术速递1.27

CV - 计算机视觉 | ML - 机器学习 | RL - 强化学习 前沿推介&#xff1a; ICLR 2023 ICLR 全称为国际学习表征会议&#xff08;International Conference on Learning Representations&#xff09;&#xff0c;今年将举办的是第 11 届&#xff0c;预计将于 5 月 1 日至 5 …

Redis实现附近商铺 | 黑马点评

一、GEO数据结构 1、入门 GEO是Geolocation的缩写&#xff0c;代表地理坐标。redis3.2中加入对GEO的支持&#xff0c;允许存储地理坐标信息&#xff0c;帮助我们根据经纬度来检索数据。 常见命令&#xff1a; GEOADD&#xff1a;添加一个地理空间信息&#xff0c;包含&…

springcloud3 Sentinel的服务熔断操作

一 服务熔断 1.1 服务熔断 Sentinel熔断降级会在调用链路中某个资源出现不稳定状态时&#xff08;调用超时或者异常比例升高&#xff09;&#xff0c;对这个资源的调用进行限制&#xff0c;让请求快速失败&#xff0c;避免影响到其他资源进而导致级联错误。 当资源被降级后&…

07_linux中断控制

裸机开发要点 通用中断控制器(GIC) 中断类型、硬件中断号、分发器和cpu接口单元 中断向量表 一级查表、二级查表 中断处理流程 进入irq模式、保护现场、获取硬件中断编号、执行中断处理函数、还原现场 设备树构造 分为 gic中断控制器设备树节点 其他外设中断控制器节点 需要…

大数据相关组件

一、 HDFSHDFS是hadoop的核心组件&#xff0c;HDFS上的文件被分成块进行存储&#xff0c;默认块的大小是64M&#xff0c;块是文件存储处理的逻辑单元。HDFS是Master和Slave的结构。分NameNode、SecondaryNameNode、DataNode这几个角色。NameNode&#xff1a;是Master节点&#…

springcloud3 Sentinel的搭建以及案例操作

一 sentinel的概念 1.1 sentinel Sentinel是分布式系统流量控制的哨兵&#xff0c;阿里开源的一套服务容错的综合性解决方案。 主要用来处理&#xff1a; 服务降级 服务熔断 超时处理 流量控制 sentinel 的使用可以分为两个部分: 核心库&#xff08;Java 客户端&#…

基于nodejs+vue的留学服务管理平台的设计与开发

目 录 摘 要 I Abstract I 第一章 绪论 1 1.1系统开发的背景 1 1.2系统开发的意义 1 1.3本文研究内容 2 第二章 系统开发技术 3 第三章 系统分析 6 3.1用户需求分析 6 3.1.1 老师用户 6 3.1.2 学生用户 6 3.1.3 管理员用户 6 3.2 系统…

6--总线

文章目录一.总线概述&#xff08;一&#xff09;总线特性&#xff08;二&#xff09;总线分类1.按功能分/按连接的部件分&#xff08;1&#xff09;片内总线/CPU内部总线&#xff08;2&#xff09;系统总线&#xff08;3&#xff09;通信总线/外部总线2.按数据传输格式分&#…

7、关系运算符与关系表达式

目录 一、关系运算符 二、关系表达式 三、优先级与结合性 一、关系运算符 关系运算符包括大于、大于等于、小于、小于等于、等于和不等于 注意&#xff1a;符号“>”&#xff08;大于等于&#xff09;与“<”&#xff08;小于等于&#xff09;的意思分别是大于或等于…

向QAbstractItemView子类如:QTreeView、QTableView等子项单元格插入窗体小部件的功能实现(第3种方法)

1.前言工作中经常会遇到这样的需求&#xff1a;向QAbstractItemView子类如QTreeView、QTableView单元格插入窗体小部件&#xff0c;如&#xff1a;进度条、按钮、单行编辑框等。下面链接的系列博文就是讲解如何实现该功能的。《向QAbstractItemView子类如:QTreeView、QTableVie…

Java 23种设计模式(9.结构型模式-外观模式)

结构型模式-外观模式 代码详解 类图 代码 public class SubOne {public void method1(){System.out.println("method1");} }public class SubTwo {public void method2(){System.out.println("method2");} }public class SubThree {public void method3(…