java編程創(chuàng)建型設(shè)計(jì)模式工廠方法模式示例詳解
1.什么是工廠方法模式?
工廠方法模式設(shè)計(jì)方案: 將披薩項(xiàng)目的實(shí)例化功能抽象成抽象方法,在不同的口味點(diǎn)餐子類中具體實(shí)現(xiàn)。
工廠方法模式: 定義了一個(gè)創(chuàng)建對(duì)象的抽象方法,由子類決定要實(shí)例化的類。工廠方法模式將對(duì)象的實(shí)例化推遲到子類。
何時(shí)使用? 不同條件下創(chuàng)建不用實(shí)例時(shí)。方法是讓子類實(shí)現(xiàn)工廠接口。
2.案例實(shí)現(xiàn)
假如說,我們現(xiàn)在有這樣一個(gè)需求:客戶在點(diǎn)披薩時(shí),可以點(diǎn)不同口味的披薩,比如北京的奶酪pizza、北京的胡椒pizza 或者是 倫敦的奶酪pizza、倫敦的胡椒pizza。披薩的制作有prepare,bake, cut, box;完成披薩店訂購功能。
那么可以畫一個(gè)簡(jiǎn)單的類圖如下:

具體的代碼我們就可以這樣寫:
首先是一個(gè)Pizza類,這里將它聲明為抽象類,主要原因是其中的prepare方法不是恒定不變的,而是由實(shí)現(xiàn)它的子類來完成,也就是說制作披薩的原材料是什么怎么做,我們此時(shí)此刻不關(guān)心,那么就可以將它延遲到子類中實(shí)現(xiàn)。
package com.szh.factory.factorymethod.pizza;
/**
* 聲明Pizza類為抽象類
*/
public abstract class Pizza {
//Pizza名稱
protected String name;
//準(zhǔn)備原材料,不同的披薩不一樣。因此,我們做成抽象方法,具體的原材料實(shí)現(xiàn)交給它的子類去完成
public abstract void prepare();
//烘烤
public void bake() {
System.out.println(name + " baking;");
}
//切割
public void cut() {
System.out.println(name + " cutting;");
}
//打包
public void box() {
System.out.println(name + " boxing;");
}
public void setName(String name) {
this.name = name;
}
}package com.szh.factory.factorymethod.pizza;
public class BJCheesePizza extends Pizza {
@Override
public void prepare() {
setName("北京的奶酪pizza");
System.out.println("北京的奶酪pizza 準(zhǔn)備原材料");
}
}package com.szh.factory.factorymethod.pizza;
public class BJPepperPizza extends Pizza {
@Override
public void prepare() {
setName("北京的胡椒pizza");
System.out.println("北京的胡椒pizza 準(zhǔn)備原材料");
}
}package com.szh.factory.factorymethod.pizza;
public class LDCheesePizza extends Pizza{
@Override
public void prepare() {
setName("倫敦的奶酪pizza");
System.out.println("倫敦的奶酪pizza 準(zhǔn)備原材料");
}
}package com.szh.factory.factorymethod.pizza;
public class LDPepperPizza extends Pizza{
@Override
public void prepare() {
setName("倫敦的胡椒pizza");
System.out.println("倫敦的胡椒pizza 準(zhǔn)備原材料");
}
}下面是訂購Pizza的相關(guān)代碼。
package com.szh.factory.factorymethod.order;
import com.szh.factory.factorymethod.pizza.Pizza;
import java.util.Scanner;
public abstract class OrderPizza {
//定義一個(gè)抽象方法, createPizza, 讓各個(gè)工廠子類自己實(shí)現(xiàn)
abstract Pizza createPizza(String orderType);
public OrderPizza() {
Pizza pizza = null;
String orderType = "";
do {
orderType = getType();
pizza = createPizza(orderType); //抽象方法, 由工廠子類完成
// 輸出pizza 制作過程
pizza.prepare();
pizza.bake();
pizza.cut();
pizza.box();
} while (true);
}
private String getType() {
Scanner scanner = new Scanner(System.in);
System.out.println("請(qǐng)輸入 pizza 種類: ");
String str = scanner.nextLine();
return str;
}
}package com.szh.factory.factorymethod.order;
import com.szh.factory.factorymethod.pizza.BJCheesePizza;
import com.szh.factory.factorymethod.pizza.BJPepperPizza;
import com.szh.factory.factorymethod.pizza.Pizza;
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();
} else {
throw new NullPointerException("暫無該P(yáng)izza種類....");
}
return pizza;
}
}package com.szh.factory.factorymethod.order;
import com.szh.factory.factorymethod.pizza.*;
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();
} else {
throw new NullPointerException("暫無該P(yáng)izza種類....");
}
return pizza;
}
}最后我們測(cè)試一下。
package com.szh.factory.factorymethod;
import com.szh.factory.factorymethod.order.BJOrderPizza;
import com.szh.factory.factorymethod.order.LDOrderPizza;
import java.util.Scanner;
public class MainTest {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
String content = scanner.next();
if (content.equals("Beijing")) {
new BJOrderPizza();
} else if (content.equals("London")) {
new LDOrderPizza();
} else {
System.out.println("無法預(yù)先匹配Pizza種類....");
scanner.close();
}
}
}
3.JDK中的工廠方法模式
在 java.util.Calendar 這個(gè)類中就有。

