面向接口編程-模塊化的設計思想

思路

最初我拿到一個問題的時候,首先想的就是他怎么實現,具體的實現,而面向接口編程需要先考慮好流程,明白變動點可能在哪里,系統的邊界在哪里,邊界的劃分決定了模塊和服務的拆分,變動點決定了需要抽象和隔離的地方。想清楚再下手。

提煉一下就是:

案例

代碼重構方案

//原來的代碼
for (ModuleEnum module : modules) {
    switch (module) {
        case ALBUM:
            Album albumData = getAlbumData(key, pageSize, pageNumber);
            if (albumData.getContent().size() == 0) {
                searchData.setAlbum(null);
            } else {
                searchData.setAlbum(albumData);
            }
            break;
        case ACTIVITY:
            Activity activityData = getActivityData(key, pageSize, pageNumber);
            if (activityData.getContent().size() == 0) {
                searchData.setActivity(null);
            } else {
                searchData.setActivity(activityData);
            }
            break;
        case COURSE:
            Course courseData = getCourseData(key, pageSize, pageNumber);
            if (courseData.getContent().size() == 0) {
                searchData.setCourse(null);
            } else {
                searchData.setCourse(courseData);
            }
            break;
        case BLOG:
            Blog blogData = getBlogData(key, pageSize, pageNumber);
            if (blogData.getBlogContents().size() == 0) {
                searchData.setBlog(null);
            } else {
                searchData.setBlog(blogData);
            }
            break;
        default:
    }
}

這里我們看到了非常多的代碼冗余,邏輯完全一樣,但是寫了四遍.再看下面:

//這里4個方法對應上面的調用(邏輯完全一樣,僅僅是返回值和其中一些調用的方法不一樣,我就只列出方法名了)
private Album getAlbumData(String key, int pageSize, int pageNumber) throws IOException;
private Activity getActivityData(String key, int pageSize, int pageNumber) throws IOException;
private Course getCourseData(String key, int pageSize, int pageNumber) throws Exception;
private Blog getBlogData(String key, int pageSize, int pageNumber) throws IOException;

重構思路

對這里的重構我考慮用如下的方法:
1.使用工廠模式調用不同的實現,替換switch.
2.在接口和實現之間增加抽象方法,提取公共邏輯

上碼

1 枚舉實現工廠方法

//枚舉類
public enum ModuleEnum {

    //在線課程
    COURSE("0", "course", "courseindex"){
        @Override
        public ISearchService build() {
            return  new CourseSearch();
        }
    },
    //培訓活動
    ACTIVITY("1", "activity", "activityindex") {
        @Override
        public ISearchService build() {
            return  new AlbumSearch();
        }
    },
    //知識庫
    ALBUM("2", "album", "albumindex") {
        @Override
        public ISearchService build() {
            return new AlbumSearch();
        }
    },
    //Q記
    BLOG("3", "blog", "blogindex") {
        @Override
        public ISearchService build() {
            return new BlogSearch();
        }
    };

    public abstract ISearchService build();
}

這里使用枚舉來實現工廠方法,因為考慮使用反射效率較低.

可以看到,我定義了一個抽象的build方法,返回實現類的接口,而在每個枚舉類中又實現了build方法 返回不同的實現類.

//調用者
ISearchService isearchservice = ModuleEnum.build();
if (isearchservice == null) {
    //防御性判斷
    throw new RuntimeException("不支持的模塊類型" + ModuleEnum.getName());
}
return isearchservice;

這里通過調用build方法來實現工廠方法的調用

2 增加抽象方法

public abstract class AbstractSearchService implements ISearchService {
    private static final Logger log = LoggerFactory.getLogger(AbstractSearchService.class);

    @Override
    public SearchData getSearchData(String key, int pageNumber, int pageSize, SearchData searchData) throws Exception {
        AbstractlModulBase modulData = getModulData(key, pageSize, pageNumber);
        if (modulData.getContent().size() == 0) {
            searchData.setAlbum(null);
        } else {
            //根據具體的子類調用具體的實現
            searchData = setData(searchData, modulData);
        }
        return searchData;
    }

    private AbstractlModulBase getModulData(String key, int pageSize, int pageNumber) {
        Page page = new Page(pageSize, pageNumber);
        //從ES中查詢基礎數據
        List<ParentBean> activityBeanList = getModulBeanList(key, page);
        //獲取模塊Id
        List<Integer> masterIdList = getParentIdList(activityBeanList);
        log.info("搜索到的模塊id = {}", masterIdList);
        //從bubbo中查詢實時數據 之后對ES中的數據和Dubbo中的實時數據進行整合 并判斷是否有數據未同步
        return getModulBeenAndMix(masterIdList, activityBeanList, page);
    }

    /**
     * 將基類轉換成不同的子類 并賦值
     * @param searchData 統一存儲查詢出的所有內容
     * @param modulData 基類
     * @return SearchData
     */
    protected abstract SearchData setData(SearchData searchData, AbstractlModulBase modulData);

