跳到主要内容

单品立减

一、功能说明

1、单品立减促销活动属于商家店铺可直接发布的促销活动。

2、同一个商家在同一时间段内只允许创建一个单品立减促销活动。

3、商家在发布单品立减活动时,可以选择店铺全部商品参与,也可以选择部分商品参与。

选择全部商品参与时,商家新创建了一个商品,那么这个商品也会自动参与到这个活动中。

4、单品立减活动开始后,不允许修改和删除活动信息。

5、对于参与单品立减促销活动的商品,用户购买商品享受的优惠可以叠加,如下:

例:参与活动的商品A单价为100元,活动规则是满100元立减2元。

购买数量(件)应支付金额(元)实支付金额(元)优惠金额(元)
1100982
22001964
33002946
44003928

二、数据库设计

1、表结构设计

单品立减促销活动表—es_minus

字段名类型与长度说明
minus_idbigint(20)主键ID
single_reduction_valuedecimal(20,2)单品立减金额
start_timebigint(20)活动开始时间(存放的是以秒为单位的时间戳)
start_time_strvarchar(50)活动开始时间(格式为2020-10-29 08:00:00),此字段属预留字段暂时无用
end_timebigint(20)活动结束时间(存放的是以秒为单位的时间戳)
end_time_strvarchar(50)活动结束时间(格式为2020-10-29 08:00:00)),此字段属预留字段暂时无用
titlevarchar(50)活动标题(名称)
range_typeint(1)商品参与活动的方式,1:全部商品,2:部分商品
disabledint(1)活动是否停用(删除),0:否,1:是
descriptionlongtext活动说明(描述/简介)
seller_idbigint(20)活动所属商家ID(关联商家店铺表--es_shop)

2、表关联说明

image-20201029154030015

单品立减促销活动表(es_minus)促销活动商品表(es_promotion_goods)
minus_idactivity_id
无字段,促销类型值为MINUSpromotion_type

三、缓存设计

1、商家在发布单品立减促销活动时,在将促销活动信息入库的同时,也会将信息放入缓存中。

缓存key值为:{STOREID_MINUS_KEY}活动ID。

缓存value值为:MinusDO.java这个实体对象信息。

2、单品立减促销活动脚本引擎

  • 脚本引擎缓存结构:

    单品立减促销活动的促销脚本引擎缓存结构有两种:

    • 当发布活动时,如果选择的是全部商品参与,那么存放的是店铺级别的缓存结构。
    • 当发布活动时,如果选择的是部分商品参与,那么存放的是SKU级别的缓存结构。
  • 脚本引擎生成和删除时机:

    • 生成:活动开始时生成。
    • 删除:活动结束时删除。

关于促销脚本引擎缓存结构可参考《促销活动脚本引擎生成架构》这篇文档。

四、代码设计

1、接口调用流程图

image-20201029164452694

2、相关代码展示

