【原创】从策略模式闲扯到lambda表达式
引言
策略模式,讲这个模式的文章很多,但都缺乏一个循序渐进的过程。讲lambda表达式的文章也很多,但基本都是堆砌一堆的概念,很少带有自己的见解。博主一时兴起,想写一篇这二者的文章。需要说明的是,在看这篇文章的时候,请忘记所有的概念。容博主一步一步的带你们入坑。
正文
假设我们有一个Hero类,其UML图如下

这个时候,我们有一个需求:
- 我们要找出type为刺客的Hero
这时,我们先封装一个要根据type类型来筛选Hero的方法
public static List<Hero> getHero(List<Hero> heroList, String type){
List<Hero> result = new ArrayList<Hero>();
for(Hero hero : heroList){
if(type.equals(hero.getType())){
result.add(hero);
}
}
return result;
}
然后呢,做如下调用
getHero(heroList, "刺客");
突然有一天,产品突然改需求,现在的需求是
- 我们要找出stature(身高)大于170的Hero
也很简单嘛,再加一个重载的getHero方法就可以嘛,重载的getHero(List heroList, int stature)方法如下
public static List<Hero> getHero(List<Hero> heroList, int stature){
List<Hero> result = new ArrayList<Hero>();
for(Hero hero : heroList){
if(hero.getStature() > stature){
result.add(hero);
}
}
return result;
}
然后呢,做如下调用
getHero(heroList, 170);
又过了几日,产品丧心病况的又改需求,现在最新的需求是
- 要找出stature(身高)大于170并且type类型为刺客的Hero
当然,你或许说了,可以再加一个getHero(List heroList, String type, int stature)方法呀。不过呢,这种改法不够优雅。方法体中存在大量相同的代码,只是判断条件稍作改变。在这种情况下使用策略模式,就能够解决这个问题。
稍微介绍一下策略模式
策略模式
意图:策略模式定义了一系列的算法,并将每一个算法封装起来,而且使他们可以相互替换,让算法独立于使用它的客户而独立变化。
主要解决:在有多种算法相似的情况下,使用 if...else 所带来的复杂和难以维护。
何时使用:一个系统有许多许多类,而区分它们的只是他们直接的行为。
如何解决:将这些算法封装成一个一个的类,任意地替换。
ps:在这里,上面的算法指的就是上面提到的判断条件。我们将判断条件封装为相应的类。
此时代码结构如下图所示