    /**
     * ES查詢
     * @param key ES查詢關鍵詞
     * @param page 分頁信息
     * @return ES查詢結果Been
     */
    protected abstract List<ParentBean> getModulBeanList(String key, Page page);

    /**
     * 從bubbo中查詢實時數據 之后對ES中的數據和Dubbo中的實時數據進行整合 并判斷是否有數據未同步
     * @param masterIdList IdList
     * @param beanList ES查詢集合
     * @param page 分頁信息
     * @return 查詢總結果
     */
    protected abstract AbstractlModulBase getModulBeenAndMix(List<Integer> masterIdList, List<ParentBean> beanList, Page page);

    /**
     * 提取IdList
     * @param parentBeans been
     * @return IdList
     */
    private List<Integer> getParentIdList(List<? extends ParentBean> parentBeans) {
        Preconditions.checkNotNull(parentBeans);
        return Lists.transform(parentBeans, new Function<ParentBean, Integer>() {
            @Override
            public Integer apply(ParentBean input) {
                return input.getId();
            }

        });
    }
}

這里針對具體的邏輯進行拆分 并將具體的實現交由子類完成

一些設計思路

功能設計

  1. 模塊功能的改變、增加、移除;系統流程的變化
  2. 適應業務發展的一種常態(頻繁的,不可抗拒的)
  3. 設計系統時考慮:模塊解耦(避免牽一發而動全身); 面向接口編程(盡量減少對調用方影響)
  4. 修改系統時考慮:擴展 > 修改 (遵循“開閉原則”) → 功能兼容

程序設計

1. 問題拆分,劃分模塊 明確輸入輸出,暫時忽略代碼細節(切忌代碼堆疊) → 定義接口(面向接口編程)

參數校驗 → 命令識別與解析 → 文件讀取 → 命令執行 → 結果輸出


變化:

  • 參數校驗模塊(功能改變):不再校驗源文件目錄參數
  • 文件讀取模塊(新流程不再使用):不再需要使用文件讀取模塊
  • 結果輸出模塊(功能改變):由輸出到結果文件 → 通過日志到控制臺 [LocalFileOutputService → LogOutputService]

模塊功能變化后的幾種改動方式(不分優劣,取決于具體場景):

  • 直接修改原模塊邏輯:核心邏輯未變,局部微調,bugfix
  • 繼承原有模塊:優勢:復用父類邏輯;局限性:單繼承
  • 新增模塊實現:核心邏輯變化,入參未變化(如結果輸出:輸出文件 → 輸出日志)
  • 擴展接口方法:核心邏輯未變,入參變化 (不同方法適用于不同的場景,方法重載,類比構造方法)

優勢:功能聚合,適用不同場景;公共邏輯可以復用

2. 數據模型的建立 → 抽象(關注當前問題域的屬性)、繼承

  • 命令實體:關注:(類型,選項,參數)
  • 文件數據:關注:(內容,文件名) 忽略(創建時間,訪問權限等)
  • 不再代表原始文件數據:而是代表中間處理結果

3. 過程抽象 → 分離變與不變(模板方法模式)

  • 流程發生變化
  • 對外接口不變

4. 各個模塊代碼的實現 → 代碼規范

  • 變量命名:見名知意、駝峰命名、常量大寫下劃線分隔等
  • 異常處理:
    1. 拋出or處理
    2. 自定義異常:類型、原因、執行現場信息、異常棧(便于定位問題代碼)
  • 日志記錄:遠離System.out.println(性能:加鎖+阻塞當前線程)

5. 性能優化 → 可選

多文件流式處理(緩解內存壓力),線程池執行(提高吞吐量)等等。

最后

這里我的實現還不是很完善,但思路基本就是這樣,重構完之后代碼可擴展性更好,代碼冗余減少.

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,786評論 6 534
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,656評論 3 419
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,697評論 0 379
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,098評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,855評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,254評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,322評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,473評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,014評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,833評論 3 355
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,016評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,568評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,273評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,680評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,946評論 1 288
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,730評論 3 393
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,006評論 2 374

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,667評論 25 708
  • 貼吧比朋友圈更容易展示自己: 1、前期不要一上來就發廣告貼 就算級別高,百度不刪,吧主看你不順眼也會刪,說實話,我...
    倪青語閱讀 389評論 0 0
  • 從前很鄙視全職媽媽,覺得自己一定不會整天圍著老公、孩子轉,不管任何時候都要做一個自力更生的現代女性。 ...
    博8429閱讀 295評論 0 0
  • 也還是這種夜晚,也還是一樣的心情。靜靜的走在月色中,被薄霧籠罩,蒙上面紗。街燈早早地就亮了,早早地撕破黃昏,燃燒黑...
    擁抱謊言擁抱你i閱讀 184評論 0 0