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世界中无数商业的或开源的组织.技术和产品共同构成了一个无比庞大的生态系统. 与大多数开发人员的普遍认知不 ...
随机推荐
- [BD] Hive
简介 基于HDFS的数据仓库工具 基于HDFS上的数据分析引擎 2.x 前:SQL -----> Hive ----> MapReduce 2.x 后:推荐执行引擎为 Spark 支持S ...
- 常用的HTML标记
一.格式标记 1.<br> <br>是一个单标记,用来强制换行. 2.<p> <p>是双标记.用来换分段落. 3.<center> < ...
- 使用 yum-cron 自动更新 Linux系统
使用 yum-cron 自动更新 Linux系统 Linux系统技术交流QQ群(1675603)验证问题答案:刘遄 我知道如何使用 yum 命令行 更新系统,但是我想用 cron 任务自动更新软件 ...
- Ansible_使用Ansible galaxy部署角色
一.介绍Anisble galaxy 1.介绍Ansibleu galaxy 1️⃣:Ansible Galaxy (官网:https://galaxy.ansible.com)是一个Ansible内 ...
- 基于LNMP架构搭建wordpress个人博客
搭建过程 注意防火墙和selinux的影响可以先关闭. 一.安装nginx # 1.更改nginx源安装nginx [root@web01 ~]# vi /etc/yum.repos.d/nginx. ...
- C++ Error 个人笔记(live)
1.error: invalid conversion from 'char' to 'char*' [-fpermissive] 原因: 把一个字符型的变量赋值给了一个字符型的指针 我的原因: 把 ...
- hugboy源库
=[个人整理的一些源库,均来自网络]= -[Ubuntu]- #阿里源 Ubuntu 20.04 deb http://mirrors.aliyun.com/ubuntu/ focal main re ...
- 串口配合DMA接收不定长数据(空闲中断+DMA接收)-(转载)
1.空闲中断和别的接收完成(一个字节)中断,发送完成(发送寄存器控)中断的一样是串口中断: 2.空闲中断是接收到一个数据以后,接收停顿超过一字节时间 认为桢收完,总线空闲中断是在检测到在接收数据后, ...
- STM32关于多线程运行的疑问
我有一个疑问,如果 STM32在操作系统环境下 能够一个线程串口中断接收数据 另一个线程 继续进行其他的操作
- SystemVerilog 语言部分(二)
接口interface: 既可以设计,也可以用来验证. 验证环境:interface使得连接变得简单不容易出错. interface可以定义端口,单双向信号,内控部使用initial always t ...