那么此时的getHero方法如下所示
public static List<Hero> getHero(List<Hero> heroList, Predicate<Hero> predicate){
List<Hero> result = new ArrayList<Hero>();
for(Hero hero : heroList){
if(predicate.test(hero)){
result.add(hero);
}
}
return result;
}
然后呢,根据需求做如下调用,想找那种类型的Hero,就传那种类型的Predicate进去。
getHero(heroList,new TMPredicate());
可是呢,机智的你又发现了一个缺陷,每次新增一个算法,要新加一个实现类。于是,机智的你提出,利用匿名内部类来做调用,不写实现类,于是调用代码变成下面这样
getHero(heroList,new Predicate<Hero>() {
@Override
public boolean test(Hero t) {
return t.getStature() > 170 && "刺客".equals(t.getType());
}
});
机智的你突然间又觉得:这么写,占用了太多的行数,看起来不够美观,于是,你决定用lambda表达式来改写,于是代码最终变成下面的样子
getHero(heroList,(t)->t.getStature() > 170 && "刺客".equals(t.getType()));
好了,到这里就结束了,是不是比我们最开始的版本简洁了不少,代码优雅了很多。lambda主要的目的就是提供一个更加简洁的代码结构,但是对于初学者,它可能反而增加阅读的难度。
当然,lambda表达式除了能简化代码代码意外,还能并行处理元素,充分利用多核CPU的性能,例如下面的代码
import java.util.Arrays;
import java.util.List;
public class Demo7 {
public static void main(String[] args) {
List<String> values = Arrays.asList("1","2","3","4");
print(values);
}
public static void print(List<String> values){
values.parallelStream().forEach(System.out :: println);//System.out表示对象,println表示方法
}
}
输出如下
3
4
1
2
总结
本文以循序渐近的方式说明了,我们为什么要用策略模式以及如何用lambda表达式改写策略模式。希望大家有所收获。
参考文献
《JAVA8实战》
【原创】从策略模式闲扯到lambda表达式的更多相关文章
- Java策略模式以及来自lambda的优化
前言 设计模式是软件工程中一些问题的统一解决方案的模型,它的出现是为了解决一些普遍存在的,却不能被语言特性直接解决的问题,随着软件工程的发展,设计模式也会不断的进行更新,本文介绍的是经典设计模式 ...
- [Java 8] (5) 使用Lambda表达式进行设计
使用Lambda表达式进行设计 在前面的几篇文章中,我们已经见识到了Lambda表达式是怎样让代码变的更加紧凑和简洁的. 这一篇文章主要会介绍Lambda表达式怎样改变程序的设计.怎样让程序变的更加轻 ...
- C#从委托、lambda表达式到linq总结
前言 本文总结学习C#必须知道的基础知识,委托.监视者模式.常用lambda表达式.linq查询,自定义扩展方法,他们之间有什么关系呢?匿名委托是如何演变成lambda表达式,lambda再如何导出l ...
- Java8 in action(1) 通过行为参数化传递代码--lambda代替策略模式
[TOC] 猪脚:以下内容参考<Java 8 in Action> 需求 果农需要筛选苹果,可能想要绿色的,也可能想要红色的,可能想要大苹果(>150g),也可能需要红的大苹果.基于 ...
- 基于 普通及Lambda方式实现策略模式
什么是策略模式 策略模式代表了解决一类算法的通用解决方案,你可以在运行时选择使用哪种方案.比如如何使用不同的条件(比如苹果的重量,或者颜色 )来筛选库存中的苹果.你可以将这一模式应用到更广泛的领域 , ...
- [Java 8] (9) Lambda表达式对递归的优化(下) - 使用备忘录模式(Memoization Pattern) .
使用备忘录模式(Memoization Pattern)提高性能 这个模式说白了,就是将需要进行大量计算的结果缓存起来,然后在下次需要的时候直接取得就好了.因此,底层只需要使用一个Map就够了. 但是 ...
- 用Java 8 Lambda表达式实现设计模式:命令模式
在这篇博客里,我将说明如何在使用 Java 8 Lambda表达式 的函数式编程方式 时实现 命令 设计模式 .命令模式的目标是将请求封装成一个对象,从对客户端的不同类型请求,例如队列或日志请求参数化 ...
- 如何优雅的将文件转换为字符串(环绕执行模式&行为参数化&函数式接口|Lambda表达式)
首先我们讲几个概念: 环绕执行模式: 简单的讲,就是对于OI,JDBC等类似资源,在用完之后需要关闭的,资源处理时常见的一个模式是打开一个资源,做一些处理,然后关闭资源,这个设置和清理阶段类似,并且会 ...
- java的线程、创建线程的 3 种方式、静态代理模式、Lambda表达式简化线程
0.介绍 线程:多个任务同时进行,看似多任务同时进行,但实际上一个时间点上我们大脑还是只在做一件事情.程序也是如此,除非多核cpu,不然一个cpu里,在一个时间点里还是只在做一件事,不过速度很快的切换 ...
随机推荐
- [译]在Linux上的提高MySQL/MariaDB安全性的12条建议
MySQL 是世界上最流行的开源数据库系统,而MariaDB(MySQL的一个分支)是世界上发展最快的开源数据库系统.安装MySQL服务器之后,它的默认配置是不安全的,保护它是一般数据库管理中的基本任 ...
- Spark性能优化【OOM】
一.异常情况 Spark on yarn模式下,当yarn为client的模式时没有OOM而cluster模式下出现OOM 二.异常分析 由于client模型没有出现OOM而cluster模式出现OO ...
- SQL 中用户定义函数的使用方法
--用户定义函数的分类: /* 1.标量函数 2.表值函数 2.1内联表值函数 返回单个SELECT语句, 它没有相关的返回变量和函数体 2.2多语句表值函数 是视图和存储过程的结合 可嵌套 */ ...
- MySQL 授予普通用户PROCESS权限
在MySQL中如何给普通用户授予查看所有用户线程/连接的权限,当然,默认情况下show processlist是可以查看当前用户的线程/连接的. mysql> grant process on ...
- C#委托(转载)
C#委托的介绍(delegate.Action.Func.predicate) from:http://www.cnblogs.com/akwwl/p/3232679.html 委托是一个类,它定义了 ...
- OpenCvSharp尝试
OpenCvSharp是封装了OpenCV的.net版本 项目地址:https://github.com/shimat/opencvsharp 简单使用: 1.NuGet安装 2.使用OpenCvSh ...
- postgre中类似oracle的sql%rowcount用法
get diagnostics cnt := row_count; 现在有两个表tab1和tab2,两个表的格式相同,tab1中有1000条数据,tab2中0条数据 创建测试功能函数 create o ...
- Linux下1号进程的前世(kernel_init)今生(init进程)----Linux进程的管理与调度(六)
前面我们了解到了0号进程是系统所有进程的先祖, 它的进程描述符init_task是内核静态创建的, 而它在进行初始化的时候, 通过kernel_thread的方式创建了两个内核线程,分别是kernel ...
- ugui SetParent在安卓上一个诡异bug
问题描述 我的环境:Unity3D 5.3.7 出问题机型:安卓模拟器.部分低配安卓机型(比如:红米2A) 以下代码是设置某个节点的父节点,在PC.Editor.大部分手机上都是正常的,但问题机型上, ...
- windows 上查看一个命令的退出码
windows 上查看一个命令的退出码可以使用下面语句 echo %errorlevel% 例如:windows 上没有ls 命令,所以使用后没有成功,查看退出码为9009 ,非0 使用dir 列出目 ...