單一職責(zé)原則的理解與實現(xiàn)

何謂單一職責(zé)原則

按字面理解,單一職責(zé)原則就是自己只負(fù)責(zé)自己的事木人,不需要理會別人的事麸锉。如果了解面對對象編程钠绍,那么應(yīng)該會很容易了解這個單一職責(zé)原則。

在面對對象編程中淮椰,每個對象只負(fù)責(zé)自己的任務(wù)五慈,比如該提供數(shù)據(jù)的就只是提供數(shù)據(jù)纳寂,該負(fù)責(zé)提供服務(wù)的就只提供服務(wù)主穗,或者只是維護(hù)對象之間的關(guān)系,這樣的開發(fā)方式代碼耦合度較低毙芜,較靈活忽媒,易擴(kuò)展。當(dāng)然也可以一個對象負(fù)責(zé)多個任務(wù)腋粥,但是任務(wù)多了修改起來就比較容易影響到其他的任務(wù)晦雨。

Object Design: Roles, Responsibilies, and Collaborations這本書中提出可以從以下幾方面判斷出一個對象的多個行為構(gòu)造出的是多職責(zé)還是單職責(zé)。
1隘冲、Information holder - 該對象設(shè)計為存儲對象并提供對象信息給其他對象
2闹瞧、Structurer - 該對象設(shè)計為維護(hù)對象與信息間的關(guān)系
3、Service provider - 該對象設(shè)計為處理任務(wù)與提供服務(wù)給其他對象
4展辞、Controller - 該對象設(shè)計為負(fù)責(zé)控制一系列職責(zé)的任務(wù)處理
5奥邮、Coordinator - 該對象設(shè)計為把任務(wù)綁定/委托到其他對象上
6、Interfacer - 該對象設(shè)計為在各個對象間負(fù)責(zé)轉(zhuǎn)化信息或者請求

一旦你知道了這些概念,那就狠容易知道你的代碼到底是多職責(zé)還是單一職責(zé)了洽腺。

實現(xiàn)物品添加購物車

有這樣的需求脚粟,有一些產(chǎn)品,需要以產(chǎn)品列表形式展示蘸朋,并且提供雙擊產(chǎn)品添加到購物車核无。有一個購物車,接收產(chǎn)品添加到購物車的操作藕坯,添加之后并顯示出來

這種簡單的需求如果使用面對過程的方式來實現(xiàn)時很容易的团南,代碼量也很少,但是不益于以后擴(kuò)展炼彪。比如資源的來源已慢、種類變了,或者添加方式變了霹购,改一個東西都容易影響到其他邏輯佑惠,使用面對對象的方式來實現(xiàn)就會比較靈活,可以把這個需求抽象齐疙,然后再慢慢實現(xiàn)膜楷。

Paste_Image.png

簡單抽象,就可以抽象出產(chǎn)品贞奋、購物車兩個對象赌厅,購物車與產(chǎn)品需要通信所以需要一個通信的對象。再細(xì)化一下轿塔,把產(chǎn)品與購物車細(xì)分為信息提供與負(fù)責(zé)交互的兩個對象并把這些對象轉(zhuǎn)化為編程對象特愿。

Paste_Image.png

由上可以抽象出6個對象,產(chǎn)品對象勾缭、事件對象揍障、購物車對象均有一個負(fù)責(zé)對外的對象,類似于門面模式俩由。Product毒嫡、Event、Cart這三個對象比較靈活幻梯,可以復(fù)用或者拓展兜畸,當(dāng)然實現(xiàn)起來會相對復(fù)雜一點,代碼量也會多一點碘梢。但是這只是相對于簡單不需要拓展的需求上咬摇,如果是比較龐大的需求或者是比較靈活的架構(gòu),使用面對對象編程的方式不僅可以節(jié)省代碼煞躬,而且擴(kuò)展維護(hù)更方便肛鹏,使用哪種方式只是根據(jù)需求來選擇了。

再對這些對象細(xì)化成基本工程,以下是基本交互思路圖龄坪。

Paste_Image.png

以下是具體代碼實現(xiàn)

Product.js 負(fù)責(zé)提供產(chǎn)品數(shù)據(jù)

function Product(id, description) {
    /**
     * 獲取商品ID
     * 
     * @return {int   }  商品id
     */
    this.getId = function() {
        return id;
    };

    /**
     * 獲取商品描述
     * 
     * @return {string} 商品描述
     */
    this.getDescription = function() {
        return description;
    }
}

module.exports = Product;

