Java行为参数化的演进
首先感谢《java8实战》一书作者某某某。
需求场景:
为一位果农设计一款软件,可以根据果农的需求筛选出相应的水果。
例如:
- 根据颜色筛选
- 根据重量筛选
- 根据颜色和重量筛选
准备工作
定义Apple类
public class Apple
{
String color;
double weight; public Apple(String color,double weight)
{
this.color=color;
this.weight=weight;
}
}
创建苹果库存
List<Apple> inventory=new ArrayList<Apple>()
{
{
add(new Apple("红色",200));
add(new Apple("绿色",80));
add(new Apple("红色",100));
add(new Apple("绿色",210));
add(new Apple("红色",105));
add(new Apple("黄色",180));
add(new Apple("红色",202));
add(new Apple("红色",99));
}
};
阶段1:值参数化 - 为每一种行为定义一个单独的方法
筛选绿苹果
List<Apple> filterGreenApple(List<Apple> inventory)
{
List<Apple> res=new ArrayList<Apple>();
for(Apple apple: inventory)
{
if("绿色".equals(apple.color))
{
res.add(apple);
}
}
return res;
}
筛选红苹果
List<Apple> filterHeavyApple(List<Apple> inventory)
{
List<Apple> res=new ArrayList<Apple>();
for(Apple apple: inventory)
{
if("红色".equals(apple.color))
{
res.add(apple);
}
}
return res;
}
无需多言,这种方式很糟糕,如果再有其它需求,还得定义更多的这种方法,其中大部分的代码都是重复的。
阶段2:值参数化 - 将条件作为参数
根据颜色筛选苹果
List<Apple> filterAppleByColor(List<Apple> inventory,String color)
{
List<Apple> res=new ArrayList<Apple>();
for(Apple apple: inventory)
{
if(color.equals(apple.color))
{
res.add(apple);
}
}
return res;
}
根据重量筛选苹果
List<Apple> filterAppleByWeight(List<Apple> inventory,double weight)
{
List<Apple> res=new ArrayList<Apple>();
for(Apple apple: inventory)
{
if(apple.weight>=weight)
{
res.add(apple);
}
}
return res;
}
这种为每一种属性定义一个方法的写法,一定程度上解决了代码重复的问题。但是,还不够。
阶段3: 值参数化 - 为属性设置标志
根据颜色或重量筛选苹果
List<Apple> filterApple(List<Apple> inventory,String color,boolean flag)
{
List<Apple> res=new ArrayList<Apple>();
for(Apple apple: inventory)
{
if(flag && color.equals(apple.color) || (!flag && apple.weight>=150))
{
res.add(apple);
}
}
return res;
}
这种方式解决了代码重复的问题,但是,当要筛选的属性很多时,需要添加过多的标志参数,使得单个方法中的代码过于复杂,容易出错。
阶段4: 行为参数化 - 使用接口对选择标准建模
定义行为接口
public interface IApplePredicate
{
boolean test(Apple apple);
}
定义筛选绿色苹果的行为类
public class GreenApplePredicate implements IApplePredicate
{
public boolean test(Apple apple)
{
return "绿色".equals(apple.color);
}
}
定义筛选重量大于150克的苹果的行为类
public class HeavyApplePredicate implements IApplePredicate
{
public boolean test(Apple apple)
{
return apple.weight>=150;
}
}
定义筛选方法,接收行为接口的实例作为参数
List<Apple> filterApples(List<Apple> inventory,IApplePredicate predicate)
{
List<Apple> res=new ArrayList<Apple>();
for(Apple apple: inventory)
{
if(predicate.test(apple))
{
res.add(apple);
}
}
return res;
}
筛选的时候,只需要传递行为类对象即可
List<Apple> greenApples=filterApples(inventory,new GreenApplePredicate());
List<Apple> heavyApples=filterApples(inventory,new HeavyApplePredicate());
通过上面这种方式,可以重复使用同一个方法,给她不同的行为来实现不同的目的。
代码结构清晰,很好。但是,必须为每一种行为定义一个类,还是比较繁琐。
阶段5: 行为参数化 - 使用匿名类避免单独定义行为类
不必为每一种行为都定义一个行为类,然后在使用具体方法的时候,传递它的对象。可以使用匿名类实现同样的效果
List<Apple> greenApples=filterApples(inventory,new IApplePredicate()
{
public boolean test(Apple apple)
{
return "绿色".equals(apple.color);
}
});
List<Apple> heavyApples=filterApples(inventory,new IApplePredicate()
{
public boolean test(Apple apple)
{
return apple.weight>=150;
}
});
阶段6: 行为参数化 - 使用lambda表达式代替匿名类
匿名类固然不错,但是,语法很是丑陋。在java8中,可以使用lambda表达式代替匿名类
List<Apple> greenApples=filterApples(inventory,apple->"绿色".equals(apple.color));
List<Apple> heavyApples=filterApples(inventory,apple -> apple.weight>=150);
阶段7: 行为参数化 - 使用泛型扩展行为的适用范围
目前为止,只能用来过滤苹果,如果想过滤梨子,就得重新定义梨的行为接口。
这样很不方便。这时,就可以使用泛型来解决这个问题。
定义泛型行为接口
public interface IPredicate<T>
{
boolean test(T t);
}
将筛选方法改造为泛型方法
<T> List<T> filter(List<T> elements,IPredicate<T> predicate)
{
List<T> res=new ArrayList<>();
for(T t: elements)
{
if(predicate.test(t))
{
res.add(t);
}
}
return res;
}
这样,筛选方法不止适用于苹果,还适用于梨子,甚至数字类型。
至此,使用行为参数化,完美的解决了需求多变的问题。
Java行为参数化的演进的更多相关文章
- 第一章 Java的I/O演进之路
I/O基础入门 Java的I/O演进 第一章 Java的I/O演进之路 1.1 I/O基础入门 1.1.1 Linux网络I/O模型简介 根据UNIX网络编程对I/O模型的分类,UNIX提供了5中I/ ...
- (基础篇 走进javaNIO)第一章-java的i/o演进之路
Java 是由 SUN公司在 1995 年首先发布 的编程语 言和计算平 台.这基础技术 支持最新 的程序 ,包括 实用程序 .游 戏和业 务应用程序 .J ava 在世界各地 的 8.5 亿 多 ...
- Java IO编程全解(一)——Java的I/O演进之路
转载请注明出处:http://www.cnblogs.com/Joanna-Yan/p/7419117.html JDK1.4之前的早期版本,Java对I/O的支持并不完善,开发人员在开发高性能I/O ...
- 性能测试十二:jmeter进阶之java请求参数化
如项目中的ip.端口号之类的,都可以在此代码中定义 public Arguments getDefaultParameters() { // TODO Auto-generated method st ...
- JAVA BIO至NIO演进
主要阐述点: 1.同步/异步 or 阻塞/非阻塞 2.网络模型演进 3.NIO代码示例 一.同步/异步 or 阻塞/非阻塞 同步/异步:核心点在于是否等待结果返回.同步即调用者必须等到结果才返回, ...
- Java Socket Server的演进 (一)
最近在看一些网络服务器的设计, 本文就从起源的角度介绍一下现代网络服务器处理并发连接的思路, 例子就用java提供的API. 1.单线程同步阻塞式服务器及操作系统API 此种是最简单的socket服务 ...
- JAVA分布式架构的演进
系统架构演化历程-初始阶段架构 初始阶段 的小型系统 应用程序.数据库.文件等所有的资源都在一台服务器上通俗称为LAMP 特征:应用程序.数据库.文件等所有的资源都在一台服务器上. 描述:通常服务器操 ...
- eventloop & actor模式 & Java线程模型演进 & Netty线程模型 总结
eventloop的基本概念可以参考:http://www.ruanyifeng.com/blog/2013/10/event_loop.html Eventloop指的是独立于主线程的一条线程,专门 ...
- Java模块化规范之争(转载)
经过近20年的发展,Java语言已成为今日世界上最成功.使用的开发者人数最多的语言之一,Java世界中无数商业的或开源的组织.技术和产品共同构成了一个无比庞大的生态系统. 与大多数开发人员的普遍认知不 ...
随机推荐
- Python正则表达式的七个使用范例
本文由 伯乐在线 - 左手的灵魂 翻译.未经许可,禁止转载!英文出处:thegeekstuff.欢迎加入翻译组.http://blog.jobbole.com/74844/ 作为一个概念而言,正则表达 ...
- 如何设计一个高性能 Elasticsearch mapping
目录 前言 mapping mapping 能做什么 Dynamic mapping dynamic=true dynamic=runtime dynamic=false dynamic=strict ...
- [刷题] 283 Move Zeros
要求 将所有的0,移动到vector的后面比如; [1,3,0,12,5] -> [1,3,12,5,0] 实现 第一版程序,时间.空间复杂度都是O(n) 1 #include<iostr ...
- 在Vim中查看文件编码和文件编码转换
在Vim中查看文件编码和文件编码转换 风亡小窝 关注 0.2 2016.09.26 22:43* 字数 244 阅读 5663评论 0喜欢 2 在Vim中查看文件编码 :set fileencodi ...
- Linux_搭建Samba服务(匿名访问)
[RHEL8]-SMBserver:[RHEL7]-SMBclient !!!测试环境我们首关闭防火墙和selinux(SMBserver和SMBclient都需要) [root@localhost ...
- 032.Python魔术方法__new__和单态模式
一 __new__ 魔术方法 1.1 介绍 触发时机:实例化类生成对象的时候触发(触发时机在__init__之前) 功能:控制对象的创建过程 参数:至少一个cls接受当前的类,其他根据情况决定 返回值 ...
- 011.Python的列表的相关操作
一 列表的相关操作 1.1 列表的拼接 lst1 = [1,2,3] lst2 = [4,5,6] res = lst1 + lst2 print(res) 执行 [root@node10 pyth ...
- 详解 WebRTC 高音质低延时的背后 — AGC(自动增益控制)
前面我们介绍了 WebRTC 音频 3A 中的声学回声消除(AEC:Acoustic Echo Cancellation)的基本原理与优化方向,这一章我们接着聊另外一个 "A" - ...
- 重新整理 .net core 实践篇—————配置系统之军令状[七](配置文件)
前言 介绍一下配置系统中的配置文件,很多服务的配置都写在配置文件中,也是配置系统的大头. 正文 在asp .net core 提供了下面几种配置文件格式的读取方式. Microsoft.extensi ...
- GO学习-(9) Go语言基础之切片
Go语言基础之切片 本文主要介绍Go语言中切片(slice)及它的基本使用. 引子 因为数组的长度是固定的并且数组长度属于类型的一部分,所以数组有很多的局限性. 例如: func arraySum(x ...