java策略模式实战示例「建议收藏」

java策略模式实战示例「建议收藏」以一个顾客价格计算策略为背景,写一个策略模式的demo参考代码:https://github.com/zhang-xiaoxiang/DesignPatterns23没有用策略模式我们一般是下面的写法,直接写一个类,在类里面直接写策略算法(功能实现)//packagecom.demo.strategy;/***NoStrategy:没有策略的做法*实现起来比较容…

大家好,又见面了,我是你们的朋友全栈君。

现已放在gitee上,可以不下载直接参考一下即: https://gitee.com/zhang-xiao-xiang/zxx-pattern

日常碰到的业务概述

  • 登录类型,支付类型,供应商渠道,不同等级会员享受的优惠券价格不一样,等等业务判断,大量if else导致拓展(侧重新增)极其困难,维护(侧重修改)自然是改起来头痛(其实一个类型的增加[拓展一个类型]往往对应这个类型的增删改查CRUD[维护]),比如业务一开始一个简单的登录,往往做一个电话号码和验证码登录的方式或者账号密码登录方式,后来随着业务的增加或者提高用户体验,那么需要拓展(新增一种)三方登录,比如新增微信登录,支付宝登录,甚至抖音登录,一大堆,你要根据登录方式来处理,那就有点恼火吧,支付方式也是同样的问题,我们可以发现一个规律,凡是可以枚举的业务,往往都需要使用设计模式才能更好的解决,比如策略模式(往往搭配工厂模式使用更配哦),水来土掩,兵来将挡,这思想和高中数学中的分类讨论思想一模一样.遇事不要慌,因为还有dang中央.所以,我们就打个栗子,举个比方,更加形象一点

 由于太多的策略模式时根据顾客VIP等级不同,得到的商品价格不一样的例子,这里还是换个汤,顺便更新了一下博客,以星座自我介绍(根据星座的类型不同,返回不同的信息)为例子,结合springboot实际感受一下在实战时的策略模式样子

先来个直观的对比一睹为快(放大效果更佳)

未使用时:我们经常直接在业务层开始了if else的常规操作

 java策略模式实战示例「建议收藏」

 使用策略模后时:注意看一下描述

java策略模式实战示例「建议收藏」

简单做个结论:这里不是说代码量减少了哈,而是说需要新增或者修改的时候,维护难度就不一样了

策略模式

经过对比分析,发现之所以出现if else和分支,还是少了面向接口编程的思想,做一件事情,假如实现方式多样,那么第一个想到的就是抽象出事情,不管是抽象类也好,做成接口也罢,反正尽量朝着多态的方向去就对了.if else做的事情就是在处理对应星座的描述信息,所以把要描述信息抽取成一个策略方法

1:面向接口编程,这里抽取业务方法(这里有个2个方法是为了对比哈,第二个就是策略模式抽取的)

package com.zhang.zxx.pattern.strategy.service;

/**
 * BusinessService:业务服务层
 *
 * @author zhangxiaoxiang
 * @date 2021/07/18
 */
public interface BusinessService {
    /**
     * 根据星座类型获取星座详情
     * @param type 星座类型 枚举
     * @return 星座描述
     */
    Object getInfo(Integer type);
    /**
     * 根据星座类型获取星座详情
     * @param type 星座类型 枚举
     * @return 星座描述
     */
    Object getInfoWithStrategy(Integer type);
}

再来一个接口 ,这里是处理策略的方法,不是业务的层面的方法

package com.zhang.zxx.pattern.strategy;

import java.util.Map;

/**
 * StrategyService:定义策略接口,,这里可以理解为if(满足条件fetchKey){执行execute的策略 }
 * 联系后面的类可以感觉这里相当于抽象了个map出来
 *
 * @author zhangxiaoxiang
 * @date 17/7/2021
 */
public interface StrategyService {
    /**
     * 匹配策略的key[这个key使用枚举管理最为合理]
     *
     * @return key
     */
    Integer fetchKey();

    /**
     * 匹配后具体策略执行
     *
     * @return 结果[这里为了对数据执行策略结果的收集,选择了Map<String, Object>较为通用,当然void或者object也行,根据实际项目来即可]
     */
    Map<String, Object> execute();
}

2:编写策略接口的实现类,这里举一个实现类就行了,我尽量做到减少篇幅

package com.zhang.zxx.pattern.strategy.strategy;

import com.zhang.zxx.pattern.strategy.MyEnum;
import com.zhang.zxx.pattern.strategy.StrategyService;
import org.springframework.stereotype.Service;

import java.util.HashMap;
import java.util.Map;

/**
 * FirstStrategyImpl:水瓶座策略类[这里具体策略execute比如为知我介绍]
 *
 * @author zhangxiaoxiang
 * @date 2021/07/18
 */
@Service
public class AquariusStrategyImpl implements StrategyService {
    /**
     * 匹配策略的key[这个key使用枚举管理最为合理]
     *
     * @return key
     */
    @Override
    public Integer fetchKey() {
        //水瓶座策略标识
        return MyEnum.AQUARIUS.getNum();
    }

    /**
     * 匹配后具体策略执行
     *
     * @return 结果[这里为了对数据执行策略结果的收集, 选择了Map<String, Object>较为通用,当然void或者object也行,根据实际项目来即可]
     */
    @Override
    public Map<String, Object> execute() {
        Map<String, Object> map=new HashMap<>(16);
        map.put("name","我是水瓶座");
        map.put("birthTime","1月20日~2月18日");
        map.put("luckyNumber","3、5、7");
        return map;
    }
}