以新增单品立减活动为例

  • API代码--MinusSellerController

    @RestController
    @RequestMapping("/seller/promotion/minus")
    @Api(description = "单品立减相关API")
    @Validated
    public class MinusSellerController {

    @ApiOperation(value = "添加单品立减", response = MinusVO.class)
    @ApiImplicitParam(name = "minus", value = "单品立减信息", required = true, dataType = "MinusVO", paramType = "body")
    @PostMapping
    public MinusVO add(@ApiIgnore @Valid @RequestBody MinusVO minus) {
    PromotionValid.paramValid(minus.getStartTime(),minus.getEndTime(),
    minus.getRangeType(),minus.getGoodsList());
    // 获取当前登录的店铺ID
    Seller seller = UserContext.getSeller();
    Long sellerId = seller.getSellerId();
    minus.setSellerId(sellerId);
    this.minusManager.add(minus);
    return minus;
    }
    }
  • 新增单品立减活动业务层代码--MinusManagerImpl

    @Service
    public class MinusManagerImpl extends AbstractPromotionRuleManagerImpl implements MinusManager {

    @Override
    @Transactional(propagation = Propagation.REQUIRED,
    rollbackFor = {RuntimeException.class,
    Exception.class, ServiceException.class})
    public MinusVO add(MinusVO minusVO) {
    //检测开始时间和结束时间
    PromotionValid.paramValid(minusVO.getStartTime(),minusVO.getEndTime(),1,null);
    this.verifyTime(minusVO.getStartTime(),minusVO.getEndTime(),
    PromotionTypeEnum.MINUS,null);

    //初步形成商品的DTO列表
    List<PromotionGoodsDTO> goodsDTOList = new ArrayList<>();
    //是否是全部商品参与
    if(minusVO.getRangeType() == 1){
    PromotionGoodsDTO goodsDTO = new PromotionGoodsDTO();
    goodsDTO.setGoodsId(-1L);
    goodsDTO.setSkuId(-1L);
    goodsDTO.setGoodsName("全部商品");
    goodsDTO.setThumbnail("path");
    goodsDTOList.add(goodsDTO);
    minusVO.setGoodsList(goodsDTOList);
    }
    //检测活动规则
    this.verifyRule(minusVO.getGoodsList());

    MinusDO minusDO = new MinusDO();
    BeanUtils.copyProperties(minusVO,minusDO);
    minusMapper.insert(minusDO);

    // 获取活动Id
    Long minusId = minusDO.getMinusId();
    minusDO.setMinusId(minusId);
    minusVO.setMinusId(minusId);

    PromotionDetailDTO detailDTO = new PromotionDetailDTO();
    detailDTO.setStartTime(minusVO.getStartTime());
    detailDTO.setEndTime(minusVO.getEndTime());
    detailDTO.setActivityId(minusVO.getMinusId());
    detailDTO.setPromotionType(PromotionTypeEnum.MINUS.name());
    detailDTO.setTitle(minusVO.getTitle());

    //将活动商品入库
    this.promotionGoodsManager.add(minusVO.getGoodsList(),detailDTO);

    String minusKey = PromotionCacheKeys.getMinusKey(minusId);
    cache.put(minusKey, minusDO);

    //启用延时任务创建促销活动脚本信息
    PromotionScriptMsg promotionScriptMsg = new PromotionScriptMsg();
    promotionScriptMsg.setPromotionId(minusId);
    promotionScriptMsg.setPromotionName(minusDO.getTitle());
    promotionScriptMsg.setPromotionType(PromotionTypeEnum.MINUS);
    promotionScriptMsg.setOperationType(ScriptOperationTypeEnum.CREATE);
    promotionScriptMsg.setEndTime(minusDO.getEndTime());
    String uniqueKey = "{TIME_TRIGGER_" + PromotionTypeEnum.MINUS.name()
    + "}_" + minusId;
    timeTrigger.add(TimeExecute.SELLER_PROMOTION_SCRIPT_EXECUTER,
    promotionScriptMsg, minusDO.getStartTime(), uniqueKey);

    return minusVO;
    }
    }
  • 单品立减促销活动脚本生成--PromotionScriptTimeTriggerExecuter

    @Component("promotionScriptTimeTriggerExecuter")
    public class PromotionScriptTimeTriggerExecuter implements TimeTriggerExecuter {
    @Override
    public void execute(Object object) {

    PromotionScriptMsg promotionScriptMsg = (PromotionScriptMsg) object;

    //如果是促销活动开始
    if (ScriptOperationTypeEnum.CREATE
    .equals(promotionScriptMsg.getOperationType())) {

    //创建促销活动脚本
    this.createScript(promotionScriptMsg);

    //促销活动开始后,立马设置一个促销活动结束的流程
    promotionScriptMsg.setOperationType(ScriptOperationTypeEnum.DELETE);
    String uniqueKey = "{TIME_TRIGGER_"
    + promotionScriptMsg.getPromotionType().name() + "}_"
    + promotionScriptMsg.getPromotionId();

    timeTrigger.add(TimeExecute.SELLER_PROMOTION_SCRIPT_EXECUTER,
    promotionScriptMsg, promotionScriptMsg.getEndTime(),
    uniqueKey);

    this.logger.debug("促销活动[" + promotionScriptMsg.getPromotionName()
    + "]开始,id=[" + promotionScriptMsg.getPromotionId()
    + "]");
    } else {

    //删除缓存中的促销脚本数据
    this.deleteScript(promotionScriptMsg);

    this.logger.debug("促销活动[" + promotionScriptMsg.getPromotionName()
    + "]结束,id=[" + promotionScriptMsg.getPromotionId()
    + "]");
    }
    }

    /**
    * 创建促销活动脚本
    * @param promotionScriptMsg 促销活动脚本消息信息
    */
    private void createScript(PromotionScriptMsg promotionScriptMsg) {
    //获取促销活动类型
    PromotionTypeEnum promotionType = promotionScriptMsg.getPromotionType();

    //获取促销活动ID
    Long promotionId = promotionScriptMsg.getPromotionId();

    //获取促销脚本相关数据
    ScriptVO scriptVO = this.getPromotionScript(promotionId, promotionType.name());

    //获取商家ID
    Long sellerId = scriptVO.getSellerId();
    //购物车(店铺)级别缓存key
    String cartCacheKey = CachePrefix.CART_PROMOTION.getPrefix() + sellerId;
    //获取商品参与促销活动的方式 1:全部商品参与,2:部分商品参与
    Integer rangeType = scriptVO.getRangeType();
    //构建促销脚本数据结构
    PromotionScriptVO promotionScriptVO = scriptVO.getPromotionScriptVO();

    //如果是全部商品都参与了促销活动
    if (rangeType.intValue() == 1) {

    //构建新的促销脚本数据
    List<PromotionScriptVO> scriptList = getScriptList(cartCacheKey,
    promotionScriptVO);

    //将促销脚本数据放入缓存中
    cache.put(cartCacheKey, scriptList);

    } else {
    //获取参与促销活动的商品集合
    List<PromotionGoodsDO> goodsList = scriptVO.getGoodsList();

    //批量放入缓存的数据集合
    Map<String, List<PromotionScriptVO>> cacheMap = new HashMap<>();

    //循环skuID集合,将脚本放入缓存中
    for (PromotionGoodsDO goods : goodsList) {

    PromotionScriptVO newScript = new PromotionScriptVO();
    BeanUtil.copyProperties(promotionScriptVO, newScript);

    //缓存key
    String cacheKey = CachePrefix.SKU_PROMOTION.getPrefix()
    + goods.getSkuId();

    //脚本结构中加入商品skuID
    newScript.setSkuId(goods.getSkuId());

    //构建新的促销脚本数据
    List<PromotionScriptVO> scriptList = getScriptList(cacheKey, newScript);

    cacheMap.put(cacheKey, scriptList);
    }

    //将sku促销脚本数据批量放入缓存中
    cache.multiSet(cacheMap);

    }
    }

    /**
    * 删除促销活动脚本
    * @param promotionScriptMsg 促销活动脚本消息信息
    */
    private void deleteScript(PromotionScriptMsg promotionScriptMsg) {
    //获取促销活动类型
    PromotionTypeEnum promotionType = promotionScriptMsg.getPromotionType();

    //获取促销活动ID
    Long promotionId = promotionScriptMsg.getPromotionId();

    if (PromotionTypeEnum.FULL_DISCOUNT.equals(promotionType)) {
    //满减满赠促销活动
    //...省略...

    } else if (PromotionTypeEnum.MINUS.equals(promotionType)) {
    //获取单品立减促销活动信息
    MinusVO minusVO = this.promotionGoodsClient.getMinusFromDB(promotionId);
    //初始化缓存中的促销活动脚本信息
    initScriptCache(minusVO.getRangeType(), minusVO.getSellerId(),
    promotionId, promotionType.name());

    } else if (PromotionTypeEnum.HALF_PRICE.equals(promotionType)) {
    //第二件半价促销活动
    //...省略...
    }
    }

    /**
    * 根据促销活动ID和促销活动类型获取促销脚本数据
    * @param promotionId 促销活动id
    * @param promotionType 促销活动类型
    * @return
    */
    private ScriptVO getPromotionScript(Long promotionId, String promotionType) {

    ScriptVO scriptVO = new ScriptVO();

    if (PromotionTypeEnum.FULL_DISCOUNT.name().equals(promotionType)) {
    //满减满赠促销活动
    //...省略...

    } else if (PromotionTypeEnum.MINUS.name().equals(promotionType)) {

    //获取单品立减促销活动信息
    MinusVO minusVO = this.promotionGoodsClient.getMinusFromDB(promotionId);

    //渲染并读取单品立减促销活动脚本信息
    String script = renderMinusScript(minusVO);

    String tips = "单品立减";

    //构建促销脚本数据结构
    PromotionScriptVO promotionScriptVO
    = getScript(script, promotionId, false, promotionType, tips);

    scriptVO.setSellerId(minusVO.getSellerId());
    scriptVO.setRangeType(minusVO.getRangeType());
    scriptVO.setPromotionScriptVO(promotionScriptVO);

    } else if (PromotionTypeEnum.HALF_PRICE.name().equals(promotionType)) {
    //第二件半价促销活动
    //...省略...

    }

    //如果是部分商品参与活动,需要查询出参与促销活动的商品skuID集合
    if (scriptVO.getRangeType().intValue() == 2) {
    List<PromotionGoodsDO> goodsList
    = this.promotionGoodsClient.getPromotionGoods(promotionId, promotionType);
    scriptVO.setGoodsList(goodsList);
    }

    return scriptVO;
    }

    /**
    * 渲染并读取单品立减促销活动脚本信息
    * @param minusVO 单品立减促销活动信息
    * @return
    */
    private String renderMinusScript(MinusVO minusVO) {
    Map<String, Object> model = new HashMap<>();

    Map<String, Object> params = new HashMap<>();
    params.put("startTime", minusVO.getStartTime().toString());
    params.put("endTime", minusVO.getEndTime().toString());
    params.put("singleReductionValue", minusVO.getSingleReductionValue());

    model.put("promotionActive", params);

    String path = "minus.ftl";
    String script = ScriptUtil.renderScript(path, model);

    logger.debug("生成单品立减促销活动脚本:" + script);

    return script;
    }
    }
  • 单品立减促销活动脚本引擎模板文件--minus.ftl

    <#--
    验证促销活动是否在有效期内
    @param promotionActive 活动信息对象(内置常量)
    .startTime 获取开始时间
    .endTime 活动结束时间
    @param $currentTime 当前时间(变量)
    @returns {boolean}
    -->
    function validTime(){
    if (${promotionActive.startTime} <= $currentTime && $currentTime <= ${promotionActive.endTime}) {
    return true;
    }
    return false;
    }

    <#--
    单品立减活动金额计算
    @param promotionActive 单品立减活动信息对象(内置常量)
    .singleReductionValue 立减金额
    @param $sku 商品SKU信息对象(变量)
    .$price 商品SKU单价
    .$num 商品数量
    @returns {*}
    -->
    function countPrice() {
    var resultPrice = ($sku.$price - ${promotionActive.singleReductionValue}) * $sku.$num;
    return resultPrice < 0 ? 0 : resultPrice.toString();
    }