- Java锁的逻辑(结合对象头和ObjectMonitor)
- 还在用饼状图?来瞧瞧这些炫酷的百分比可视化新图形(附代码实现)⛵
- 自动注册实体类到EntityFrameworkCore上下文,并适配ABP及ABPVNext
- 基于Sklearn机器学习代码实战
在学开发的第二年就开始听说要想代码写得好,一定要会设计模式。于是就兴致冲冲的啃了《Head First 设计模式》,看完之后对于策略模式映像很深刻,觉得这个模式好,易上手,应用广,我又能优化一波代码了(装波逼了),于是兴致冲冲的打开了我的 IDEA,开整!!! 。
在讲诉我的策略模式首秀前,我们先回顾下策略模式的基本概念.
策略模式 。
简单的来说当做某个事情有多个方式的时候,可以抽象为接口,然后每个实现是一种解决方式,由调用方来选择不同的实现方式.
理解了后我开始对我们的代码进行了重构,当时我第一家公司有这样一段代码,大概是这个意思(时间长了,我凭记忆重写的).
有这样一个抽奖的方法,我们后台控制中奖率,不同的时候我们会调整不同的中奖策略.
public class NumStrategy {
enum RandomEnum{
/**
* 平均策略
*/
AVERAGE,
/**
* 80%的几率中奖
*/
RANDOM28;
}
/**
* 抽奖方法,根据不同的策略进行抽奖
* @param randomEnum
* @return true:代表中奖 false:代表没中奖
*/
public boolean luckDraw(RandomEnum randomEnum){
if(randomEnum.equals(RandomEnum.AVERAGE)){
Random random = new Random();
int num = random.nextInt(100);
return num >= 50;
}else if(randomEnum.equals(RandomEnum.RANDOM28)){
Random random = new Random();
int num = random.nextInt(100);
return num >= 20;
}
return false;
}
}
我一看,这不就是妥妥的策略模式吗?开搞.
一顿改造之后变成了这样:
public class NumStrategy2 {
enum RandomEnum{
/**
* 平均策略
*/
AVERAGE,
/**
* 80%的几率中奖
*/
RANDOM28;
}
/**
* 抽奖方法,根据不同的策略进行抽奖
* @param randomEnum
* @return true:代表中奖 false:代表没中奖
*/
public boolean luckDraw(RandomEnum randomEnum){
if(randomEnum.equals(RandomEnum.AVERAGE)){
return new AverageStrategy().luckDraw();
}else if(randomEnum.equals(RandomEnum.RANDOM28)){
return new Random28Strategy().luckDraw();
}
return false;
}
interface LuckDrawStrategy{
boolean luckDraw();
}
class AverageStrategy implements LuckDrawStrategy{
@Override
public boolean luckDraw() {
Random random = new Random();
int num = random.nextInt(100);
return num >= 50;
}
}
class Random28Strategy implements LuckDrawStrategy{
@Override
public boolean luckDraw() {
Random random = new Random();
int num = random.nextInt(100);
return num >= 20;
}
}
}
改造完成之后我满意的提交了代码,但是在组长 review 的时候给我又改了回来。说你整这么多类干嘛?我理直气壮的说我这是用策略模式优化代码。他说没必要,先改回去吧.
我愤愤的接受了,但心里想着:哎,你连策略模式都不懂?
经过这么多年,我开始理解我当时的做法其实不对,本来很简单的代码,而且里面的逻辑不会有变动,其实不需要抽象出来。我的改动有过度设计之嫌。把原来的 30 行代码搞成了 80 行 。
一报还一报,这几年我见过太多次当年的我这样写代码的了.
即 为了用设计模式而用设计模式。而忘了设计模式的初衷是为了代码更易理解,更可靠,更易维护 .
甚至还见过有人学了策略模式后说要把项目里所有的 if/else 都安排上策略模式.
又过了一年多,在一次面试的时候,也有着关于策略模式的讨论.
【面试官】问:你说你用过策略模式,请问你为什么用它?
【我】:为了抽离各个不同实现逻辑,优化 if/else,使代码更简单易懂 。
【面试官】:你具体说说,怎么去掉的 if/else 。
【我】:内心 OS(背的知识点,我也好久没用了啊)。我硬着头皮说,我可以使用工厂模式+策略模式来做.
【面试官】:那你工厂模式的那里不是也要用 if/else 判断吗?
【我】:。。。额。唔。。。那确实还是要用到 if/else 。
把我问住了,我支支吾吾的回答确实还是要 if/else 来判断一次,只不过把判断移到了工厂模式里面去了.
我下来后又去实践了下,想着放在 map 里行不行呢?
public class NumStrategy3 {
enum RandomEnum{
/**
* 平均策略
*/
AVERAGE,
/**
* 80%的几率中奖
*/
RANDOM28;
}
static Map<RandomEnum,LuckDrawStrategy> map = new HashMap<>();
static{
map.put(RandomEnum.RANDOM28,new Random28Strategy());
map.put(RandomEnum.AVERAGE,new AverageStrategy());
}
/**
* 抽奖方法,根据不同的策略进行抽奖
* @param randomEnum
* @return true:代表中奖 false:代表没中奖
*/
public boolean luckDraw(RandomEnum randomEnum){
LuckDrawStrategy luckDrawStrategy = map.get(randomEnum);
return luckDrawStrategy.luckDraw();
}
interface LuckDrawStrategy{
boolean luckDraw();
}
static class AverageStrategy implements LuckDrawStrategy{
@Override
public boolean luckDraw() {
Random random = new Random();
int num = random.nextInt(100);
return num >= 50;
}
}
static class Random28Strategy implements LuckDrawStrategy{
@Override
public boolean luckDraw() {
Random random = new Random();
int num = random.nextInt(100);
return num >= 20;
}
}
}
终于是解决了 if/else 的情况,不过这样很短的 if/else,里面逻辑不怎么变动时,我个人是不建议用策略模式,这里只是示例.
又过了几年,当初的菜鸟也成长为了一个老鸟.
当时项目里有这样一个代码:
下面的代码我进行了一些简化,我们有一个功能,对页面上的指标进行计算,不同的指标对应不同的计算方法。页面上指标一期做 4 个,后续会做到十几个.
public interface TransferService {
String transfer();
}
@Service
public class SearchTransformService {
@Autowired
private UserTransferService userTransferService;
@Autowired
private AgeTransferService ageTransferService;
@Autowired
private InterestTransferService interestTransferService;
/**
* 根据不同的编码进行转换
* @param code
* @return
*/
public String transform(String code){
if(code.equals("user")){
return userTransferService.transfer();
}else if(code.equals("age")){
return ageTransferService.transfer();
}else if(code.equals("interest")){
return interestTransferService.transfer();
}
return "";
}
}
可以看到这样的业务场景下,这样的写法 if/else 就会很长,后续十几个的情况下就很难维护。另外 code 使用的是魔数,也是不好的一种写法。我对此进行了优化如下:
enum CodeEnum {
USER("user"),
AGE("age"),
INTEREST("interest"),
;
private String code;
public String getCode() {
return code;
}
CodeEnum(String code) {
this.code = code;
}
private static final Map<String, CodeEnum> map = Arrays.stream(CodeEnum.values()).collect(Collectors.toMap(CodeEnum::getCode, Function.identity()));
public CodeEnum of(String code) {
return map.get(code);
}
}
public interface TransferService {
String transfer();
CodeEnum transCode();
}
@Service
public class AgeTransferService implements TransferService {
@Override
public String transfer() {
return null;
}
@Override
public CodeEnum transCode() {
return CodeEnum.AGE;
}
}
@Service
public class SearchTransformService implements InitializingBean {
@Autowired
private List<TransferService> transferServiceList;
private Map<CodeEnum, TransferService> transferServiceMap;
@Override
// 项目启动时将实现类放入到map中去
public void afterPropertiesSet() throws Exception {
transferServiceMap = transferServiceList.stream().collect(Collectors.toMap(TransferService::transCode, Function.identity()));
}
/**
* 根据不同的编码进行转换
* @param code
* @return
*/
public String transform(String code){
TransferService transferService = transferServiceMap.get(CodeEnum.of(code));
Assert.notNull(transferService,"找不到对应的转换器");
return transferService.transfer();
}
}
重构后是不是就很简洁了呢?如果后续新增新的编码转换器,只需要先在枚举里定义,然后新增实现类实现方法就行了,不需要对关心是怎么调用的,只关心具体的实现逻辑,降低了维护成本.
这才是策略模式的真正应用吧。不要再乱用了,哈哈哈.
最后此篇关于知道策略模式!但不会在项目里使用?的文章就讲到这里了,如果你想了解更多关于知道策略模式!但不会在项目里使用?的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
尝试使用集成到 QTCreator 的表单编辑器,但即使我将插件放入 QtCreator.app/Contents/MacOS/designer 也不会显示。不过,相同的 dylib 文件确实适用于独
在此代码示例中。 “this.method2();”之后会读到什么?在返回returnedValue之前会跳转到method2()吗? public int method1(int returnedV
我的项目有通过gradle配置的依赖项。我想添加以下依赖项: compile group: 'org.restlet.jse', name: 'org.restlet.ext.apispark', v
我将把我们基于 Windows 的客户管理软件移植到基于 Web 的软件。我发现 polymer 可能是一种选择。 但是,对于我们的使用,我们找不到 polymer 组件具有表格 View 、下拉菜单
我的项目文件夹 Project 中有一个文件夹,比如 ED 文件夹,当我在 Eclipse 中指定在哪里查找我写入的文件时 File file = new File("ED/text.txt"); e
这是奇怪的事情,这个有效: $('#box').css({"backgroundPosition": "0px 250px"}); 但这不起作用,它只是不改变位置: $('#box').animate
这个问题在这里已经有了答案: Why does OR 0 round numbers in Javascript? (3 个答案) 关闭 5 年前。 Mozilla JavaScript Guide
这个问题在这里已经有了答案: Is the function strcmpi in the C standard libary of ISO? (3 个答案) 关闭 8 年前。 我有一个问题,为什么
我目前使用的是共享主机方案,我不确定它使用的是哪个版本的 MySQL,但它似乎不支持 DATETIMEOFFSET 类型。 是否存在支持 DATETIMEOFFSET 的 MySQL 版本?或者有计划
研究 Seam 3,我发现 Seam Solder 允许将 @Named 注释应用于包 - 在这种情况下,该包中的所有 bean 都将自动命名,就好像它们符合条件一样@Named 他们自己。我没有看到
我知道 .append 偶尔会增加数组的容量并形成数组的新副本,但 .removeLast 会逆转这种情况并减少容量通过复制到一个新的更小的数组来改变数组? 最佳答案 否(或者至少如果是,则它是一个错
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
noexcept 函数说明符是否旨在 boost 性能,因为生成的对象中可能没有记录异常的代码,因此应尽可能将其添加到函数声明和定义中?我首先想到了可调用对象的包装器,其中 noexcept 可能会产
我正在使用 Angularjs 1.3.7,刚刚发现 Promise.all 在成功响应后不会更新 angularjs View ,而 $q.all 会。由于 Promises 包含在 native
我最近发现了这段JavaScript代码: Math.random() * 0x1000000 10.12345 10.12345 >> 0 10 > 10.12345 >>> 0 10 我使用
我正在编写一个玩具(物理)矢量库,并且遇到了 GHC 坚持认为函数应该具有 Integer 的问题。是他们的类型。我希望向量乘以向量以及标量(仅使用 * ),虽然这可以通过仅使用 Vector 来实现
PHP 的 mail() 函数发送邮件正常,但 Swiftmailer 的 Swift_MailTransport 不起作用! 这有效: mail('user@example.com', 'test
我尝试通过 php 脚本转储我的数据,但没有命令行。所以我用 this script 创建了我的 .sql 文件然后我尝试使用我的脚本: $link = mysql_connect($host, $u
使用 python 2.6.4 中的 sqlite3 标准库,以下查询在 sqlite3 命令行上运行良好: select segmentid, node_t, start, number,title
我最近发现了这段JavaScript代码: Math.random() * 0x1000000 10.12345 10.12345 >> 0 10 > 10.12345 >>> 0 10 我使用
我是一名优秀的程序员,十分优秀!