設(shè)計(jì)模式之開閉原則

相關(guān)鏈接:
0. 設(shè)計(jì)模式之六大原則總結(jié)
1. 設(shè)計(jì)模式之單一職責(zé)原則
2. 設(shè)計(jì)模式之里式替換原則
3. 設(shè)計(jì)模式之依賴倒置原則
4. 設(shè)計(jì)模式之接口隔離原則
5. 設(shè)計(jì)模式之迪米特法則
6. 設(shè)計(jì)模式之開閉原則

1.1 定義

開閉原則的全稱是 Open Close Principle(OCP),軟件中的對(duì)象(類、模塊、函數(shù)等)應(yīng)該對(duì)于擴(kuò)展是開放的,對(duì)于修改是封閉的。

1.2 問題產(chǎn)生

在軟件的生命周期內(nèi),因?yàn)樽兓?、系統(tǒng)升級(jí)和維護(hù)等原因需要對(duì)軟件原有代碼進(jìn)行修改時(shí),可能會(huì)將錯(cuò)誤引入原本已經(jīng)測(cè)試過的舊版本中,破壞原有系統(tǒng)。

1.3 解決方案

開閉原則是為軟件實(shí)體的未來事件而制定的對(duì)現(xiàn)行開發(fā)設(shè)計(jì)進(jìn)行約束的一個(gè)原則,當(dāng)軟件需要變化時(shí),盡量通過擴(kuò)展軟件實(shí)體的行為來實(shí)現(xiàn)變化,而不是通過修改已有的代碼來實(shí)現(xiàn)變化。

1.4 舉個(gè)例子

書店銷售書籍

image.png

上面有兩個(gè)類 BookStore NoveIBook 和 一個(gè)接口 IBook,代碼如下:

// 接口 IBook
@protocol IBook <NSObject>
- (NSString *)getName;
- (NSString *)getAuthor;
- (int)getPrice;
@end

// NoveIBook類
@interface NoveIBook () < IBook >
{
    NSString *name;
    NSString *price;
    NSString *author;
}
@end

@implementation

- (id)noveIBook:(NSString *)name author:(NSString *)author price:(int)price{
    NoveIBook *book = [NoveIBook new];
    book.name = name;
    book.author = author;
    book.price = price;
    return noveIBook;
}

- (NSString *)getName{
    return name;
}

- (NSString *)getAuthor{
  return author;
}

- (int)getPrice{
  return price;
}
@end

// BookStore
static NSMutableArray <id <IBook>>*books;

@interface BookStore ()

@end

@implementation BookStore

+ (void)buildBooks{
    if (!books) {
        books = [NSMutableArray array];
        [books addObject:[NoveIBook noveIBook:@"天龍八部" author:@"金庸" price:32]];
        [books addObject:[NoveIBook noveIBook:@"巴黎圣母院" author:@"雨果" price:42]];
        [books addObject:[NoveIBook noveIBook:@"悲慘世界" author:@"雨果" price:32]];
    }
}

@end

現(xiàn)在需求變了,書籍的價(jià)格可能會(huì)隨著活動(dòng)或者其它情況有折扣調(diào)整,以打折銷售為例:所有 40 元以上的書籍九折銷售,其它 8 折銷售,那么這樣的代碼結(jié)構(gòu)怎么去處理需求變化的情況呢?有以下三種方法可以處理:

  • 修改接口
    在 IBook 上新增一個(gè) getOffPrice()方法,專門用于進(jìn)行打折處理,所有的實(shí)現(xiàn)類實(shí)現(xiàn)該方法,但是修改的后果就是,實(shí)現(xiàn)該接口的類需要修改,最后獲取的時(shí)候也需要調(diào)用 getOffPrice 而不是 getPrice 方法。同時(shí) IBook 作為協(xié)議應(yīng)該是穩(wěn)定可靠的,不應(yīng)該經(jīng)常變化,否則作為契約的作用就失去效能了

  • 修改實(shí)現(xiàn)類
    直接在 NoveIBook 的 getPrice 方法中修改,大多數(shù)情況下,這是一個(gè)好辦法,可能實(shí)際中我們很多情況都是這樣處理的

  • 通過擴(kuò)展實(shí)現(xiàn)變化
    增加一個(gè)子類 OffNovelBook 繼承自 NoveIBook,重寫 getPrice 方法,然后在 BookStore 添加書籍的地方,將實(shí)現(xiàn)類修改為 OffNovelBook,那么獲取的價(jià)格就是打折扣的價(jià)格

1.5 總結(jié)

開閉原則對(duì)擴(kuò)展開放,對(duì)修改關(guān)閉,并不意味著不做任何修改,低層模塊的變化,必然要有高層模塊進(jìn)行耦合,否則就是一個(gè)孤立無意義的代碼片段。
這個(gè)原則很虛,包含很廣,具體的理解可以通過具體的設(shè)計(jì)模式的設(shè)計(jì)思維去加深理解。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

友情鏈接更多精彩內(nèi)容