Java设计模式之 —— 装饰者(Decorator) — 点炒饭

Java设计模式之 —— 装饰者(Decorator) — 点炒饭

下面会用做炒饭的例子来描述一下装饰者设计模式,装饰者设计模式比较适合用于拓展功能,我要做炒饭,最简单的就是油盐加米饭就可以做出炒饭,但是我们需要用装饰者的思想做出蛋炒饭,肉丝炒饭,培根炒饭。
要是不用设计模式,我们用java实现这个那就是写一个类,类里面放上各种食材(肉丝、鸡蛋、培根之类的),在我们给素炒饭添加添加食材的时候就用if语句来判断。一般来讲我们很可能写成下面这样的:

 素炒饭
 .
 .
 .
 if(添加 肉丝){
     name: 肉丝炒饭
     price:素炒饭价钱 + 肉丝价钱
 }else if(鸡蛋){
     name: 鸡蛋炒饭
     price:素炒饭价钱 + 鸡蛋价钱
 }else if{
     ...
 }
 .
 .
 .

如果我们需要添加食材,或者一种食材需要添加两次(放两份儿肉丝),我们每次都需要去修改类中的代码,一旦我们的食材种类多了,那就很操蛋了,你们是不是?


这里写图片描述

普通的写法不利于扩展,有修改或者增加新的食材都需要在类里面修改代码。这个时候就该考虑装饰者设计模式,装饰者设计模式的宗旨是 “欢迎扩展,拒绝修改” ,下面是Head Frist 设计模式的描述:

装饰者设计模式:动态地将责任附加到对象上。若要扩展功能,装饰者提供了比继承更有弹性的替代方案。

做炒饭我可以添加食材,但是不需要修改类里面的代码,是不是很好?
下面我用装饰者设计模式来做炒饭,我们先来创建一个主食的超类,不管是炒饭还是炒面还是炒粉都要实现这个接口,里面有两个方法,一个计算name一个计算price。

1.主食接口超类:

/**
 * @Description: 主食接口超类
 * @Author: Danxingxi
 * @CreateDate: 2016/10/13 9:11
 */
public interface IFood {
    /**
     * 主食价格
     */
    int caculatePrice();

    /**
     * 主食描述
     */
    String description();

}

2.然后我们给食材也常见一个超类,我们的肉丝、鸡蛋、培根都实现这个接口:

/**
 * @Description: 添加食材的超类接口,就是继承了IFood接口,当
 *               食材实现这个接口的时候需要用到IFood中的方法来对价格和描述做出修改
 * @Author: Danxingxi
 * @CreateDate: 2016/10/13 9:31
 */
public interface IFoodDecorator extends IFood {

    //这里为了食材添加别的功能也可以增加新的方法,比如食材的功效之类的
    
}

3.炒饭类:

/**
 * @Description:  素炒饭 价格 5 元
 * @Author: Danxingxi
 * @CreateDate: 2016/10/13 9:26
 */
public class ChaoFan implements IFood {
    /**
     * 主食价格
     */
    @Override
    public int caculatePrice() {
        return 5;
    }

    /**
     * 主食描述
     */
    @Override
    public String description() {
        return "炒饭";
    }
}

4.炒面类

/**
 * @Description:  素炒面  价格 4 元
 * @Author: Danxingxi
 * @CreateDate: 2016/10/13 9:27
 */
public class ChaoMian implements IFood {
    /**
     * 主食价格
     */
    @Override
    public int caculatePrice() {
        return 4;
    }

    /**
     * 主食描述
     */
    @Override
    public String description() {
        return "炒面";
    }
}

5.肉丝食材类:

/**
 * @Description: 肉丝添加食材
 *               价格: 5 元
 * @Author: Danxingxi
 * @CreateDate: 2016/10/13 9:35
 */
public class RoseDecorator implements IFoodDecorator {

    private IFood iFood;  //主食的引用,这样可以直接对主食的name和price做修改