以上就是java編程創(chuàng)建型設(shè)計(jì)模式工廠方法模式示例詳解的詳細(xì)內(nèi)容,更多關(guān)于創(chuàng)建型設(shè)計(jì)模式工廠方法模式的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Mybatis的sql語句執(zhí)行異常后打印到日志問題
文章介紹了一種Mybatis異常日志打印方案,主要通過Mybatis攔截器獲取執(zhí)行的sql語句,并利用ThreadLocal存儲(chǔ),以避免多線程下的sql語句覆蓋問題,當(dāng)異常發(fā)生時(shí),從ThreadLocal中取出sql語句并打印到單獨(dú)的日志文件中,方便數(shù)據(jù)恢復(fù),該方案經(jīng)過壓力測(cè)試2024-10-10
Spring?Boot中application配置文件的生效順序及應(yīng)用范圍
Spring?Boot的一個(gè)重要特性就是它的自動(dòng)配置,這一特性在很大程度上依賴于名稱為application的配置文件,本文將詳細(xì)介紹在Spring?Boot中,這些配置文件的加載順序以及每份文件的應(yīng)用范圍,需要的朋友可以參考下2024-03-03
Springboot實(shí)現(xiàn)發(fā)送郵件及注冊(cè)激活步驟
為了方便郵件發(fā)送功能的使用,我們用郵件發(fā)送功能實(shí)現(xiàn)用戶注冊(cè),實(shí)現(xiàn)步驟大概就是進(jìn)行用戶注冊(cè)同時(shí)發(fā)送一封激活郵件,郵件里附帶激活鏈接,關(guān)于Springboot發(fā)送郵件注冊(cè)激活功能的實(shí)現(xiàn)參考下本文吧2021-06-06
SpringBoot中使用Redis?Stream實(shí)現(xiàn)消息監(jiān)聽示例
本文主要介紹了SpringBoot中使用Redis?Stream實(shí)現(xiàn)消息監(jiān)聽示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06
SpringBoot集成itext實(shí)現(xiàn)html轉(zhuǎn)PDF
iText是著名的開放源碼的站點(diǎn)sourceforge一個(gè)項(xiàng)目,是用于生成PDF文檔的一個(gè)java類庫,本文主要介紹了如何利用itext實(shí)現(xiàn)html轉(zhuǎn)PDF,需要的可以參考下2024-03-03
SpringBoot+Thymeleaf實(shí)現(xiàn)生成PDF文檔
Thymeleaf是一個(gè)現(xiàn)代的服務(wù)器端?Java?模板引擎,適用于?Web?和獨(dú)立環(huán)境。Thymeleaf?的主要目標(biāo)是為您的開發(fā)工作流程帶來優(yōu)雅的自然模板,本文就來用它實(shí)現(xiàn)生成PDF,感興趣的可以了解一下2022-09-09
Springboot?application.yml配置文件拆分方式
這篇文章主要介紹了Springboot?application.yml配置文件拆分方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05

