当前位置: 首页 > article >正文

设计模式在实际业务中应用 - 模版方法

1. 业务背景

作者在工作中主要主导 A 业务线的系统建设,A 业务线主要是零售场景酒水的售卖与即时配送服务。为了方便运营在自研系统中对多平台商品进行管理而开发的三方平台商品管理功能,本次介绍的模版方法模式则是在该功能开发过程中的落地实践。

2. 技术方案选型

该业务场景可选设计为三种:

  • 自研系统根据自身业务形态对商品领域进行抽象建模,在自研系统里对商品领域的操作最终映射到三方平台;

  • 自研系统对三方平台商品进行抽象建模,保存三方平台商品在某个时刻的快照,所有操作均是对快照的操作,进而最终映射到三方平台;

  • 自研系统充当一个适配器和转发层,所有对商品的操作都直接映射到三方平台;

从系统建设角度来说最好的选择是第一种,但是对于一个业务方想要更快的提升运营效率来说,更好的选择是第二种和第三种,我们从自身业务情况、性能、后续向第一种方案演进的角度选择了第二种方案。

3. 模版方法应用

以下代码仅为演示对模版方法的应用和展示代码设计思路,有部分方法并未实现,但通过注释说明方法内做了什么操作。

3.1. 模版抽象类

@Component
public abstract class ThirdPlatformStoreGoodsOperations {

    /**
     * 商品上架
     * @param storeGoodsShelvesParam 商品上架参数
     * @return 新商品记录ID
     */
    public Long shelves(StoreGoodsShelvesParam storeGoodsShelvesParam) {

        // 获取门店本地商品
        ThirdPlatformGoods localStoreGoods = this.getLocalStoreGoods(storeGoodsShelvesParam.getId());
        if (localStoreGoods == null) {
            throw new HistoricalDataException("商品数据已更新,请刷新当前页面");
        }

        // 上架三方平台门店商品
        this.shelvesThirdPlatformStoreGoods(localStoreGoods);

        // 刷新本地商品
        StoreGoodsRefreshParam storeGoodsRefreshParam = new StoreGoodsRefreshParam();
        BeanUtils.copyProperties(storeGoodsShelvesParam, storeGoodsRefreshParam);
        return this.refreshOne(storeGoodsRefreshParam);
    }

    /**
     * 商品下架
     * @param storeGoodsWithdrawParam 商品下架参数
     * @return 新商品记录ID
     */
    public Long withdraw(StoreGoodsWithdrawParam storeGoodsWithdrawParam) {

        // 获取门店本地商品
        ThirdPlatformGoods localStoreGoods = this.getLocalStoreGoods(storeGoodsWithdrawParam.getId());
        if (localStoreGoods == null) {
            throw new HistoricalDataException("商品数据已更新,请刷新当前页面");
        }

        // 下架三方平台门店商品
        this.withdrawThirdPlatformStoreGoods(localStoreGoods);

        // 刷新本地商品
        StoreGoodsRefreshParam storeGoodsRefreshParam = new StoreGoodsRefreshParam();
        BeanUtils.copyProperties(storeGoodsWithdrawParam, storeGoodsRefreshParam);
        return this.refreshOne(storeGoodsRefreshParam);
    }

    /**
     * 增加商品库存
     * @param storeGoodsIncreaseStockParam 增加商品库存参数
     * @return 新商品记录ID
     */
    public Long increaseStock(StoreGoodsIncreaseStockParam storeGoodsIncreaseStockParam) {

        // 获取门店本地商品
        ThirdPlatformGoods localStoreGoods = this.getLocalStoreGoods(storeGoodsIncreaseStockParam.getId());
        if (localStoreGoods == null) {
            throw new HistoricalDataException("商品数据已更新,请刷新当前页面");
        }

        // 增加三方平台店铺商品库存
        this.increaseThirdPlatformStoreGoodsStock(storeGoodsIncreaseStockParam, localStoreGoods);

        // 刷新本地商品
        StoreGoodsRefreshParam storeGoodsRefreshParam = new StoreGoodsRefreshParam();
        BeanUtils.copyProperties(storeGoodsIncreaseStockParam, storeGoodsRefreshParam);
        return this.refreshOne(storeGoodsRefreshParam);
    }

    /**
     * 增加商品库存
     * @param storeGoodsDecreaseStockParam 减少商品库存参数
     * @return 新商品记录ID
     */
    public Long decreaseStock(StoreGoodsDecreaseStockParam storeGoodsDecreaseStockParam) {

        // 获取门店本地商品
        ThirdPlatformGoods localStoreGoods = this.getLocalStoreGoods(storeGoodsDecreaseStockParam.getId());
        if (localStoreGoods == null) {
            throw new HistoricalDataException("商品数据已更新,请刷新当前页面");
        }

        // 减少三方平台店铺商品库存
        this.decreaseThirdPlatformStoreGoodsStock(storeGoodsDecreaseStockParam, localStoreGoods);

        // 刷新本地商品
        StoreGoodsRefreshParam storeGoodsRefreshParam = new StoreGoodsRefreshParam();
        BeanUtils.copyProperties(storeGoodsDecreaseStockParam, storeGoodsRefreshParam);
        return this.refreshOne(storeGoodsRefreshParam);
    }

