成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

沒事最好不要用工廠模式

開發 前端
如果我們還是用工廠模式來實現的話,那每一種都要編寫一個工廠類,過多的類會難以維護,那怎么解決呢?

哈嘍,大家好,我是指北君。在軟件開發過程中,我們會用到很多設計模式,之前介紹過的單例模式,還有后面會介紹的代理模式、適配器模式、建造者模式等等,合適的場景運用合適的設計模式,你會發現業務邏輯會清晰很多。

但是對于工廠模式,我奉勸大家,沒事最好不要用,什么,你問為啥?

1.什么是工廠模式

Define an interface for creating an object,but let subclasses decide which class toinstantiate.Factory Method lets a class defer instantiation to subclasses.

定義一個創建對象的接口,讓其子類自己決定實例化哪一個工廠類,工廠模式使其創建過程延遲到子類進行。

說人話:提供創建對象的接口,將創建對象的過程屏蔽,從而達到靈活的目的。

2.工廠模式分類

一般情況下,工廠模式分為三類:

①簡單工廠模式(Simple Factory)

②工廠方法模式(Factory Method)

③抽象工廠模式(Abstract Factory)

這三種模式從上到下逐步抽象,并且更具一般性。

需要說明的是:GOF 在《設計模式》一書中將工廠模式分為兩類:工廠方法模式(Factory Method)與抽象工廠模式(Abstract Factory),將簡單工廠模式(Simple Factory)看為工廠方法模式的一種特例,兩者歸為一類。

下面我們分別介紹這三種工廠模式。

2.1 簡單工廠(Simple Factory)

比如有這樣一個需求:

根據導入的不同文件(docx,xlsx,pptx),選擇不同的解析器進行解析。

簡單工廠有三個核心對象:

  • 工廠:簡單工廠模式的核心,它負責實現創建所有實例的內部邏輯。工廠類的創建產品類的方法可以被外界直接調用,創建所需的產品對象。
  • .抽象產品 :簡單工廠模式所創建的所有對象的父類,它負責描述所有實例所共有的公共接口。
  • 具體產品:是簡單工廠模式的創建目標,所有創建的對象都是充當這個角色的某個具體類的實例。

①抽象解析器

public interface IOfficeParser {
void parse();
}

②具體解析器(docx,xlsx,pptx)

public class WordParser implements IOfficeParser{
private String filePath;
public WordParser(String filePath){
this.filePath = filePath;
}
@Override
public void parse() {
System.out.println("解析 docx 文件");
}
}
public class ExcelParser implements IOfficeParser{
private String filePath;
public ExcelParser(String filePath){
this.filePath = filePath;
}
@Override
public void parse() {
System.out.println("解析 xlsx 文件");
}
}
public class PptParser implements IOfficeParser {
private String filePath;
public PptParser(String filePath){
this.filePath = filePath;
}
@Override
public void parse() {
System.out.println("解析 pptx 文件");
}
}

③構造解析器的工廠

public class OfficeParserFactory {

public static IOfficeParser getParser(String filePath) throws Exception {
String fileExtension = getFileExtension(filePath);
IOfficeParser parser = null;
if("docx".equalsIgnoreCase(fileExtension)){
parser = new WordParser(filePath);
}else if("xlsx".equalsIgnoreCase(fileExtension)){
parser = new ExcelParser(filePath);
}else if("pptx".equalsIgnoreCase(fileExtension)){
parser = new PptParser(filePath);
}else{
throw new Exception("file is not supported:"+fileExtension);
}
return parser;
}

private static String getFileExtension(String filePath){
// 解析文件名獲取文件擴展名,比如 文檔.docx,返回 docx
String fileExtension = filePath.substring(filePath.lastIndexOf(".")+1);
return fileExtension;
}
}

④測試類

public class SimpleFactoryTest {

public static void main(String[] args) throws Exception {
String filePath = "文檔.docx";
IOfficeParser parser = OfficeParserFactory.getParser(filePath);
parser.parse();

String filePath1 = "表格.xlsx";
IOfficeParser parser1 = OfficeParserFactory.getParser(filePath1);
parser1.parse();
}
}

