【Java设计模式】用 披萨订购案例 详细讲解三种工厂模式

本文介绍了如何使用Java模拟披萨制作过程,通过类的继承和多态展示抽象工厂和工厂方法模式的应用,探讨了OCP原则在扩展性上的局限,并提供了不同设计模式在实现订购披萨逻辑中的优缺点比较。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

System.out.println(name + " cutting");

}

public void box() {

System.out.println(name + " boxing");

}

public void setName(String name) {

this.name = name;

}

}

分析: 这个类用来代表制作披萨的整个流程:准备阶段prepare()烘烤阶段bake()切割阶段cut()打包阶段box() ,假设各个披萨的准备阶段需要的材料不一样,所以把准备阶段定义为一个抽象方法 ,其它三个阶段都一样。

步骤二:创建两个披萨类

public class CheesePizza extends Pizza{

@Override

public void prepare() {

System.out.println(“奶酪披萨正在准备中”);

}

}

分析: 这个类代表奶酪披萨,简单重写一下准备阶段

public class BeefPizza extends Pizza{

@Override

public void prepare() {

System.out.println(“牛肉披萨正在准备中”);

}

}

分析: 这个类代表牛肉披萨,也简单重写一下准备阶段

步骤三:制定订购披萨类

public class OrderPizza {

public OrderPizza() {

Pizza pizza = null;

do {

String pizzaType = getType();

if (“cheese”.equalsIgnoreCase(pizzaType)) {

pizza = new CheesePizza();

pizza.setName(“cheese”);

} else if (“beef”.equalsIgnoreCase(pizzaType)) {

pizza = new BeefPizza();

pizza.setName(“beef”);

} else {

break;

}

pizza.prepare();

pizza.bake();

pizza.cut();

pizza.box();

} while (true);

}

// 写一个方法,可以获取希望订购的披萨种类

private String getType() {

try {

BufferedReader strin = new BufferedReader(new InputStreamReader(System.in));

System.out.println(“input pizza 种类:”);

String str = strin.readLine();

return str;

} catch (IOException e) {

e.printStackTrace();

return “”;

}

}

}

分析: 慢慢看代码,可以明白,订购披萨的逻辑代码写在了该类的构造器中,getType()方法是用来获取希望订购的披萨种类。但是,如果我们需要添加新的披萨,就需要从这个类中继续添加相应的逻辑语句,从而修改了订购披萨的这个类,就违反了OCP原则

步骤四:创建运行类

public class PizzaStore {

public static void main(String[] args) {

new OrderPizza();

}

}

运行结果:

在这里插入图片描述

优缺点分析


  • 优点:比较好理解,简单易操作

  • 缺点:违反了设计模式的ocp原则,即对扩展开放,对修改关闭。即当我们给类增加新功能的时候,尽量不修改代码,或者尽可能少修改代码

二、使用简单工厂

===========================================================================

类图


在这里插入图片描述

基本介绍


  • 简单工厂模式是属于创建型模式,是工厂模式的一种。简单工厂模式是由一个工厂对象决定创建出哪一种产品类的实例。简单工厂模式是工厂模式家族中最简单实用的模式

  • 简单工厂模式定义了一个创建对象的类,由这个类来封装实例化对象的行为(代码)

  • 在软件开发中,当我们会用到大量的创建某种、某类或者某批对象时,就会使用到工厂模式

步骤概括


步骤一:创建简单工厂

public class SimpleFactory {

public static Pizza createPizza2(String orderType) {

Pizza pizza = null;

if (“beef”.equalsIgnoreCase(orderType)) {

pizza = new BeefPizza();

pizza.setName(" beef ");

} else if (“cheese”.equalsIgnoreCase(orderType)) {

pizza = new CheesePizza();

pizza.setName(“cheese”);

}

return pizza;

}

}

分析: 简单工厂又叫做静态工厂,我们写一个静态方法,可以方便后面代码的调用,这里用到的类,跟用传统方法用到的类一样,没有改变

步骤二:制定订购披萨类

public class OrderPizza2 {

public OrderPizza2() {

do {

String orderType = getType();

Pizza pizza = SimpleFactory.createPizza2(orderType);

if (pizza != null) {

pizza.prepare();

pizza.bake();

pizza.cut();

pizza.box();

} else {

System.out.println(" 订购披萨失败 ");

break;

}

} while (true);

}

private String getType() {

try {

BufferedReader strin = new BufferedReader(new InputStreamReader(System.in));

System.out.println(“input pizza 种类:”);

String str = strin.readLine();

return str;

} catch (IOException e) {

e.printStackTrace();

return “”;

}

}

}

分析: 该类的构造器中用到了简单工厂类SimpleFactory,这样,我们就不需要在订购披萨这一行为中去写增加新的披萨的代码了,而是从工厂中写添加新的披萨的代码,就不用再改动这个类。

运行结果:

在这里插入图片描述

