策略模式在公司项目中的运用实践,看完又可以涨一波实战经验了!

简介: 策略模式在公司项目中的运用实践,看完又可以涨一波实战经验了!

营销系统是一个动态的、有机地结合的系统,经常会随着业务的不断变化发生调整,因此从事这一业务的开发可让我头疼了。


之前在工作中就不乏一次遇到过随意调整营销策略的情况,在部分场景下由于使用了硬编码的方式来实现,因此在调整策略的时候显得特别不灵活。


下边我列举一个曾经遇到过的应用场景:


业务部门需要上线一款新型的产品,用户在线上购买了对应的产品,然后下单支付之后需要享受不同的服务内容,这些服务包含了赠送优惠券,发送红包补贴,加积分,升级等服务项。并且上线之后,可能会随着市场的因素的调整,部分服务内容也会有所下架,后期调整因素极高。


下边是一张用户建模的图:


image.png


线上买单,到选择购买的产品类型,再到后续下单之后执行不同的营销规则,每个产品对应不同的服务项目并且服务项目的内容还可能会随时调整。


举个实际案例来说,线上有这么几款服务产品供消费者选购:


1.999元会员套餐


  • 正常会员服务期1个月


  • 发放5张优惠券


2.1999元会员套餐


  • 正常会员服务期2个月


  • 发放6张优惠券


  • 邀请新人加入app,新人在n天内购买套餐有优惠


3.2999元会员套餐



  • 正常会员服务期3个月
  • 发放7张优惠券
  • 满2500元消费,返现50元红包

….


大致看看,不同的产品对应不同的促销规则,似乎毫无规律可言。


但是如果通过抽象的逻辑将其中的共同部分抽取出来,就会发现其实是有规则可循了。

下边我给出来一段 “不那么完整的代码案例” (关于这种营销手段的设计核心在于思路,没有完美的代码,只有不断精进的设计)


这段代码主要采用来策略模式的设计思路,不同的产品对应不同的策略,产品和策略之间的关联可以通过使用数据库的方式来做绑定。


首先可以将每个服务项目看作是一条营销的规则手段,因此我定义来一个marketing对象:


/**
 * 营销对象实体类
 *
 * @Author idea
 * @Date created in 9:39 上午 2020/5/4
 */
@NoArgsConstructor
@Data
@Builder
@AllArgsConstructor
public class MarketingPO {
    /**
     * 主键id
     */
    private Integer id;
    /**
     * 营销手段名称 存储class的名称
     */
    private String marketingName;
    /**
     * 入参 多个可以逗号分割
     */
    private String inputVal;
    /**
     * 描述
     */
    private String des;
    /**
     * 创建时间
     */
    private Date createTime;
    /**
     * 更新时间
     */
    private Date updateTime;
}


接着便是产品和不同营销手段之间做关联


/**
 * 通过产品id和营销手段做关联
 *
 * @Author idea
 * @Date created in 3:37 下午 2020/5/4
 */
@Data
@Builder
@AllArgsConstructor
@NoArgsConstructor
public class MarketingProductPO {
    /**
     * 主键id
     */
    private Integer id;
    /**
     * 营销工具id
     */
    private Integer marketingId;
    /**
     * 产品编号
     */
    private String productNo;
    /**
     * 描述
     */
    private String des;
    /**
     * 是否有效
     */
    private Integer validStatus;
    /**
     * 创建时间
     */
    private Date createTime;
    /**
     * 更新时间
     */
    private Date updateTime;
}


接着是dao层的部分,不过这里我简单化地将持久层逻辑写在来代码里面,只做参考:


/**
 * 模拟dao层操作
 *
 * @Author idea
 * @Date created in 10:20 上午 2020/5/4
 */
@Repository
public class MarketingDao implements IMarketingDao {
    private static List<MarketingPO> MARKETING_LIST = new ArrayList();
    static {
        MarketingPO disCountMarket = MarketingPO.builder()
                .id(1).marketingName("com.sise.idea.present.impl.DiscountStrategy").des("折扣优惠").inputVal("7").build();
        MarketingPO redPacketMarket = MarketingPO.builder()
                .id(2).marketingName("com.sise.idea.present.impl.RedPacketStrategy").des("红包优惠").inputVal("8").build();
        MarketingPO newMemberCouponMarket = MarketingPO.builder()
                .id(3).marketingName("com.sise.idea.present.impl.NewMemberCouponStrategy").des("新人优惠券发送").inputVal("10").build();
        MARKETING_LIST.add(newMemberCouponMarket);
        MARKETING_LIST.add(disCountMarket);
        MARKETING_LIST.add(redPacketMarket);
    }
    @Override
    public List<MarketingPO> selectMarketingByIds(List<Integer> idList) {
        List<MarketingPO> marketingPOS = new ArrayList<>(idList.size());
        for (MarketingPO marketingPO : MARKETING_LIST) {
            if (idList.contains(marketingPO.getId())) {
                marketingPOS.add(marketingPO);
            }
        }
        return marketingPOS;
    }
}
/**
 * @Author idea
 * @Date created in 3:45 下午 2020/5/4
 */