    /**
     * 刷新店铺单个商品
     * @param storeGoodsRefreshParam 刷新店铺单个商品参数
     * @return 新商品记录ID
     */
    public Long refreshOne(StoreGoodsRefreshParam storeGoodsRefreshParam) {

        // 获取门店本地商品
        ThirdPlatformGoods localStoreGoods = this.getLocalStoreGoods(storeGoodsRefreshParam.getId());
        if (localStoreGoods == null) {
            throw new HistoricalDataException("商品数据已更新,请刷新当前页面");
        }

        // 获取三方门店商品
        StoreGoodsInfoResult storeGoodsInfoResult = this.getThirdPlatformStoreGoods(storeGoodsRefreshParam, localStoreGoods);

        // 刷新本地商品
        return this.refreshLocalStoreGoods(storeGoodsRefreshParam, storeGoodsInfoResult);
    }

    /**
     * 刷新店铺商品
     * @param storeGoodsRefreshParam 刷新店铺单个商品参数
     * @param storeGoodsInfoResult 三方门店商品
     */
    private Long refreshLocalStoreGoods(StoreGoodsRefreshParam storeGoodsRefreshParam,
                                        StoreGoodsInfoResult storeGoodsInfoResult) {
        // 更新本地商品
    }

    /**
     * 获取本地店铺商品信息
     * @param localStoreGoodsId 本地商品ID
     * @return 本地店铺商品信息
     */
    private ThirdPlatformGoods getLocalStoreGoods(Long localStoreGoodsId) {
        // 获取本地商品
    }

    /**
     * 上架三方店铺商品
     * @param localStoreGoods 本地店铺商品信息
     */
    protected abstract void shelvesThirdPlatformStoreGoods(ThirdPlatformGoods localStoreGoods);

    /**
     * 下架三方店铺商品
     * @param localStoreGoods 本地店铺商品信息
     */
    protected abstract void withdrawThirdPlatformStoreGoods(ThirdPlatformGoods localStoreGoods);

    /**
     * 获取三方平台店铺商品
     * @param storeGoodsRefreshParam 刷新店铺单个商品参数
     * @param localStoreGoods 本地商品
     * @return 三方门店商品
     */
    protected abstract StoreGoodsInfoResult getThirdPlatformStoreGoods(StoreGoodsRefreshParam storeGoodsRefreshParam,
                                                                       ThirdPlatformGoods localStoreGoods);

    /**
     * 增加三方平台店铺商品库存
     * @param storeGoodsIncreaseStockParam 增加三方平台店铺商品库存参数
     * @param localStoreGoods 本地店铺商品信息
     */
    protected abstract void increaseThirdPlatformStoreGoodsStock(StoreGoodsIncreaseStockParam storeGoodsIncreaseStockParam,
                                                                 ThirdPlatformGoods localStoreGoods);

    /**
     * 减少三方平台店铺商品库存
     * @param storeGoodsDecreaseStockParam 减少三方平台店铺商品库存参数
     * @param localStoreGoods 本地店铺商品信息
     */
    protected abstract void decreaseThirdPlatformStoreGoodsStock(StoreGoodsDecreaseStockParam storeGoodsDecreaseStockParam,
                                                                 ThirdPlatformGoods localStoreGoods);

    /**
     * 获取对应平台
     * @return 平台枚举
     */
    protected abstract ThirdPlatformEnums getPlatform();
}

3.2. 模版实现类

public class ELEStoreGoodsTemplate extends ThirdPlatformStoreGoodsOperations {

    @Override
    protected void shelvesThirdPlatformStoreGoods(ThirdPlatformGoods localStoreGoods) {
        // 饿了么平台商品上架
    }

    @Override
    protected void withdrawThirdPlatformStoreGoods(ThirdPlatformGoods localStoreGoods) {
        // 饿了么平台商品下架
    }

    @Override
    protected StoreGoodsInfoResult getThirdPlatformStoreGoods(StoreGoodsRefreshParam storeGoodsRefreshParam,
                                                              ThirdPlatformGoods localStoreGoods) {
        // 获取饿了么平台商品信息
        return null;
    }

    @Override
    protected void increaseThirdPlatformStoreGoodsStock(StoreGoodsIncreaseStockParam storeGoodsIncreaseStockParam,
                                                        ThirdPlatformGoods localStoreGoods) {
        // 增加饿了么平台商品库存
    }

    @Override
    protected void decreaseThirdPlatformStoreGoodsStock(StoreGoodsDecreaseStockParam storeGoodsDecreaseStockParam, ThirdPlatformGoods localStoreGoods) {

        // 减少饿了么平台商品库存
    }

    @Override
    protected ThirdPlatformEnums getPlatform() {
        return ThirdPlatformEnums.ELEMO;
    }
}

4. 与 Spring 结合管理模版实现类

4.1. 基于 Spring IoC 容器依赖查找管理模版

