設計模式:工廠模式,解除耦合的利器

鄙人薛某發表於2019-07-10

工廠模式是使用頻率很高的一種設計模式,在面試中也經常問到,今天我們就來學習它。

為什麼要用工廠模式?

解答這個問題前,我們先來了解什麼是工廠模式。

工廠模式其實也稱建立模式,是用於建立物件的一種方式。本質上就是用工廠方法來代替new例項化物件。

舉個例子:我們在編寫程式碼的時候,在一個A類中通過new的方式例項化了類B,那麼A類和B類之間就存在耦合,如果以後修改了B類的程式碼和使用方式,例如需要在建構函式中傳入引數,那麼A類也就需要跟著修改了,一個類的依賴可能影響不大,但若有多個類依賴了B類,那麼這個工作量將會相當的大,這無疑是件非常痛苦的事。這種情況下,我們需要把建立例項的工作單獨分離,與呼叫方解耦,也就是使用工廠方法建立例項的工作封裝起來。這樣我們在需要呼叫物件的時候就不需要關心那些複雜的例項化問題。

工廠模式

工廠模式可分為兩類:簡單工廠模式和工廠模式。

簡單工廠模式

定義一個介面和實現類,建立一個工廠類這些實現類進行例項的建立。

我們用球來舉例,定義一個基本的介面Ball,和一個抽象方法Play (玩),

public interface Ball {
    void play();
}

建立一個籃球的類和一個足球的類,並實現該介面,

public class BasketBall implements Ball {
    public void play() {
        System.out.println("打籃球~~~");
    }
}
public class FootBall implements Ball {
    public void play() {
        System.out.println("踢足球~~~");
    }
}

然後,建立一個工廠類,可以用於生產籃球或者足球,

public class BallFactory {

    public Ball produce(String type) {
        if ("basketball".equals(type)) {
            return new BasketBall();
        } else if ("football".equals(type)) {
            return new FootBall();
        }
        return null;
    }
}

工廠類建好以後,我們就可以例項化工廠類,並呼叫 produce 方法來建立對應的例項物件,

public static void main(String[] args) {
    BallFactory factory = new BallFactory();
    Ball ball = factory.produce("basketball");
    ball.play();
}

結果輸出:打籃球~~~

這就是簡單工廠模式的基本實現,用關係圖來表示就是:
設計模式:工廠模式,解除耦合的利器
這種模式的優點是程式碼簡單,能夠根據具體的引數返回對應的例項物件。

當然缺點也很明顯,就是工廠類集中了所有例項的建立邏輯,如果增加業務就要多出相應的工廠方法,不僅程式碼可能變得臃腫,也容易違反GRASPR的高內聚的責任分配原則

工廠模式

又稱多型性工廠模式,是對簡單工廠模式的改進。工廠模式中,一個子類對應一個工廠類,這些工廠類都實現了一個工廠介面。這相當於把一個簡單工廠類拆分成多個工廠,這樣程式碼就不會都耦合在同一個類裡了。

具體的產品介面和實現類還是複用上面的程式碼,我們只需關注工廠方法的邏輯即可,

先建立一個工廠的介面

public interface IFactory {
    void produce();
}

然後建立對應業務的工廠類

public class BasketFactory implements IFactory {
    public Ball produce() {
        return new BasketBall();
    }
}
public class FootFactory implements IFactory {
    public Ball produce() {
        return new FootBall();
    }
}

測試程式碼

public static void main(String[] args) {
    BasketFactory basketFactory = new BasketFactory();
    Ball basket = basketFactory.produce();
    FootFactory footFactory = new FootFactory();
    Ball foot = footFactory.produce();
    basket.play();
    foot.play();
}

輸出結果是:

打籃球~~~
踢足球~~

如上所示,如果需要新增新的產品,如排球,我們就多寫一個工廠類即可,這樣就不會把所有的業務都耦合到一個工廠類中了,用關係圖表示如下:
設計模式:工廠模式,解除耦合的利器
最後,總結一下工廠模式的優點吧,

1、良好的封裝性,程式碼結構清晰,呼叫者只需知道產品的類名即可,不需要知道建立物件的過程,降低程式碼間的耦合。

2、擴充套件性優秀,如果增加一個產品類,只需增加一個對應的工廠類。

3、遮蔽產品類。產品類的實現如何變化,呼叫者都不需要關心,只需關心產品的介面,只要介面保持不變,系統中的上層模組就不會發生變化。

4、工廠模式是典型的解耦框架,高層模組只需要知道產品的抽象類,其他的實現類都不需要關心,符合迪米特法則,符合依賴倒置原則,符合里氏替換原則。

參考:

《設計模式之禪》

相關文章