@Repository
public class MarketingProductDao implements IMarketingProductDao {
    private static List<MarketingProductPO> MARKET_PRODUCT_LIST = new ArrayList<>();
    static {
        MarketingProductPO marketingProductPO = MarketingProductPO.builder()
                .productNo("p111")
                .marketingId(2)
                .validStatus(1)
                .des("2999套餐-发放优惠券")
                .build();
        MarketingProductPO marketingProductPO2 = MarketingProductPO.builder()
                .productNo("p111")
                .marketingId(3)
                .validStatus(1)
                .des("2999套餐-满额红包返现")
                .build();
        MARKET_PRODUCT_LIST.add(marketingProductPO);
        MARKET_PRODUCT_LIST.add(marketingProductPO2);
    }
    @Override
    public List<MarketingProductPO> selectByProductNo(String productNo) {
        List<MarketingProductPO> marketingProductPOS = new ArrayList<>();
        for (MarketingProductPO marketingProductPO : MARKET_PRODUCT_LIST) {
            //产品编码一致 而且规则有效
            if(marketingProductPO.getProductNo().equals(productNo) && marketingProductPO.getValidStatus()==1){
                marketingProductPOS.add(marketingProductPO);
            }
        }
        return marketingProductPOS;
    }
}


接着便是对所有的营销手段都做了一层统一的封装和抽象:


package com.sise.策略模式.present;
/**
 * 关于营销手段的策略
 *
 * @Author idea
 * @Date created in 9:20 上午 2020/5/4
 */
public interface IMarketingStrategy {
    /**
     * 服务赠送的策略执行
     *
     * @param param 参数
     * @return
     */
    boolean doMarketing(Object ...param);
}


接下来便是不同的营销手段对应不同的实现,这里面我简单做了一些实现:


@Service
public class RedPacketStrategy implements IMarketingStrategy {
    @Override
    public boolean doMarketing(Object... param) {
        System.out.println("红包赠送策略");
        return false;
    }
}
@Service
public class DiscountStrategy implements IMarketingStrategy {
    @Override
    public boolean doMarketing(Object... param) {
        System.out.println("打折优惠");
        return false;
    }
}
@Service
public class NewMemberCouponStrategy implements IMarketingStrategy {
    @Override
    public boolean doMarketing(Object... param) {
        System.out.println("新人赠送策略");
        return false;
    }
}
@Service
public class UpgradeStrategy implements IMarketingStrategy {
    @Override
    public boolean doMarketing(Object... param) {
        System.out.println("升级策略");
        return false;
    }
}


既然有了不同营销手段的具体实现方式,那么对于购买不同的产品也需要查询到不同的营销手段,这个时候就需要有一个转换中间者的角色出现了:


/**
 * 营销工具核心执行器
 *
 * @Author idea
 * @Date created in 9:34 上午 2020/5/4
 */
public interface IMarketingCoreService {
    /**
     * 执行不同的营销工具
     *
     * @param productNo 产品编码
     * @return
     */
    boolean doMarketingJob(String productNo) throws Exception;
}
/**
 * 营销工具核心执行器
 *
 * @Author idea
 * @Date created in 9:34 上午 2020/5/4
 */
@Service
public class MarketingCoreService implements IMarketingCoreService {
    @Resource
    private IMarketingDao iMarketingDao;
    @Resource
    private IMarketingProductDao iMarketingProductDao;
    @Resource
    private ApplicationContext applicationContext;
    @Override
    public boolean doMarketingJob(String productNo) throws ClassNotFoundException {
        System.out.println("doMarketingJob begin =============");
        System.out.println(productNo);
        List<MarketingProductPO> marketingProductPOS = iMarketingProductDao.selectByProductNo(productNo);
        if (marketingProductPOS != null) {
            List<Integer> marketingIdList = marketingProductPOS.stream().map(MarketingProductPO::getMarketingId).collect(Collectors.toList());
            List<MarketingPO> marketingPOS = iMarketingDao.selectMarketingByIds(marketingIdList);
            for (MarketingPO marketingPO : marketingPOS) {
                String marketingName = marketingPO.getMarketingName();
                Class<?> clazz = Class.forName(marketingName);
                IMarketingStrategy marketingStrategy = (IMarketingStrategy) applicationContext.getBean(clazz);
                marketingStrategy.doMarketing(marketingPO.getInputVal());
            }
            System.out.println("doMarketingJob end =============");
            return true;
        }
        System.out.println("doMarketingJob setting is empty ===========");
        return false;
    }
}