⑤總結

這便是簡單工廠,客戶端避免了直接創建解析器的責任,只需要調用工廠類去解析就行了。

可以從開閉原則(對擴展開放,對修改關閉)來分析簡單工廠模式:當增加一種文件解析,比如老版本的 doc 格式。這時候只需要新增一個 parser 類即可,客戶端(理解為測試類,調用端)不用改變,然后在工廠類 OfficeParserFactory 新增一個 else-if 分支即可。

這時候可能有同學會問了,那修改了 OfficeParserFactory 類,不就違反開閉原則了嗎?但其實只要不是頻繁的添加新的 parser,偶爾修改一下 OfficeParserFactory 類,稍微不符合開閉原則,也是可以接受的。

看上去比較完美,細心的同學可能會問,所有的解析類對象創建都在 OfficeParserFactory 類中,假設某個解析類,比如 doc 創建parser 對象并不是簡單的 new ,還包括一些其它的操作,這時候難道把這些代碼也全部寫到 OfficeParserFactory 中嗎?有沒有更優雅的寫法呢?

有,就是下面要介紹的 工廠模式。

2.2 工廠方法(Factory Method)

為了解決上面的問題,我們可以為工廠類在創建一個工廠,也就是工廠的工廠,用來創建工廠類對象。

①給每一個具體解析器創建工廠

public class ExcelParserFactory implements IOfficeParserFactory {

@Override
public IOfficeParser createParser() {
// TODO 進行創建對象的一些操作
return new ExcelParser();
}
}

②創建解析器的工廠

public class OfficeParserFactory {

public static IOfficeParser getParser(String filePath) throws Exception {
String fileExtension = getFileExtension(filePath);
IOfficeParserFactory parserFactory = OfficeParserFactoryMap.getOfficeParseFactory(fileExtension);
if(parserFactory == null){
throw new Exception("file is not supported:"+fileExtension);
}
IOfficeParser parser = parserFactory.createParser();
return parser;
}

private static String getFileExtension(String filePath){
// 解析文件名獲取文件擴展名,比如 文檔.docx,返回 docx
String fileExtension = filePath.substring(filePath.lastIndexOf(".")+1);
return fileExtension;
}
}

③創建解析器工廠的工廠類

public class OfficeParserFactoryMap {
private static final Map<String, IOfficeParserFactory> parserFactoryCached = new HashMap<>();
static {
parserFactoryCached.put("docx",new WordParserFactory());
parserFactoryCached.put("xlxs",new ExcelParserFactory());
parserFactoryCached.put("pptx",new PptParserFactory());
}
public static IOfficeParserFactory getOfficeParseFactory(String type){
if(type == null || type.isEmpty()){
return null;
}
return parserFactoryCached.get(type.toLowerCase());
}


}

④測試類

public class FactoryTest {
public static void main(String[] args) throws Exception {
String filePath = "文檔.docx";
IOfficeParser parser = OfficeParserFactory.getParser(filePath);
parser.parse();
}
}

⑤總結

在工廠模式中,如果我們要增加新的文件解析,比如 mdb 格式(office access套件),就只需要創建新的 parser 類和 parserFactory 類,并且在 OfficeParserFactoryMap 類中將新的 parserFactory 類添加到 map 中即可。代碼的改動非常少,基本上是符合開閉原則的。

但是,我們看到工廠模式新增了很多 factory 類,會增加代碼的復制性,如果每個 factory 類只是做簡單的 new 操作,則沒必要使用該模式,直接用簡單工廠模式即可。

2.3 抽象工廠(Abstract Factory)

這種模式比較特殊,使用場景不多,大家簡單了解一下就行。

我們知道 doc 和 docx 都是 office word 文檔后綴,類似 xls 和 xlsx 都是 office Excel 表格后綴,還有 ppt 和 pptx。doc/xlx/ppt 都是舊版本 office 文件后綴,都是二進制組成,解析的時候有共同之處,而 docx/xlsx/pptx 是office新版本文件后綴,是通過 ooxml 結構組成。相當于一組是老的office,一組是新的office。