產(chǎn)品控制器ProductController.js, 負(fù)責(zé)對外交互

var ProductRepository = require('./ProductRepository.js');

function ProductController(productRepository, eventAggregator) {
    // 獲取所有的物品
    var products = productRepository.getProducts();
    this.onProductSelect = function(id) {
        var product;
        products.forEach(function(pro) {
            if(pro.getId() == id){
                product = pro;
            }
            console.log('product is ' + pro.getId());
        });
        // 觸發(fā)雙擊添加購物車事件
        eventAggregator.publish('productSelected', {
            product: product
        });
    }
    // 列出所有物品
    products.forEach(function(product) {
        var id = product.getId(),
            description = product.getDescription();

        console.log('product ' + id + ' , and description is ' + description);
        // 觸發(fā)雙擊添加購物車
        if(id == 1){
            this.onProductSelect(id);
        }
    }.bind(this));
}

module.exports = ProductController;

購物車基本功能

function Cart(eventAggregator) {
    this.items = []; // 購物車列表
    /**
     * 添加物品到購物車
     * 
     * @param {object} item 商品對象
     */
    this.addItem = function(item) {
        this.items.push(item);
        console.log('add Item ' + item);
        // 觸發(fā)添加購物車事件
        eventAggregator.publish('itemAdded', item);
    };
}

module.exports = Cart;

購物車控制器CartController.js 昭雌,負(fù)責(zé)購物車事件處理

function CartController(eventAggregator, cart) {
    // 訂閱物品添加事件
    eventAggregator.subscribe('itemAdded', function(eventArgs) {
        var id = eventArgs.getId(),
            description = eventArgs.getDescription();

        console.log('the ' + id + 'has been add to cart &' + description);
    });

    // 訂閱物品加入購物車事件
    eventAggregator.subscribe('productSelected', function(eventArgs) {
        console.log('recieved productSelected event ' + eventArgs);
        cart.addItem(eventArgs.product);
    });
}

module.exports = CartController;

事件對象Event.js,提供基本事件處理

function Event(name) {
    this.handlers = []; // 事件回調(diào)數(shù)組

    /**
     * 獲取事件名稱
     *              
     * @return {objecg} 事件對象
     */
    this.getName = function() {
        return name;
    };

    /**
     * 給事件添加處理函數(shù)
     *
     */
    this.addHandler = function(handler) {
        this.handlers.push(handler);
    };

    /**
     * 刪除已經(jīng)添加的事件處理函數(shù)
     * 
     * @param  {function} handler 需要刪除的事件處理函數(shù)
     * @return {void}
     */
    this.removeHandler = function(handler) {
        var i = 0,
            len = this.handlers.length;
        for (; i < len; i++) {
            if(this.handlers[i] == handler){
                this.handlers.splice(i, 1);
                break;
            }
        }
    };

    /**
     * 執(zhí)行事件處理函數(shù)
     * 
     * @param  {*} eventArgs 處理函數(shù)調(diào)用參數(shù)
     * @return {void}
     */
    this.fire = function(eventArgs) {
        this.handlers.forEach(function(h) {
            h(eventArgs);
        });
    }
}

module.exports = Event;

事件聚合器EventAggregator.js健田,負(fù)責(zé)提供發(fā)布烛卧、訂閱方法給其他對象進(jìn)行通信

var Event = require('./Event.js');

function EventAggregator() {
    this.events = []; // 事件對象集合

    /**
     * 根據(jù)事件名稱獲取事件對象
     * 
     * @param  {string} eventName 事件名稱
     * @return {object}           事件對象
     */
    function getEvent(eventName) {
        var event;
        this.events.forEach(function(ev) {
            if (ev.getName() == eventName) {
                event = ev;
            }
        });
        //console.log('get event after ' + event);
        return event;
    }

    /**
     * 事件發(fā)布
     * 
     * @param  {string} eventName 事件名稱
     * @param  {*     } eventArgs 調(diào)用事件處理函數(shù)時傳的參數(shù)
     * @return {void}           
     */
    this.publish = function(eventName, eventArgs) {
        var event = getEvent.call(this, eventName);
        if (!event) {
            event = new Event(eventName);
            this.events.push(event);
        }
        event.fire(eventArgs);
    };

    /**
     * 訂閱事件
     * 
     * @param  {string} eventName 事件名稱
     * @param  {function} handler   事件觸發(fā)時處理函數(shù)
     * @return {void}           
     */
    this.subscribe = function(eventName, handler) {
        var event = getEvent.call(this, eventName);
        if (!event) {
            event = new Event(eventName);
            this.events.push(event);
            console.log('add event ' + event.getName());
        }
        event.addHandler(handler);
    }
}