具体的思路就和策略模式有点类似:


策略模式


模式定义:定义一系列算法,将每个算法都封装起来,并且它们可以互换。策略模式是一种对象行为模式。


例如下图:


image.png


最后为了方便测试,我在工程里面引入了spring-context的依赖:


<!-- 关于spring的核型模块代码       -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.0.1.RELEASE</version>
        </dependency>


测试的入口代码:


/**
 * @Author idea
 * @Date created in 10:14 上午 2020/5/4
 */
public class ApplicationDemo {
    public static void main(String[] args) throws Exception {
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext();
        applicationContext.scan("com.sise.idea.present");
        //启动上下文
        applicationContext.refresh();
        IMarketingCoreService marketingCoreService = applicationContext.getBean(MarketingCoreService.class);
        marketingCoreService.doMarketingJob("p111");
    }
}


最后根据规则,通过产品编码来搜索到指定的营销手段,并执行对应的程序逻辑:


image.png


设计不足点


文章上边我曾经提及过,没有完美点代码,只有随着业务需求不断变化的设计思路,因此在真正落地整套营销系统的时候,还需要额外考虑很多的要素。例如说目前的这种设计只能满足于针对单个产品层面,如果以后有出现针对完整订单层面(例如说总支付订单满xxx元,享受xxx优惠)的还需要额外去思考,加上不同的营销手段之间是否有出现互斥的场景都是会有可能遇到的情况。


设计模式,是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性、程序的重用性。文中我并没有过多地去讲解什么是xx模式,但是当通过某种较为灵活的方式来实现某样功能时,可能就已经使用了设计模式。


https://gitee.com/IdeaHome_admin/design_pattern/tree/master/design-model/src/main/java/com/sise/%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/present


END


相关文章
|
Java 程序员
收藏!阿里毕玄16篇文章,深度讲解Java开发、系统设计、职业发展
阿里毕玄结合自己的经历深度讲解Java开发、系统设计、职业发展等问题,快来一键收藏吧。
34824 1
|
5月前
|
存储 边缘计算 Cloud Native
“论模型驱动架构设计方法及其应用”写作框架,软考高级,系统架构设计师
模型驱动架构设计是一种用于应用系统开发的软件设计方法,以模型构造、模型转换和精化为核心,提供了一套软件设计的指导规范。在模型驱动架构环境下,通过创建出机器可读和高度抽象的模型实现对不同问题域的描述,这些模型独立于实现技术,以标准化的方式储存,利用模型转换策略来驱动包括分析、设计和实现等在内的整个软件开发过程。
333 3
|
5月前
|
设计模式 C#
技术经验分享:C#设计模式
技术经验分享:C#设计模式
28 0
|
设计模式 算法 前端开发
软件开发常见的一些设计模式,留着供自己研究和面试使用
说到软件开发,就不得不提到设计模式,比如大家基本上都用过什么MVC框架开发各种系统,一些好的设计模式不仅能让软件运行的更为流畅,更能让开发人员的工作效率大大提高。本文就来列举一些常用的设计模式,供大家参考收藏。
131 1
|
设计模式 Cloud Native 架构师
分享一份美团T9大牛总结的神仙微服务架构设计模式PDF
微服务作为一项在云中部署应用和服务的新技术已成为当下最新的热门话题。 企业和服务提供商正在寻找更好的方法将应用程序部署在云环境中,微服务被认为是未来的方向。通过将应用和服务分解成更小的、松散耦合的组件,它们可以更加容易升级和扩展,理论上是这样。
|
设计模式 算法 程序员
一张截图,告诉你字节跳动的《设计模式宝典》到底有多强
格局小了,算法像是单兵的作战能力和武器装备,设计模式像打仗列的阵型。只是单挑的话, 阵型就不重要了(叫单例模式也可以吧);如果是群斗,请参考戚家军是如何用鸳鸯阵吊打单兵作战能力爆表的日本武士;
|
Dubbo Java 应用服务中间件
策略模式史上最佳实践,没有之一!!!
策略模式史上最佳实践,没有之一!!!
172 0
|
设计模式 安全 关系型数据库
2w行代码、200个实战项目,助你修炼5大编程基本功
2w行代码、200个实战项目,助你修炼5大编程基本功
162 0
|
设计模式 开发框架 Java
设计模式: 实际场景的落地应用(含开源项目和生产环境实践)--持续更新中
设计模式: 实际场景的落地应用(含开源项目和生产环境实践)--持续更新中
1213 2
设计模式: 实际场景的落地应用(含开源项目和生产环境实践)--持续更新中
|
设计模式 uml 容器
大厂技术布道师!就这么使用责任链模式,绝对没错!
大厂技术布道师!就这么使用责任链模式,绝对没错!
131 0
大厂技术布道师!就这么使用责任链模式,绝对没错!
下一篇
无影云桌面