优点分析


  • 使用简单工厂模式来创建对象,更加的方便灵活,不需要修改订购披萨的逻辑

三、使用工厂方法

===========================================================================

新的需求


客户在点披萨时,可以点不同口味的披萨,比如 北京的奶酪pizza、北京的胡椒pizza 或者是伦敦的奶酪pizza、伦敦的胡椒pizza

类图


在这里插入图片描述

基本介绍


  • 工厂方法模式设计方案:将披萨项目的实例化功能抽象成抽象方法,在不同的口味点餐子类中具体实现。

  • 工厂方法模式:定义了一个创建对象的抽象方法,由子类决定要实例化的类。工厂方法模式将对象的实例化推迟到子类。

步骤概括


步骤一:创建四个披萨类

public class BJCheesePizza extends Pizza {

@Override

public void prepare() {

setName(“北京的奶酪pizza”);

System.out.println(“北京的奶酪pizza 准备原材料”);

}

}

分析: Pizza类跟上面的代码一样,我就没有再次写了。此类是用来创建北京的奶酪口味的披萨

public class BJPepperPizza extends Pizza{

@Override

public void prepare() {

setName(“北京的胡椒pizza”);

System.out.println(“北京的胡椒pizza 准备原材料”);

}

}

分析: 此类是用来创建北京的辣椒口味的披萨

public class LDCheesePizza extends Pizza {

@Override

public void prepare() {

setName(“伦敦的奶酪pizza”);

System.out.println(“伦敦的奶酪pizza 准备原材料”);

}

}

分析: 此类是用来创建伦敦的奶酪口味的披萨

public class LDPepperPizza extends Pizza {

@Override

public void prepare() {

setName(“伦敦的胡椒pizza”);

System.out.println(“伦敦的胡椒pizza 准备原材料”);

}

}

分析: 此类是用来创建伦敦的辣椒口味的披萨

步骤二:创建订购披萨抽象类

public abstract class OrderPizza {

abstract Pizza createPizza(String orderType);

public OrderPizza() {

do {

String orderType = getType();

Pizza pizza = createPizza(orderType); //抽象方法,由工厂子类完成

if (pizza == null){

System.out.println(“订购披萨失败”);

break;

}

pizza.prepare();

pizza.bake();

pizza.cut();

pizza.box();

} while (true);

}

private String getType() {

try {

BufferedReader strin = new BufferedReader(new InputStreamReader(System.in));

System.out.println(“input pizza 种类:”);

String str = strin.readLine();

return str;

} catch (IOException e) {

e.printStackTrace();

return “”;

}

}

}

分析: 此类中定义一个抽象方法createPizza(), 让各个工厂子类自己实现,构造器中写订购披萨的代码逻辑;getType()方法跟原来的没有区别。

public class BJOrderPizza extends OrderPizza {

@Override

Pizza createPizza(String orderType) {

Pizza pizza = null;

if(orderType.equals(“cheese”)) {

pizza = new BJCheesePizza();

} else if (orderType.equals(“pepper”)) {

pizza = new BJPepperPizza();

}

return pizza;

}

}

分析: 此类用来继承OrderPizza类,成为北京地区的订购披萨分销商

public class LDOrderPizza extends OrderPizza {

@Override

Pizza createPizza(String orderType) {

Pizza pizza = null;

if(orderType.equals(“cheese”)) {

pizza = new LDCheesePizza();

} else if (orderType.equals(“pepper”)) {

pizza = new LDPepperPizza();

}

return pizza;

}

}

分析: 此类也用来继承OrderPizza类,成为伦敦地区的订购披萨分销商

步骤三:创建运行类

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

最后

Java架构进阶面试及知识点文档笔记

这份文档共498页,其中包括Java集合,并发编程,JVM,Dubbo,Redis,Spring全家桶,MySQL,Kafka等面试解析及知识点整理

image

Java分布式高级面试问题解析文档

其中都是包括分布式的面试问题解析,内容有分布式消息队列,Redis缓存,分库分表,微服务架构,分布式高可用,读写分离等等!

image

互联网Java程序员面试必备问题解析及文档学习笔记

image

Java架构进阶视频解析合集
《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!
,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!**

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

最后

Java架构进阶面试及知识点文档笔记

这份文档共498页,其中包括Java集合,并发编程,JVM,Dubbo,Redis,Spring全家桶,MySQL,Kafka等面试解析及知识点整理

[外链图片转存中…(img-8FPNXRWk-1712072526645)]

Java分布式高级面试问题解析文档

其中都是包括分布式的面试问题解析,内容有分布式消息队列,Redis缓存,分库分表,微服务架构,分布式高可用,读写分离等等!

[外链图片转存中…(img-LaBKYz67-1712072526645)]

互联网Java程序员面试必备问题解析及文档学习笔记

[外链图片转存中…(img-LVuDgsuj-1712072526646)]

Java架构进阶视频解析合集
《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值