如果我們還是用工廠模式來實現的話,那每一種都要編寫一個工廠類,過多的類會難以維護,那怎么解決呢?

抽象工廠模式就是針對這種特殊的場景誕生,我們可以讓一個工廠復制創建多個不同類型的對象,而不是只創建一個 parser 對象。

具體代碼實現如下:

public interface IOfficeParserFactory {

IOfficeParser createParser();

IOldOfficeParser createOldParser();
}
public class ExcelParserFactory implements IOfficeParserFactory {

@Override
public IOfficeParser createParser() {
return new ExcelParser();
}

@Override
public IOldOfficeParser createOldParser() {
return new DocParser();
}
}

3.簡單工廠和工廠方法區別

簡單工廠:將創建不同對象的邏輯放在一個工廠類中。

工廠方法:將創建不同對象的邏輯放在不同工廠類中,先用一個工廠類的工廠類得到某個工廠,在某這個工廠來創建對象。

這樣講區別就很明顯了,如果創建對象的邏輯比較復雜,要做各種初始化操作,這時候使用工廠方法,能夠將復雜的創建邏輯拆分到多個工廠類中;而創建對象的邏輯很簡單,就沒必要額外創建多個工廠類,直接使用簡單工廠即可。

4.工廠模式的作用

封裝變化:創建邏輯有可能變化,封裝成工廠類之后,創建邏輯的變更對調用者透明。

代碼復用:創建代碼抽離到獨立的工廠類之后可以復用。

隔離復雜性:封裝復雜的創建邏輯,調用者無需了解如何創建對象。

控制復雜度:將創建代碼抽離出來,讓原本的函數或類職責更單一,代碼更簡潔。

責任編輯:武曉燕 來源: Java技術指北
相關推薦

2011-12-07 10:26:51

唐駿云計算

2021-03-06 22:50:58

設計模式抽象

2016-11-21 20:11:15

蘋果iPhone生產線

2011-11-17 16:03:05

Java工廠模式Clojure

2021-04-20 08:53:51

工廠模式電商公司設計模式

2016-10-21 10:00:01

HTML標簽WEB

2021-01-20 07:28:02

nullcollections對象

2024-03-06 13:19:19

工廠模式Python函數

2020-08-21 07:23:50

工廠模式設計

2022-01-12 13:33:25

工廠模式設計

2023-04-27 13:25:22

索引合并MySQL

2018-03-12 10:57:14

JavaKotlin語法

2023-02-26 15:49:08

元宇宙ChatGPT

2021-09-29 13:53:17

抽象工廠模式

2020-10-19 09:28:00

抽象工廠模式

2009-01-15 10:55:29

JavaScript設計模式抽象工廠

2015-08-04 08:56:14

swift子類

2022-05-09 08:04:50

工廠模式設計模式

2010-04-19 09:30:00

工廠模式PHP設計模式

2013-11-26 16:29:22

Android設計模式
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产亚洲成av人在线观看导航 | 成人教育av| 国产精品色av | 成人小视频在线 | 久久狼人天堂 | 黄视频网址| 国产玖玖| 久久国内| 久久夜色精品国产 | 中文字幕 在线观看 | 在线观看电影av | 欧美精品在线播放 | 一区二区三区四区国产 | 亚洲深夜福利 | 日日操夜夜操天天操 | 亚洲精品电影 | 一级毛片在线播放 | 亚洲国产成人在线视频 | jlzzjlzz欧美大全 | 97超碰免费 | 一级做a爰片久久毛片免费看 | 欧美激情久久久 | 四虎成人在线播放 | 亚洲不卡一 | 亚洲精品v | 99久久日韩精品免费热麻豆美女 | 中文字幕亚洲一区 | 日本精品久久久一区二区三区 | 亚洲午夜精品视频 | 色综合99 | 依人成人 | 国产精品视频导航 | 91精品国产乱码久久久久久久久 | 国产精品视频网址 | 日韩一区二区三区在线观看 | 成人午夜在线 | 国产精品69毛片高清亚洲 | 久久精品网| 欧美a级成人淫片免费看 | 午夜国产一级 | 日韩免费一区二区 |