Java策略模式以及来自lambda的优化
前言
设计模式是软件工程中一些问题的统一解决方案的模型,它的出现是为了解决一些普遍存在的,却不能被语言特性直接解决的问题,随着软件工程的发展,设计模式也会不断的进行更新,本文介绍的是经典设计模式-策略模式以及来自java8的lambda的对它的优化。
什么是策略模式
定义
策略模式定义了算法家族,分别封装起来,让他们之间可以互相替换,此模式让算法的变化,不会影响到使用算法的客户。 (摘自<大话设计模式>)
理解
我的理解还是很简单,策略模式定义的是封装算法,其实算法就是行为的一种,所以我的理解是只要是某个地方需要根据不同的情况执行不同的策略,就可以尝试使用策略模式,无需管内部究竟是算法还是什么业务,这样做的好处就是在于可以消除掉条件判断语句,将行为独立出来进行测试修改等等,下面举一个例子再来说明它的优势与优化点。
例子
传统实现
这里举一个加法,减法,乘法的例子,使用策略模式进行封装
策略算法公共接口,定义了计算的方法
public interface Strategy {
/**
* 计算接口
* @param x
* @param y
* @return
*/
int calclate(int x, int y);
}
子类,加法,减法,乘法,作为策略接口的不同实现
加法
public class Add implements Strategy {
@Override
public int calclate(int x, int y) {
return x + y;
}
}
减法
public class Sub implements Strategy {
@Override
public int calclate(int x, int y) {
return x - y;
}
}
乘法
public class Mul implements Strategy {
@Override
public int calclate(int x, int y) {
return x * y;
}
}
上下文环境类,用于客户端与这些算法接口交互,通过getResult()方法对外交互
public class Context {
private Strategy strategy;
public Context(Strategy strategy) {
this.strategy = strategy;
}
public int getResult(int x, int y) {
return strategy.calclate(x, y);
}
}
客户端,根据不同的需求实例化不同的Context类,统一使用getResuslt方法获得结果
public class Client {
public static void main(String[] args) {
Context context;
int x = 10;
int y = 5;
String strategy = "add";
switch (strategy) {
case "add":
context = new Context(new Add());
break;
case "sub":
context = new Context(new Sub());
break;
default:
context = new Context(new Mul());
break;
}
System.out.println(context.getResult(x, y));
}
}
输出结果
15
策略模式将这些加减乘除不同的算法使用策略子类进行包裹,客户端根据需求实例化Context类,然后通过Context类的getResult方法实现接口的调用,注意这里与简单工厂模式的区别,简单工厂模式通过Factory实例化出不同的类,然后调用实例类的getResult()的方法获得结果,而策略模式是通过调用Context
的getResult方法获得结果,所以简单工厂模式的客户端需要与工厂类以及实例化类相关联,策略模式的客户端只与Context类相关联,可以说耦合度进一步降低了。
结合简单工厂模式进行优化
可以看出上文的一部分逻辑判断语句选择放在了客户端中,这是不太好的,而策略模式客户端相关联的类只有Context,所以这一部分逻辑完全可以仿照简单工厂模式转移到Context类中。
Context类,将switch判断转移其中
public class Context {
private Strategy strategy;
public Context(String strategy) {
switch (strategy) {
case "add":
this.strategy = new Add();
break;
case "sub":
this.strategy = new Sub();
break;
default:
this.strategy = new Mul();
break;
}
}
public int getResult(int x, int y) {
return strategy.calclate(x, y);
}
}
客户端类,完全只负责交互,这里只与Context耦合
public class Client {
public static void main(String[] args) {
Context context;
int x = 10;
int y = 5;
//add
context = new Context("add");
System.out.println(context.getResult(x, y));
//sub
context = new Context("sub");
System.out.println(context.getResult(x, y));
//mul
context = new Context("mul");
System.out.println(context.getResult(x, y));
}
}
总结与思考
总结
为了方便理解与代码精简,这里只做了一个简单的加减乘的算法封装,目的是为了让大家理解策略模式的封装算法行为,通过Context类与客户端进行交互,与简单工厂的不同的是,耦合度更低,并且可以通过结合简单工厂模式的特点之一,将判断逻辑转移到Context类中,下面是uml图
这个模式涉及到三个角色:
- 上下文环境(Context)角色:持有一个Strategy的引用,负责与客户端进行交互
- 抽象策略(Strategy)角色:这是一个抽象角色,通常由一个接口或抽象类实现。此角色给出所有的具体策略类所需的接口。在上文中即是Strategy类。
- 具体策略(ConcreteStrategy)角色:包装了相关的算法或行为,是策略类的子类,对策略接口的不同实现,在上文的例子中是add,sub,mul类
优点
- 解耦,将算法的公共部分抽象出来统一调用,客户端不与算法交互,通过Context来交互
- 算法子类的添加,修改与复用变得十分容易
- 由于每个算法类都单独抽离出来,方便单元测试
- 通过这些算法子类可以消除环境中的判断语句
可优化点
- 同命令模式一样,如果有100个不同的算法就要生成大量的子类,类的数量过多
- 部分逻辑语句只是转移到了Context种,并没有完全消除,实例化依然避免不了进行判断
思考
和上一篇的优化命令模式一样,既然是对算法也就是行为的封装,那么在策略模式中使用lambda封装算法接口一定也是可行的。
使用lambda进行优化
经过上一章命令模式的优化经验,策略模式优化起来可谓得心应手,这里考虑到我们需要封装的行为是接受两个相同类型的参数,做加减乘的运算,返回同类型的结果的算法,这里选用的函数接口是BinaryOperator<T>
,该接口接受两个T类型的参数,返回一个T类型的结果,当然也可以选用更加常见BiFuntion<T,K,R>
,该接口接受T,K类型的两个参数,返回R类型的结果,三种类型(T,K,R)可以一致。
在这里只需将原来用于封装算法的算法接口(Strategy)以及他们的子类直接使用BinaryOperator进行替即可,类的数量减少为两个,代码如下。
使用BinaryOperator接口优化后的Context类
public class Context {
private BinaryOperator<Integer> strategy;
public Context(BinaryOperator<Integer> strategy) {
this.strategy = strategy;
}
public int getResult(int x, int y) {
return strategy.apply(x, y);
}
}
客户端代码
public class Client {
public static void main(String[] args) {
Context context;
int x = 10;
int y = 5;
//add
context = new Context((integer, integer2) -> integer + integer2);
System.out.println(context.getResult(x, y));
//sub
context = new Context((integer, integer2) -> integer - integer2);
System.out.println(context.getResult(x, y));
//mul
context = new Context((integer, integer2) -> integer * integer2);
System.out.println(context.getResult(x, y));
}
}
- 在维持了策略模式的优点的情况下,类的数量由6个减少到了2个,可谓是非常的精简了
- 然而通过观察发现,客户端这里似乎还需要手动的编写lambda代码算法,感觉这样的写法达不到失去了原先算法复用的效果,当然可以像上文在context中定义swich判断,然后进行不同的lambda表达式的传入,可有没有更好的方法呢?当然有,比较敏感的同学在观看前面代码的时候应该就有感觉,传统的实现传入的是常量,而常量在实际中使用枚举效果往往更好,因此在这里,我们也使用枚举类结合lambda表达,再一步优化
使用枚举类结合lambda进一步的优化
封装lambda算法的StrategyEnum枚举,对外提供get方法获得内部的封装的算法
public enum StrategyEnum {
ADD((x, y) -> x + y),
SUB((x, y) -> x - y),
MUL((x, y) -> x * y);
private BinaryOperator<Integer> operator;
StrategyEnum(BinaryOperator<Integer> operator) {
this.operator = operator;
}
public BinaryOperator<Integer> get() {
return operator;
}
}
Context类内拥有Enum成员变量,通过调用get方法获得函数对象再调用apply对参数进行操作
public class Context {
private StrategyEnum strategy;
public Context(StrategyEnum strategy) {
this.strategy = strategy;
}
public int getResult(int x, int y) {
return strategy.get().apply(x, y);
}
}
客户端代码,为了方便显示静态导入枚举类
public class Client {
public static void main(String[] args) {
int x = 10;
int y = 5;
//add
System.out.println(new Context(ADD).getResult(x, y));
//sub
System.out.println(new Context(SUB).getResult(x, y));
//mul
System.out.println(new Context(MUL).getResult(x, y));
}
}
运行结果
15
5
50
Process finished with exit code 0
运行良好,可以看到使用枚举类封装lambda算法对外提供算法的效果是惊人的,虽然和之前的lambda直接优化相比多了一个类,但是无论从代码的精简程度还是风格以及可读性上,都强上了一个档次,最后版本的策略模式的代码量几乎已经不能再少了,十分简洁,同时保留了策略模式的所有特性,包括算法的复用,测试,与客户端的解耦,同时对上文提到的几个可优化点都完成了优化。
结尾
上文的例子为了方便理解举的例子十分简单,但是包含的思想确是一致的,在实际运用中,你所需要的封装内容不一定是算法,可以是业务逻辑,可以是一段处理过程,只要观察代码中是否出现的大量的swich或者if用来判断 结构相同但是内容不同的一段代码时,就该考虑使用上面讲述的 策略模式+lambda+枚举的方式,相信代码的质量可以提高不少。
关于本文代码
本文的代码与md文章同步更新在github中的strategy-mode模块下,欢迎fork _
上一篇:java命令模式以及来自lambda的优化
下一篇:java简单工厂模式以及来自lambda的优化
Java策略模式以及来自lambda的优化的更多相关文章
- Java命令模式以及来自lambda的优化
前言 设计模式是软件工程中一些问题的统一解决方案的模型,它的出现是为了解决一些普遍存在的,却不能被语言特性直接解决的问题,随着软件工程的发展,设计模式也会不断的进行更新,本文介绍的是经典设计模式 ...
- Java简单工厂模式以及来自lambda的优化
前言 设计模式是软件工程中一些问题的统一解决方案的模型,它的出现是为了解决一些普遍存在的,却不能被语言特性直接解决的问题,随着软件工程的发展,设计模式也会不断的进行更新,本文介绍的是经典设计模式 ...
- 新来的"大神"用策略模式把if else给"优化"了,技术总监说:能不能想好了再改?
本文来自作者投稿,原作者:上帝爱吃苹果 目前在魔都,贝壳找房是我的雇主,平时关注一些 java 领域相关的技术,希望你们能在这篇文章中找到些有用的东西.个人水平有限,如果文章有错误还请指出,在留言区一 ...
- 【原创】从策略模式闲扯到lambda表达式
引言 策略模式,讲这个模式的文章很多,但都缺乏一个循序渐进的过程.讲lambda表达式的文章也很多,但基本都是堆砌一堆的概念,很少带有自己的见解.博主一时兴起,想写一篇这二者的文章.需要说明的是,在看 ...
- JAVA策略模式
<JAVA与模式>之策略模式 在阎宏博士的<JAVA与模式>一书中开头是这样描述策略(Strategy)模式的: 策略模式属于对象的行为模式.其用意是针对一组算法,将每一个算法 ...
- Java策略模式(Strategy模式) 之体验
<JAVA与模式>之策略模式 在阎宏博士的<JAVA与模式>一书中开头是这样描述策略(Strategy)模式的: 策略模式属于对象的行为模式.其用意是针对一组算法,将每一个算法 ...
- java策略模式(及与工厂模式的区别)
按一般教程中出现的例子理解: 简单工厂模式:客户端传一个条件进工厂类,工厂类根据条件创建相应的产品类对象,并return给客户端,供客户端使用.即客户端使用的是工厂类生产的产品对象. 策略模式:客户端 ...
- Java策略模式(Strategy)
一.定义 定义一组算法,将每个算法都封装起来,并且使它们之间可以互换.策略模式使这些算法在客户端调用它们的时候能够互不影响地变化.(Java的TreeSet集合中,构造方法可传入具体的比较器对象以实现 ...
- java策略模式拙见
面向对象的两个基本准则: 单一职责:一个类只有一个发生变化的原因 开闭原则:对拓展开放,对修改关闭 <Java开发手册>中,有这样的规则:超过3层的 if-else 的逻辑判断代码可以使用 ...
随机推荐
- java1.8新特性
转自:http://www.oschina.NET/translate/everything-about-Java-8 建议去看原文,此处转载只是为了记录. 这篇文章是对Java8中即将到来的改进做一 ...
- centos6.6配置rsync+sersync实现实时同步分布式多客户端分发同步
1.sersync项目: sersync项目利用inotify与rsync技术实现对服务器数据实时同步到解决方案,其中inotify用于监控sersync所在服务器上文件系统的事件变化,rsync是目 ...
- Java零碎知识点
█ 举个例子:Iterator iter = map.entrySet().iterator(); xx.yy() ,表示一个xx对象的yy方法 ,xx.yy().zz()中 xx.yy()返回 ...
- 201521123044 《Java程序设计》第6周学习总结
1. 本章学习总结 1.1 面向对象学习暂告一段落,请使用思维导图,以封装.继承.多态为核心概念画一张思维导图,对面向对象思想进行一个总结. 注1:关键词与内容不求多,但概念之间的联系要清晰,内容覆盖 ...
- 201521123004 《Java程序设计》第1周学习总结
1. 本章学习总结 (1)安装各种软件(jdk,eclipse,git(安装不了)) 注册账号(博客,网易邮箱(QQ邮箱不能用)码云) 创建项目(码云,Java) (2)了解JAVA语言的发展史(su ...
- 框架应用:Spring framework (五) - Spring MVC技术
软件开发中的MVC设计模式 软件开发的目标是减小耦合,让模块之前关系清晰. MVC模式在软件开发中经常和ORM模式一起应用,主要作用是将(数据抽象,数据实体传输和前台数据展示)分层,这样前台,后台,数 ...
- JVM(三)内存回收(一)
最近花了相当长一段时间在看Hotspot JVM的GC和内存分配,本节先总结和回顾一下内存回收的相关知识点,内存的分配放到下节再讨论. 一.什么是JVM的GC GC即Garbage Collectio ...
- 初试 Vue.js
1.为什么我会想要来弄弄vue这个前端框架呢? 答:前段时间被小程序刷屏了,然后就去弄了一下小程序,嗯挺简单的:头脑一发热后就想到vue2也发布一段时间了,何不也来尝尝vue2.0的味道,最后发现它们 ...
- Lucene第二篇【抽取工具类、索引库优化、分词器、高亮、摘要、排序、多条件搜索】
对Lucene代码优化 我们再次看回我们上一篇快速入门写过的代码,我来截取一些有代表性的: 以下代码在把数据填充到索引库,和从索引库查询数据的时候,都出现了.是重复代码! Directory dire ...
- 教你在Java接口中定义方法
基本上所有的Java教程都会告诉我们Java接口的方法都是public.abstract类型的,没有方法体的. 但是在JDK8里面,你是可以突破这个界限的哦. 假设我们现在有一个接口:TimeClie ...