3:即便有了策略接口和对应的实现类,但是仍然不能使用,此时需要一个策略辅助或者叫做处理类来帮忙,像一个工厂一样的类,其实就是工程模式的实现哈

package com.zhang.zxx.pattern.strategy;

import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * StrategyHandler:策略处理类[可以理解为策略工厂类]
 *
 * @author zhangxiaoxiang
 * @date 18/7/2021
 */
@Component
@Slf4j
public class StrategyHandler implements InitializingBean, ApplicationContextAware {
    /**
     * 存放策略的map,可以理解为策略的注册中心
     */
    private final Map<Integer, StrategyService> strategyServiceMap = new ConcurrentHashMap<>(16);
    /**
     * spring的上下文
     */
    private ApplicationContext applicationContext;

    /**
     * 将StrategyService的类都按照定义好的规则(fetchKey),放入strategyServiceMap中
     */
    @Override
    public void afterPropertiesSet() {
        //初识化把所有的策略bean放进ioc,用于使用的时候获取
        Map<String, StrategyService> matchBeans = applicationContext.getBeansOfType(StrategyService.class);
        //策略注入的bean做key,策略实现类做value
        matchBeans.forEach((key, value) ->{
            strategyServiceMap.put(value.fetchKey(), value);
            log.info("初始化策略模式的键值对 key={},value={}",key,value);
        });
    }

    /**
     * 注入applicationContext
     *
     * @param applicationContext ac
     * @throws BeansException e
     */
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }

    /**
     * 通过key获取对应的策略实现
     *
     * @param key key(String类型或者整形都行,保持和策略接口一致就行)
     * @return strategyService
     */
    public StrategyService getStrategy(Integer key) {
        if (null==strategyServiceMap.get(key)) {
            //默认策略
            return strategyServiceMap.get(0);
        }
        return strategyServiceMap.get(key);
    }
}

其实到这里已经完了,结构大致如图

java策略模式实战示例「建议收藏」

如果觉得文章有点乱还是建议看完整代码吧,确实要全部展现出来篇幅太大哈

小结和抛出一些观点:有个缺点就是类膨胀,就是策略类太多的情况下,这个类就太多了,当然有方式处理,但是结合实际,最终还是妥协选择类膨胀,因为这个也不算什么大缺点,可以忽略.其实java的JDK8的函数式编程和Lambda表达式(简化匿名类等写法)可以让策略模式更加优雅,其实就是相当于JDK8新特性是把23中设计模式更加抽象的方式用在新语法上了,符合时代潮流,拓展java的函数式编程领域,可以大概参考哈新特性  https://zhangxiaoxiang.blog.csdn.net/article/details/100638661

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/162247.html原文链接:https://javaforall.net

(0)
上一篇 2022年7月20日 上午9:16
下一篇 2022年7月20日 上午9:36


相关推荐

  • F分布的概率密度函数_F分布的统计量是

    F分布的概率密度函数_F分布的统计量是定义:设X1∼χ2(m),X2∼χ2(n)X_{1}\sim\chi^{2}(m),X_{2}\sim\chi^{2}(n)X1​∼χ2(m),X2​∼χ2(n),X1X_{1}X1​与X2X_{2}X2​相互独立,则称随机变量F=X1/mX2/nF=\frac{X_{1}/m}{X_{2}/n}F=X2​/nX1​/m​服从自由度为mmm及nnn的FFF分布,mmm称为第一自由度,n\boldsymbol{n}n称为第二自由

    2022年10月10日
    5
  • Golang中Int32转换为int16丢失精度的具体过程

    Golang中Int32转换为int16丢失精度的具体过程为什么80%的码农都做不了架构师?>>>…

    2022年8月15日
    4
  • 文明距离(civil)

    文明距离(civil)文明距离 civil 题目描述你被一个一向箔打中了 现在你掉到了一个一维空间中 也就是一个数轴上 在这个数轴上 每秒会在一段连续的区间上出现 文明 而你在每一秒开始的时候 可以花费 x 的代价移动 x 的距离 其中 x 是任意非负实数 当你移动结束以后 若你离 文明 的距离为 y 你就需要花费 y 的代价使用 大眼睛 来观测这个文明 不然你就要被黑暗森林攻击了 此处距离是指你到这段区间

    2026年3月16日
    2
  • linux 同步IO: sync、fsync与fdatasync

    linux 同步IO: sync、fsync与fdatasync传统的UNIX实现在内核中设有缓冲区高速缓存或页面高速缓存,大多数磁盘I/O都通过缓冲进行。当将数据写入文件时,内核通常先将该数据复制到其中一个缓冲区中,如果该缓冲区尚未写满,则并不将其排入输出队列,而是等待其写满或者当内核需要重用该缓冲区以便存放其他磁盘块数据时,再将该缓冲排入输出队列,然后待其到达队首时,才进行实际的I/O操作。这种输出方式被称为延迟写(delayedwrite)(Bach

    2022年5月31日
    35
  • 服务器的公网ip 和内网ip

    服务器的公网ip 和内网ip原文地址:https://zhidao.baidu.com/question/814783729071869532.html服务器公网ip  可以用于域名解析ip,服务器远程登录ip,是最主要的服务器ip地址。内网ip  不能用于域名解析。  不可以直接用于服务器远程登录,其主要作用是:跟当前帐号下的其他同集群的机器通信。  一些小型企业或者学校,通常都是申请一个固定的IP地址,…

    2022年5月6日
    38
  • 在虚拟机中安装windows server2008_在虚拟机上安装win7

    在虚拟机中安装windows server2008_在虚拟机上安装win7虚拟机里的windowsxp的安装

    2022年8月16日
    6

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号