技术标签: java策略模式简介 B-java基础+jvm+并发+设计模式
现已放在gitee上,可以不下载直接参考一下即: https://gitee.com/zhang-xiao-xiang/zxx-pattern
由于太多的策略模式时根据顾客VIP等级不同,得到的商品价格不一样的例子,这里还是换个汤,顺便更新了一下博客,以星座自我介绍(根据星座的类型不同,返回不同的信息)为例子,结合springboot实际感受一下在实战时的策略模式样子
未使用时:我们经常直接在业务层开始了if else的常规操作
使用策略模后时:注意看一下描述
简单做个结论:这里不是说代码量减少了哈,而是说需要新增或者修改的时候,维护难度就不一样了
经过对比分析,发现之所以出现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的JDK8的函数式编程和Lambda表达式(简化匿名类等写法)可以让策略模式更加优雅,其实就是相当于JDK8新特性是把23中设计模式更加抽象的方式用在新语法上了,符合时代潮流,拓展java的函数式编程领域,可以大概参考哈新特性 https://zhangxiaoxiang.blog.csdn.net/article/details/100638661
文章浏览阅读645次。这个肯定是末尾的IDAT了,因为IDAT必须要满了才会开始一下个IDAT,这个明显就是末尾的IDAT了。,对应下面的create_head()代码。,对应下面的create_tail()代码。不要考虑爆破,我已经试了一下,太多情况了。题目来源:UNCTF。_攻防世界困难模式攻略图文
文章浏览阅读2.9k次,点赞3次,收藏10次。偶尔会用到,记录、分享。1. 数据库导出1.1 切换到dmdba用户su - dmdba1.2 进入达梦数据库安装路径的bin目录,执行导库操作 导出语句:./dexp cwy_init/[email protected]:5236 file=cwy_init.dmp log=cwy_init_exp.log 注释: cwy_init/init_123..._达梦数据库导入导出
文章浏览阅读1.9k次。1. 在官网上下载KindEditor文件,可以删掉不需要要到的jsp,asp,asp.net和php文件夹。接着把文件夹放到项目文件目录下。2. 修改html文件,在页面引入js文件:<script type="text/javascript" src="./kindeditor/kindeditor-all.js"></script><script type="text/javascript" src="./kindeditor/lang/zh-CN.js"_kindeditor.js
文章浏览阅读2.3k次,点赞6次,收藏14次。SPI的详情简介不必赘述。假设我们通过SPI发送0xAA,我们的数据线就会变为10101010,通过修改不同的内容,即可修改SPI中0和1的持续时间。比如0xF0即为前半周期为高电平,后半周期为低电平的状态。在SPI的通信模式中,CPHA配置会影响该实验,下图展示了不同采样位置的SPI时序图[1]。CPOL = 0,CPHA = 1:CLK空闲状态 = 低电平,数据在下降沿采样,并在上升沿移出CPOL = 0,CPHA = 0:CLK空闲状态 = 低电平,数据在上升沿采样,并在下降沿移出。_stm32g431cbu6
文章浏览阅读1.2k次,点赞2次,收藏8次。数据链路层习题自测问题1.数据链路(即逻辑链路)与链路(即物理链路)有何区别?“电路接通了”与”数据链路接通了”的区别何在?2.数据链路层中的链路控制包括哪些功能?试讨论数据链路层做成可靠的链路层有哪些优点和缺点。3.网络适配器的作用是什么?网络适配器工作在哪一层?4.数据链路层的三个基本问题(帧定界、透明传输和差错检测)为什么都必须加以解决?5.如果在数据链路层不进行帧定界,会发生什么问题?6.PPP协议的主要特点是什么?为什么PPP不使用帧的编号?PPP适用于什么情况?为什么PPP协议不_接收方收到链路层数据后,使用crc检验后,余数为0,说明链路层的传输时可靠传输
文章浏览阅读587次。软件测试工程师移民加拿大 无证移民,未受过软件工程师的教育(第1部分) (Undocumented Immigrant With No Education to Software Engineer(Part 1))Before I start, I want you to please bear with me on the way I write, I have very little gen...
文章浏览阅读304次。Thinkpad X250笔记本电脑,装的是FreeBSD,进入BIOS修改虚拟化配置(其后可能是误设置了安全开机),保存退出后系统无法启动,显示:secure boot failed ,把自己惊出一身冷汗,因为这台笔记本刚好还没开始做备份.....根据错误提示,到bios里面去找相关配置,在Security里面找到了Secure Boot选项,发现果然被设置为Enabled,将其修改为Disabled ,再开机,终于正常启动了。_安装完系统提示secureboot failure
文章浏览阅读10w+次,点赞93次,收藏352次。1、用strtok函数进行字符串分割原型: char *strtok(char *str, const char *delim);功能:分解字符串为一组字符串。参数说明:str为要分解的字符串,delim为分隔符字符串。返回值:从str开头开始的一个个被分割的串。当没有被分割的串时则返回NULL。其它:strtok函数线程不安全,可以使用strtok_r替代。示例://借助strtok实现split#include <string.h>#include <stdio.h&_c++ 字符串分割
文章浏览阅读2.3k次。1 .高斯日记 大数学家高斯有个好习惯:无论如何都要记日记。他的日记有个与众不同的地方,他从不注明年月日,而是用一个整数代替,比如:4210后来人们知道,那个整数就是日期,它表示那一天是高斯出生后的第几天。这或许也是个好习惯,它时时刻刻提醒着主人:日子又过去一天,还有多少时光可以用于浪费呢?高斯出生于:1777年4月30日。在高斯发现的一个重要定理的日记_2013年第四届c a组蓝桥杯省赛真题解答
文章浏览阅读851次,点赞17次,收藏22次。摘要:本文利用供需算法对核极限学习机(KELM)进行优化,并用于分类。
文章浏览阅读1.1k次。一、系统弱密码登录1、在kali上执行命令行telnet 192.168.26.1292、Login和password都输入msfadmin3、登录成功,进入系统4、测试如下:二、MySQL弱密码登录:1、在kali上执行mysql –h 192.168.26.129 –u root2、登录成功,进入MySQL系统3、测试效果:三、PostgreSQL弱密码登录1、在Kali上执行psql -h 192.168.26.129 –U post..._metasploitable2怎么进入
文章浏览阅读257次。本文将为初学者提供Python学习的详细指南,从Python的历史、基础语法和数据类型到面向对象编程、模块和库的使用。通过本文,您将能够掌握Python编程的核心概念,为今后的编程学习和实践打下坚实基础。_python人工智能开发从入门到精通pdf