    /**
     * 在构建食材的时候告诉是用于在什么主食中,
     * 我们只需要使用IFood接口就可以来修改主食的价格和描述,这就是IFood超类的好处
     * @param food
     */
    public RoseDecorator(IFood food){
        this.iFood = food;
    }

    /**
     * 主食价格
     */
    @Override
    public int caculatePrice() {
        /**在主食本来具有的价格上增加 5 元**/
        return iFood.caculatePrice() + 5;
    }

    /**
     * 主食描述
     */
    @Override
    public String description() {
        /**在主食本来具有的描述上增加 '肉丝' 描述 **/
        return "肉丝 "+iFood.description();
    }
}

6.鸡蛋食材类:

/**
 * @Description: 鸡蛋食材  价格 3 元
 * @Author: Danxingxi
 * @CreateDate: 2016/10/13 9:53
 */
public class EggDecorator implements IFoodDecorator {

    private IFood iFood;

    /**
     * 在构建食材的时候告诉是用于在什么主食中,
     * 我们只需要使用IFood接口就可以来修改主食的价格和描述,这就是IFood超类的好处
     * @param food
     */
    public EggDecorator(IFood food){
        this.iFood = food;
    }

    /**
     * 修改 主食价格
     */
    @Override
    public int caculatePrice() {
        return iFood.caculatePrice() + 3;
    }

    /**
     * 修改 主食描述
     */
    @Override
    public String description() {
        return "鸡蛋 " + iFood.description() ;
    }
}

7.使用

我是做android开发的,我在activity中测试的:

package com.java.ui;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.TextView;

import com.danxx.views.R;
import com.java.decorator.ChaoFan;
import com.java.decorator.ChaoMian;
import com.java.decorator.EggDecorator;
import com.java.decorator.IFood;
import com.java.decorator.RoseDecorator;

/**
 * @Description: 装饰者设计模式测试
 * @Author: Danxingxi
 * @CreateDate: 2016/10/13 10:00
 */
public class ActivityDecorator extends AppCompatActivity {

    private TextView name,price;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_decorator);
        name = (TextView) findViewById(R.id.name);
        price = (TextView) findViewById(R.id.price);
    }

    /**
     * 做一份肉丝炒饭
     * @return
     */
    public void roseChaoFan(View view){
        IFood iFood = new RoseDecorator(new ChaoFan());
        name.setText("主食: "+ iFood.description());
        price.setText("价格: "+ iFood.caculatePrice() +" 元");
    }

    /**
     * 做一份肉丝鸡蛋炒饭
     * @return
     */
    public void roseEggChaoFan(View view){
        IFood iFood = new RoseDecorator(new EggDecorator(new ChaoFan()));
        name.setText("主食: "+ iFood.description());
        price.setText("价格: "+ iFood.caculatePrice() +" 元");
    }

    /**
     * 做一份鸡蛋肉丝炒面
     * @return
     */
    public void eggRoseChaoMian(View view){
        IFood iFood = new EggDecorator(new RoseDecorator(new ChaoMian()));
        name.setText("主食: "+ iFood.description());
        price.setText("价格: "+ iFood.caculatePrice() +" 元");
    }

}

效果:


这里写图片描述

8.总结

源代码地址:Java设计模式装饰者

装饰者模式 对于扩展功能来说很好,比如我们的炒饭,我们需要添加牛肉食材,我们只需新建一个实现了IFoodDecorator 接口的类,并在两个方法中作出自己的修改就可以了,这样就把牛肉食材增加了,如果我们要增加炒粉,那就增加一个实现了IFood接口的炒粉类实现对应的方法就可以了,是不是很方便,很好扩展,还不用修改之前的

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,527评论 5 470
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,314评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,535评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,006评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,961评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,220评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,664评论 3 392
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,351评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,481评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,397评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,443评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,123评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,713评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,801评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,010评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,494评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,075评论 2 341

推荐阅读更多精彩内容