【设计模式通过在苹果官网购买iPhone配件了解装饰器模式
背景
一个iPhone,可以套上保护壳(ConcDecorator A),也可以再套上无线充(ConcDecorator B),得到最后的效果(execute)。
苹果无需发布新的型号(继承),我们简单地通过装饰(组合)就可以得到想要的最终产品。

模式定义
Attach additional responsibilities to an object dynamically keeping the same interface. Decorators provide a flexible alternative to subclassing for extending functionality.
为动态保持相同接口的对象附加额外的职责。装饰器为扩展功能提供了一种灵活的选择,而不是子类化。
可以看到,是十分符合我们的需求的,每一个装饰品如保护壳、无线充,就是一个个可以嵌套的装饰器。

https://refactoringguru.cn/design-patterns/decorator
模式实现

1.定义iPhone接口
主要定义了iPhone模板的说明接口方法getDesc,后续以这个方法为核心,展开嵌套。
package com.example.designpattern.decorator;
/**
 * 手机
 *
 * @author hongcunlin
 */
public interface Phone {
    /**
     * 介绍
     *
     * @return 介绍内容
     */
    String getDesc();
}2.实现iPhone接口
简单试下getDesc的方法,目前是一个裸机的iPhone。
package com.example.designpattern.decorator.impl;
import com.example.designpattern.decorator.Phone;
/**
 * 苹果手机iPhone 14 Pro Max
 *
 * @author hongcunlin
 */
public class Iphone implements Phone {
    @Override
    public String getDesc() {
        return "iPhone";
    }
}3.定义iPhone装饰器抽象类
装饰器的超类,里边存放iPhone属性,围绕它进行嵌套。
package com.example.designpattern.decorator.impl.decorator;
import com.example.designpattern.decorator.Phone;
/**
 * 手机装饰器
 *
 * @author hongcunlin
 */
public abstract class IphoneDecorator implements Phone {
    /**
     * 手机(抽象类存在的意义,否则就使用接口了)
     */
    protected Phone phone;
    /**
     * 构造方法
     *
     * @param phone 手机
     */
    IphoneDecorator(Phone phone) {
        this.phone = phone;
    }
}4.1.实现iPhone装饰器之保护壳
嵌套一
package com.example.designpattern.decorator.impl.decorator;
import com.example.designpattern.decorator.Phone;
/**
 * iPhone保护壳
 *
 * @author hongcunlin
 */
public class IphoneCaseDecorator extends IphoneDecorator {
    /**
     * 构造方法
     *
     * @param phone 手机
     */
    public IphoneCaseDecorator(Phone phone) {
        super(phone);
    }
    @Override
    public String getDesc() {
        return "带保护壳的" + phone.getDesc();
    }
}4.2.实现iPhone装饰器之无线充
嵌套二
package com.example.designpattern.decorator.impl.decorator;
import com.example.designpattern.decorator.Phone;
/**
 * iPhone无线充
 *
 * @author hongcunlin
 */
public class IphoneChargerDecorator extends IphoneDecorator {
    /**
     * 构造方法
     *
     * @param phone 手机
     */
    public IphoneChargerDecorator(Phone phone) {
        super(phone);
    }
    @Override
    public String getDesc() {
        return "带无线充的" + phone.getDesc();
    }
}5.测试
这里我们先初始化一个iPhone,接着带上保护壳看看效果,最后再带上无线充看看效果。
package com.example.designpattern.decorator;
import com.example.designpattern.decorator.impl.Iphone;
import com.example.designpattern.decorator.impl.decorator.IphoneCaseDecorator;
import com.example.designpattern.decorator.impl.decorator.IphoneChargerDecorator;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;
@SpringBootTest
public class DesignPatternTest {
    @Test
    public void test() {
        // 常规iPhone
        Phone iphone = new Iphone();
        System.out.println(iphone.getDesc());
        // 带保护壳的iPhone
        iphone = new IphoneCaseDecorator(iphone);
        System.out.println(iphone.getDesc());
        // 带无线充的带保护壳的iPhone
        iphone = new IphoneChargerDecorator(iphone);
        System.out.println(iphone.getDesc());
    }
}
可以看到,随着装饰器的层层嵌套,iPhone的模样越来越复杂,是符合我们的预期的。
最后
装饰器主要是通过组合的模式,对类内容进行扩展,而不是通过继承的方式,特别是在Java只能单继承的情况,行之有效。
后面有空接着闲聊23种设计模式中的其他种。
 
 
                     
                     
                     
                    