设计模式—装饰器模式
一、概念
装饰器模式(Decorator Pattern),也叫包装模式 Wrapper Pattern.允许向一个现有的对象添加新的功能,同时又不改变其结构。这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装。
优点:装饰类和被装饰类可以独立发展,不会相互耦合,装饰模式是继承的一个替代模式,装饰模式可以动态扩展一个实现类的功能。
缺点:多层装饰比较复杂。
使用场景: 1、扩展一个类的功能。 2、动态增加功能,动态撤销。
二、结构图
- Component接口,是装饰类和被装饰类的基本类型。
- ConcreteComponent类,是Component接口的实现类,也叫被装饰类,本身具有一定的功能,装饰类在被装饰类的功能上进行拓展。
- Decorator抽象类,是Component接口的实现类,内部维护了一个被装饰类的实例,通过构造器来进行初始化,他只是一个模板,真正的装饰类是其子类。
- ConcreteDecorator类,是Decorator抽象类的子类,是一个具体的装饰类,每一个装饰类都有自己特有的装饰效果。
三、场景适用
场景:你在大街上饥饿难忍,看到一个卖手抓饼的,老板告诉你,‘基础’手抓饼5元一个,‘普通’手抓饼多加一个鸡蛋,价格也多一元,‘富贵’手抓饼再加一根烤肠,价格再多两元,‘豪华’手抓饼再加一份牛肉,价格再多三元。
1.定义一个接口,输出你选择的手抓饼类型和手抓饼的价格
/**
* 基础接口,装饰类和被装饰类的父接口
*/
public interface IShreddedPancake {
void show();//用于输出手抓饼类型和加价
}
2.创建基础手抓饼类(被装饰类),实现于接口IShreddedPancake
public class BaseShreddedPancake implements IShreddedPancake{
@Override
public void show() {
System.out.println("基础手抓饼,一个5元...");
}
}
3.创建抽象装饰类,实现于接口IShreddedPancake
public abstract class Decorator implements IShreddedPancake{
private IShreddedPancake iShreddedPancake;//维护一个被装饰类对象
public Decorator(IShreddedPancake iShreddedPancake) {//通过构造器进行初始化
this.iShreddedPancake = iShreddedPancake;
}
@Override
public void show() {
iShreddedPancake.show();
}
}
4.创建具体的装饰类,继承于抽象装饰类Decorator
加鸡蛋的装饰类
public class DecoratorWithEgg extends Decorator{
private IShreddedPancake iShreddedPancake;
public DecoratorWithEgg(IShreddedPancake iShreddedPancake) {
super(iShreddedPancake);
}
@Override
public void show() {
super.show();
System.out.println("再加个蛋,多加1元...");
}
}
加烤肠的装饰类
public class DecoratorWithRoastSausage extends Decorator{
private IShreddedPancake iShreddedPancake;
public DecoratorWithRoastSausage(IShreddedPancake iShreddedPancake) {
super(iShreddedPancake);
}
@Override
public void show() {
super.show();
System.out.println("再加根烤肠,多加2元...");
}
}
加牛肉的装饰类
public class DecoratorWithBeef extends Decorator{
private IShreddedPancake iShreddedPancake;
public DecoratorWithBeef(IShreddedPancake iShreddedPancake) {
super(iShreddedPancake);
}
@Override
public void show() {
super.show();
System.out.println("再加份牛肉,多加3元...");
}
}
输出结果: