从java7到java8,最主要的变化可以总结为

□Lambda表达式

□ 方法引用

□流和默认方法

让我们通过一个小例子感受一下

情景 1 集合对象排序 (对list中的苹果按照重量排序)

  1. Collections.sort(list,new Comparator<Apple>(){
  2.    public
    int compare(Apple a1,Apple a2){
  3.       return a1.getWeight()-a2.getWeight();
  4.    }
  5. });

在java8里面,这种实现只需要一行很简短的代码

  1. list.sort(java.util.Comparator.comparing(Apple::getWeight))

这其中涉及到lambda表达式,接口中静态方法,方法引用等java8新特性,而且一看就明白其功能,给list排序,排序依据weight

Java多线程历程 java1.0 线程和锁------à java5 线程池和并发集合----------à java7 添加分支/合并(fork/join)--à java8 添加stream支持并发处理数据

在java8中加入Streams可以把 接口默认方法和把代码块传递给方法的简介方式(方法引用、lambda表达式)

把代码传递给方法 即是一种行为参数化思想,和函数式编程思想极为相近。

情景2 对目录文件过滤,筛选所有隐藏文件

  1. File [] hiddenFiles=new File(".").listFiles(new FileFilter(){
  2.    public boolean accept(File file){
  3.       return file.isHidden();
  4.    }
  5. });

使用方法引用,直接将方法当做参数传递进去

  1. File [] hiddenFiles=new File(".").listFiles(File::isHidden)

既然对象File已经有了一个isHidden方法,我们不必再把它包含在啰嗦的匿名内部类里面过滤了。

与用对象引用(对象引用是用new创建的)类似,写下File::isHidden就创建一个方法引用,可以传递它

如可以写成这样

  1. FileFilter filter=File::isHidden;
  2. File [] isHiddenFiles2=new File(".").listFiles(filter);

从第一行代码可以看出,方法引用确实可以当做一个变量来使用

 

情景3 lambda传递条件代码 (筛选符合条件的苹果)

  1. public static List<Apple> filterGreenApple(List<Apple> list){
  2.  
  3.    List<Apple> result=new ArrayLst<>();
  4.    for(Apple apple : list){
  5.       if("green".equals(apple.getColor())){
  6.          result.add(apple); //筛选颜色为绿色的苹果
  7.       }
  8.    }
  9.    return result;
  10. }
  11. public static List<Apple> filterWeighterApple(List<Apple> list){
  12.  
  13.    List<Apple> result=new ArrayLst<>();
  14.    for(Apple apple : list){
  15.       if(apple.getWeight()>150){
  16.          result.add(apple); //筛选重量大于150g的苹果
  17.       }
  18.    }
  19.    return result;
  20. }

如果客户要筛选红苹果,筛选红且重量大于150g的苹果,…难道每次都要添加一个函数吗,而且这些函数除了条件筛选不一样,其他部分都是一样的。

首先我们可以把公用代码提出来

  1. public static List<Apple> filterApple(List<Apple> list){
  2.    List<Apple> result=new ArrayList<>();
  3.    for(Apple apple:list){
  4.       if(....){ //苹果筛选条件
  5.          result.add(apple);
  6.       }
  7.    }
  8.    return result;
  9. }
  10. 进一步,我们可以把筛选条件传递进去
    过滤函数可以改成这样
  11. public static List<Apple> filterApple(List<Apple> list,Predicate<Apple> p){
  12.    List<Apple> result=new ArrayList<>();
  13.    for(Apple apple:list){
  14.       if(p.test(apple)){
  15.          result.add(apple);
  16.       }
  17.    }
  18.    return result;
  19. }
  20. Public interface Predicate<Apple>{
  21.     Boolean test(Apple apple);
  22. }

即过滤函数有两个参数,一个时过滤的对象集合,一个时过滤的 条件对象(Predicate),而该条件对象应该有个可以返回Boolean的函数来确定是否符合条件(策略设计模式)

这样设计之后,把filterApple方法迭代集合的逻辑和要应用到集合中每个元素对象上的行为(谓词)区分开了。

比如要筛选出绿苹果

  1. public class GreenColorPredicate<Apple> implements Predicate<Apple>{
  2.    public boolean test(Apple apple){
  3.       if("green".equals(apple.getColor())){
  4.          return true;
  5.       }
  6.       return false;
  7.    }
  8. }
  9. GreenColorPredicate<Apple> greenPredicate=new GreenColorPredicate<>();
  10. filterApple(list,greenPredicate);

 

或者lambda表达式

  1. filterApple(list,(Apple apple)->"green".equals(apple.getColor()));

我们可以将上面的函数抽象化成泛型函数,这样就可以过滤香蕉,橘子等对象集合啦!!

行为参数化最大好处就是可以适应不断变化的需求。

 

总结:

1 行为参数化,就是一个方法接受多个不同的行为作为参数,在内部使用它们,完成不同行为的能力。

2 行为参数化可以让代码更好适应不同需求,减轻未来工作量。

3 传递代码,即将新行为作为参数传递给方法。可以用lambda表达式或者方法引用。

4 java api包含很多可以用不同行为进行参数化的方法,最常用就是排序,线程等。

 

 

Java8实战系列一的更多相关文章

  1. GitHub实战系列汇总篇

    基础: 1.GitHub实战系列~1.环境部署+创建第一个文件 2015-12-9 http://www.cnblogs.com/dunitian/p/5034624.html 2.GitHub实战系 ...

  2. GitHub实战系列~1.环境部署+创建第一个文件 2015-12-9

    GitHub实战系列汇总:http://www.cnblogs.com/dunitian/p/5038719.html ———————————————————————————————————————— ...

  3. GitHub实战系列~2.把本地项目提交到github中 2015-12-10

    GitHub实战系列汇总:http://www.cnblogs.com/dunitian/p/5038719.html ———————————————————————————————————————— ...

  4. GitHub实战系列~3.提交github的时候过滤某些文件 2015-12-10

    GitHub实战系列汇总:http://www.cnblogs.com/dunitian/p/5038719.html ———————————————————————————————————————— ...

  5. GitHub实战系列~4.把github里面的库克隆到指定目录+日常使用 2015-12-11

    GitHub实战系列汇总:http://www.cnblogs.com/dunitian/p/5038719.html ———————————————————————————————————————— ...

  6. 05.GitHub实战系列~5.发布版本之分支操作+Tag讲解 2015-12-14

    GitHub实战系列汇总:http://www.cnblogs.com/dunitian/p/5038719.html ———————————————————————————————————————— ...

  7. 07.GitHub实战系列~7.Git之VS2013团队开发(如果不想了解git命令直接学这篇即可)

    GitHub实战系列汇总:http://www.cnblogs.com/dunitian/p/5038719.html ———————————————————————————————————————— ...

  8. Spark入门实战系列--10.分布式内存文件系统Tachyon介绍及安装部署

    [注]该系列文章以及使用到安装包/测试数据 可以在<倾情大奉送--Spark入门实战系列>获取 .Tachyon介绍 1.1 Tachyon简介 随着实时计算的需求日益增多,分布式内存计算 ...

  9. [.NET领域驱动设计实战系列]专题十一:.NET 领域驱动设计实战系列总结

    一.引用 其实在去年本人已经看过很多关于领域驱动设计的书籍了,包括Microsoft .NET企业级应用框架设计.领域驱动设计C# 2008实现.领域驱动设计:软件核心复杂性应对之道.实现领域驱动设计 ...

随机推荐

  1. 20190312 Windows上Kafka集群

    1. 复制config/server.properties为server1.properties和server2.properties 2. 以server1.properties为例,修改配置 br ...

  2. 2、Python-流程控制

    比较运算符 运算符 描述 示例 == 检查两个操作数的值是否相等,如果是则条件变为真. 如a=3,b=3则(a == b) 为 true. != 检查两个操作数的值是否相等,如果值不相等,则条件变为真 ...

  3. 设计模式---组件协作模式之模板方法模式(Tempalte Method)

    前提:组件协作模式 现代软件专业分工之后的第一个结构是“框架与应用程序的划分”,“组件协作”模式通过晚期绑定,来实现框架与应用程序之间的松耦合,是二者之间协作时常见的模式. 我们常常使用框架来写自己的 ...

  4. js中闭包的概念和用法

    闭包:主要的作用是 封装变量,收敛权限.防止变量被污染.比如Jquery框架就运用了大量的闭包.为什么呢? 问个问题?框架是如何来避免你声明的变量和它自带的变量不发生的冲突的?????很明显,需要闭包 ...

  5. RocketMQ在windows环境下的安装

    原博地址:https://www.jianshu.com/p/4a275e779afa 一.预备环境 1.系统 Windows 2. 环境 JDK1.8.Maven.Git 二. RocketMQ部署 ...

  6. JAVA记录-WebService开发部署

    JWS.Axis2.cxf 1.下载axis2.war和axis2.bin.zip 2.将axis2.war包部署到Tomcat下,启动Tomcat测试:http://localhost:8089/a ...

  7. Docker部署Consul集群

    服务介绍 Consul是一种分布式.高可用.支持水平扩展的服务注册与发现工具.包含的特性有:服务发现.健康检查.键值存储.多数据中心和服务管理页面等. 官方架构设计图: 图中包含两个Consul数据中 ...

  8. 宕机不等于关机,阴魂不散的vm

    今天早上刚到公司,就发现研发环境的机器连不上了. 公司研发环境的部署比较简单,物理机上装VMware Esxi 6 ,然后在esxi上装虚机. 检查发现:esxi ping不通,客户端也连不上:物理机 ...

  9. 由-webkit-transform-style:preserve-3d;所想

    看一个用css3写幻灯片的demo用到了这么几个属性 .demo{ -webkit-transform-style:preserve-3d; -webkit-perspective:800px; -w ...

  10. FASTREPORT COM/ActiveX报表如何保存到C++项目中?

    可以的. VC++ : ... IStream * pStream;CreateStreamOnHGlobal(NULL, true, &pStream);pStream->AddRef ...