@Component
public class ThirdPlatformGoodsTemplateManage {

    public Map<ThirdPlatformEnums, ThirdPlatformStoreGoodsOperations> templateMap = new HashMap<>(5);

    public ThirdPlatformGoodsTemplateManage(ListableBeanFactory listableBeanFactory){

        ObjectProvider<ThirdPlatformStoreGoodsOperations> beanProvider = listableBeanFactory.getBeanProvider(ThirdPlatformStoreGoodsOperations.class);
        for (ThirdPlatformStoreGoodsOperations template : beanProvider) {
            templateMap.put(template.getPlatform(), template);
        }
    }

    public ThirdPlatformStoreGoodsOperations getTemplate(ThirdPlatformEnums thirdPlatformEnum) {

        ThirdPlatformStoreGoodsOperations thirdPlatformStoreGoodsOperations = templateMap.get(thirdPlatformEnum);
        Assert.notNull(thirdPlatformStoreGoodsOperations, String.format("%s平台模版未找到!", thirdPlatformEnum.getDesc()));
        return thirdPlatformStoreGoodsOperations;
    }

    public ThirdPlatformStoreGoodsOperations getTemplate(String platformCode) {
        ThirdPlatformEnums thirdPlatformEnum = ThirdPlatformEnums.getInstanceByCode(platformCode);
        ThirdPlatformStoreGoodsOperations thirdPlatformStoreGoodsOperations = templateMap.get(thirdPlatformEnum);
        Assert.notNull(thirdPlatformStoreGoodsOperations, String.format("%s平台模版未找到!", platformCode));
        return thirdPlatformStoreGoodsOperations;
    }
}

4.2. 基于 BeanPostProcessor + 注解方式管理模版

@Component
public class ThirdPlatformStoreGoodsOperationsInitialization implements BeanPostProcessor {

    @Resource
    private ThirdPlatformGoodsTemplateManage thirdPlatformGoodsTemplateManage;

    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        if (bean instanceof ThirdPlatformStoreGoodsOperations && bean.getClass().isAnnotationPresent(ThirdPlatformStoreGoodsOperationsTemplate.class)){
            ThirdPlatformStoreGoodsOperations thirdPlatformStoreGoodsOperations = (ThirdPlatformStoreGoodsOperations) bean;
            thirdPlatformGoodsTemplateManage.templateMap.put(thirdPlatformStoreGoodsOperations.getPlatform(), thirdPlatformStoreGoodsOperations);
        }
        return bean;
    }
}

改进建议: 模版CODE可通过注解属性获取

5. 为什么选择模版方法

看完代码实现再回过头来,说说选择模版方法的原因:

  • 从功能角度来说对三方平台商品管理无非就是那么几个商品新增、商品编辑、商品库存调整、商品上下架等;
  • 从商品领域建模对三方平台商品领域行为抽象可分为:创建、修改、增加库存、减少库存、覆盖库存、上架、下架等;
  • 从代码复用的角度来说不同平台商品的操作不同的只有最终调用三方平台的差异上,其他代码都可复用,假设第一次开发只先适配美团,那么在美团适配后复用的代码理论上不需要做任何修改;
  • 从代码可扩展角度来说,如果增加其他平台,仅需要继承模版抽象类,实现抽象方法即可;
  • 从关注点分离角度来说后续适配其他平台仅仅需要关注如何操作三方平台即可,这对新人或者不了解的人来继续迭代有很大好处,关注点集中在如何适配三方平台上;

http://www.kler.cn/a/148461.html

相关文章:

  • 设计模式之工厂模式,但是宝可梦
  • python制作一个简单的端口扫描器,用于检测目标主机上指定端口的开放状态
  • 山泽光纤HDMI线:铜线的隐藏力量
  • 【专题】计算机网络之网络层
  • idea 解决缓存损坏问题
  • pip3 install -e .[stable]讲解
  • 万媒易发:以RPA自动化和AIGC为基础实现多平台分发
  • modbus协议及modbus TCP协议
  • 类指针压缩空间
  • 【Vue】图片切换
  • 【视觉SLAM十四讲学习笔记】第三讲——四元数
  • 一些关于开关电源经典回答
  • Java面向对象第6天
  • C 标准库 - <stdlib.h>和<string.h>详解
  • 基于mvc的大学生家教信息网站系统php+vue
  • INFINI Gateway 与华为鲲鹏完成产品兼容互认证
  • 5.golang字符串的拆解和拼接
  • 耗时一个星期整理的APP自动化测试工具大全
  • 【网络】传输层 --- 详解TCP协议
  • lv11 嵌入式开发 WDT实验 12
  • C语言:输入10个整数,写一个函数将其中最小的数和第一个数对换,把最大的数和最后一个数对换。(指针)
  • 14 网关实战:网关聚合API文档
  • 基于51单片机冰箱温度控制器设计
  • Sass混合器的详细使用教程
  • squid代理服务器(传统代理、透明代理、反向代理、ACL、日志分析)
  • ESP32-Web-Server编程- JS 基础 3