module.exports = EventAggregator;

以上代碼就是購物車需求的實現(xiàn),代碼相對于面向過程變成的確挺多妓局,但是職責(zé)清晰总放,易于理解。以上代碼實現(xiàn)均參考湯姆大叔博客好爬。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末局雄,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子存炮,更是在濱河造成了極大的恐慌炬搭,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,104評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件穆桂,死亡現(xiàn)場離奇詭異宫盔,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)享完,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,816評論 3 399
  • 文/潘曉璐 我一進(jìn)店門灼芭,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人般又,你說我怎么就攤上這事彼绷。” “怎么了茴迁?”我有些...
    開封第一講書人閱讀 168,697評論 0 360
  • 文/不壞的土叔 我叫張陵寄悯,是天一觀的道長。 經(jīng)常有香客問我笋熬,道長热某,這世上最難降的妖魔是什么腻菇? 我笑而不...
    開封第一講書人閱讀 59,836評論 1 298
  • 正文 為了忘掉前任胳螟,我火速辦了婚禮,結(jié)果婚禮上筹吐,老公的妹妹穿的比我還像新娘糖耸。我一直安慰自己,他們只是感情好丘薛,可當(dāng)我...
    茶點故事閱讀 68,851評論 6 397
  • 文/花漫 我一把揭開白布嘉竟。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪舍扰。 梳的紋絲不亂的頭發(fā)上倦蚪,一...
    開封第一講書人閱讀 52,441評論 1 310
  • 那天,我揣著相機(jī)與錄音边苹,去河邊找鬼陵且。 笑死,一個胖子當(dāng)著我的面吹牛个束,可吹牛的內(nèi)容都是我干的慕购。 我是一名探鬼主播,決...
    沈念sama閱讀 40,992評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼茬底,長吁一口氣:“原來是場噩夢啊……” “哼沪悲!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起阱表,我...
    開封第一講書人閱讀 39,899評論 0 276
  • 序言:老撾萬榮一對情侶失蹤殿如,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后最爬,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體握截,經(jīng)...
    沈念sama閱讀 46,457評論 1 318
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,529評論 3 341
  • 正文 我和宋清朗相戀三年烂叔,在試婚紗的時候發(fā)現(xiàn)自己被綠了谨胞。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,664評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡蒜鸡,死狀恐怖胯努,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情逢防,我是刑警寧澤叶沛,帶...
    沈念sama閱讀 36,346評論 5 350
  • 正文 年R本政府宣布,位于F島的核電站忘朝,受9級特大地震影響灰署,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜局嘁,卻給世界環(huán)境...
    茶點故事閱讀 42,025評論 3 334
  • 文/蒙蒙 一溉箕、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧悦昵,春花似錦肴茄、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,511評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽抗楔。三九已至,卻和暖如春拦坠,著一層夾襖步出監(jiān)牢的瞬間连躏,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,611評論 1 272
  • 我被黑心中介騙來泰國打工贞滨, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留反粥,地道東北人。 一個月前我還...
    沈念sama閱讀 49,081評論 3 377
  • 正文 我出身青樓疲迂,卻偏偏與公主長得像才顿,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子尤蒿,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,675評論 2 359

推薦閱讀更多精彩內(nèi)容

  • 1.埋點是做什么的 2.如何進(jìn)行埋點 3.埋點方案的設(shè)計 近期常被問到這個問題郑气,我擔(dān)心我的答案會將一些天真爛漫的孩...
    lxg閱讀 2,019評論 0 1
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,304評論 25 707
  • 用兩張圖告訴你,為什么你的 App 會卡頓? - Android - 掘金 Cover 有什么料腰池? 從這篇文章中你...
    hw1212閱讀 12,745評論 2 59
  • 如果你問我對一個人最高的評價是什么? “有趣奏属】缈纾” 有人說一個人有趣是因為有愛和被愛;有人說有趣是因為經(jīng)歷得多囱皿、見得...
    captain_嫣閱讀 4,756評論 54 140
  • 昨天早上爸爸跑到格格床上來看她嘱腥,丫頭睜開眼對著爸爸說:“你終于回來了耕渴!”因為爸爸昨晚很晚才回來,她都已經(jīng)睡著了齿兔。孩...
    格格麻麻閱讀 258評論 0 0