本文适用于想要了解Java8 Function接口编程及闭包表达式的筒鞋。

概述###

在实际开发中,常常遇到使用模板模式的场景: 主体流程是不变的,变的只是其中要调用的具体方法。 其特征是:

  BeginTodo ---> Something different to do ---> others todo ---> End

其中BeginTodo ,others todo,End 都是不变的,只有 Something different to do 是根据业务变化的。 如果采用 Java 来实现,通常要为 Something different to do 定义回调接口 Callback , 然后在主体流程中调用这个回调接口,而在实际业务中,创建 Callback 的实现类传入进去。

Java8 Function 以及闭包对此提供了更为简洁方便的语言支持。 Function 是Java8对函数的抽象,用于描述和接收任何单参数单返回的函数,类似于 Callback 的作用; 而在使用的时候,只要将方法引用或 lambda 表达式传给 Function 即可。

代码示例###

举例来说,要编写一个通用方法,实现多次执行同一个测试方法,统计失败次数。

Java-Function####

package com.xxx.trade;

import junit.framework.AssertionFailedError;
import junit.framework.TestCase;
import org.junit.Test; import java.util.function.Function; /**
* Created by shuqin on 16/11/15.
*/
public class GeneralExecTestMoreTimes extends TestCase { public void execMoreTimes(Function f) {
execMoreTimes(f, new Object());
} public void execMoreTimes(Function f, int num) {
execMoreTimes(f, null, num);
} public void execMoreTimes(Function f, Object t) {
execMoreTimes(f, t, 20);
} public void execMoreTimes(Function f, Object t, int num) {
int failed = 0;
for (int i = 0; i < num; i++) {
try {
f.apply(t);
} catch (AssertionFailedError afe) {
failed += 1;
} catch (Exception ex) {
failed += 1;
}
}
System.out.println("----- failed: " + failed + " -----");
assertEquals(0, failed);
} @Test
public void testExecMoreTimes() {
execMoreTimes((o) -> {
System.out.println("haha");
return 1;
}, 5);
} }

这里 execMoreTimes 方法接收一个函数式接口 Function 以及将应用于的参数 t 。 那么什么可以传给 Function 接口呢? 可以是任何单参数单返回值的函数,或者单参数单返回值的 lambda 表达式。例如代码中的 (o) -> { // some codes } 。

Groovy闭包####

如果使用 Groovy 编写闭包,会更简洁: 只要定义一个由大括号 {} 包围的代码块,并赋给一个变量即可(如下面的test1),甚至可以直接在函数里调用一个由大括号包围的代码块参数(如下面的test2)。 这样, 使用代码块或函数就会更加直接、灵活、自由,而不会受制于语法,也不需要定义一堆接口了。

package com.xxx.trade

import org.junit.Test

/**
* Created by shuqin on 16/11/15.
*/
class GeneralExecTestMoreTimesTest extends GeneralExecTestMoreTimes { @Test
public void test1() {
def closure = {
System.out.println("here is test")
}
execMoreTimes(closure)
} @Test
public void test2() {
execMoreTimes({
throw new Exception("throw exception in test")
})
}
}

应用场景####

函数接口可以用于任何使用回调接口的场景,一个典型的应用场景是批量处理。 比如有三个小组对同一个列表进行处理,一个用于状态同步,一个用于退款,一个用于取消。 那么,可以实现一个批量处理的通用函数,然后调用三个小组的自定义函数即可。

package zzz.study.utils;

import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer; /**
* Created by shuqin on 17/1/19.
*/
public class BatchProcessUtil { public static String batchProcessOrders(Consumer<String> processFunction, List<String> orders) {
StringBuilder result = new StringBuilder();
for (String orderNo: orders) {
String orderNoTrimed = orderNo.trim();
try {
processFunction.accept(orderNo);
result.append(orderNoTrimed + " OK , 请稍后查看!\n");
} catch (Exception e) {
result.append(orderNoTrimed + " Failed, 请稍后重试!\n");
}
}
return result.toString();
} public static void sync(String orderNo) {
System.out.println("sync order state " + orderNo);
} public static void refund(String orderNo) {
System.out.println("refund for " + orderNo);
} public static void cancel(String orderNo) {
System.out.println("cancel " + orderNo);
} public static void main(String[] args) {
List<String> orders = Arrays.asList(new String[] {"E001", "E002", "E003"});
batchProcessOrders((orderNo) -> sync(orderNo), orders);
batchProcessOrders((orderNo) -> refund(orderNo), orders);
batchProcessOrders((orderNo) -> cancel(orderNo), orders);
}
}

输出是

sync order state E001
sync order state E002
sync order state E003
refund for E001
refund for E002
refund for E003
cancel E001
cancel E002
cancel E003

常用函数接口###

常用函数接口主要有:

  1. Consumer (接收单参数无返回值的函数或lambda表达式), 方法是 void accept(T t);
  2. BiConsumer<T, U> (接收双参数无返回值的函数或 lambda表达式),方法是 void accept(T t, U u) ;
  3. Function<T, R> (接收单参数有返回值的函数或lambda表达式), 方法是 R apply(T t);
  4. BiFunction<T, U, R> (接收双参数有返回值的函数或lambda表达式),方法是 R apply(T t, U u);
  5. Predicate (接收单参数返回布尔值的函数或lambda表达式),方法是 boolean test(T t);
  6. Supplier (无参数返回值的函数或 lambda), 方法是 T get();
  7. 接受原子类型参数的函数接口,这里不一一列举了。可参考 java8 package java.util.function;

小结###

为什么要使用 Function 以及闭包呢?

  • 在语法上比定义回调接口、创建匿名类更加简洁;
  • 尝试使用新的语言特性,理解多样化的编程思想,提升编程表达能力。

Java8函数接口实现回调及Groovy闭包的代码示例的更多相关文章

  1. Java8-Function使用及Groovy闭包的代码示例

    导航 定位 概述 代码示例 Java-Function Groovy闭包 定位 本文适用于想要了解Java8 Function接口编程及闭包表达式的筒鞋. 概述 在实际开发中,常常遇到使用模板模式的场 ...

  2. java8 函数接口 Predicate例子

    import java.util.HashSet; import java.util.Collection; import java.util.function.Predicate; public c ...

  3. java8 函数接口

    [前言] java8新特性 java8 Optional使用总结 java8 lambda表达式 Java 8 时间日期使用 1.函数式接口新特性 java8中引入了函数式接口新特性,使用@Funct ...

  4. 开始Java8之旅(四) --四大函数接口

    前言   Java8中函数接口有很多,大概有几十个吧,具体究竟是多少我也数不清,所以一开始看的时候感觉一脸懵逼,不过其实根本没那么复杂,毕竟不应该也没必要把一个东西设计的很复杂. 几个单词   在学习 ...

  5. [一] java8 函数式编程入门 什么是函数式编程 函数接口概念 流和收集器基本概念

      本文是针对于java8引入函数式编程概念以及stream流相关的一些简单介绍 什么是函数式编程?   java程序员第一反应可能会理解成类的成员方法一类的东西 此处并不是这个含义,更接近是数学上的 ...

  6. Java8函数之旅(四) --四大函数接口

    前言   Java8中函数接口有很多,大概有几十个吧,具体究竟是多少我也数不清,所以一开始看的时候感觉一脸懵逼,不过其实根本没那么复杂,毕竟不应该也没必要把一个东西设计的很复杂. 几个单词   在学习 ...

  7. Java8学习笔记(二)--三个预定义函数接口

    三个函数接口概述 JDK预定义了很多函数接口以避免用户重复定义.最典型的是Function: @FunctionalInterface public interface Function<T, ...

  8. [二] java8 函数式接口详解 函数接口详解 lambda表达式 匿名函数 方法引用使用含义 函数式接口实例 如何定义函数式接口

    函数式接口详细定义 package java.lang; import java.lang.annotation.*; /** * An informative annotation type use ...

  9. java8函数式接口详解、函数接口详解、lambda表达式匿名函数、方法引用使用含义、函数式接口实例、如何定义函数式接口

    函数式接口详细定义 函数式接口只有一个抽象方法 由于default方法有一个实现,所以他们不是抽象的. 如果一个接口定义了一个抽象方法,而他恰好覆盖了Object的public方法,仍旧不算做接口的抽 ...

随机推荐

  1. JDK1.8版本,java并发框架支持锁包括

    1.自旋锁,自旋,jvm默认是10次,由jvm自己控制,for去争取锁 2.阻塞锁 被阻塞的线程,不会争夺锁 3.可重入锁,多次进入改锁的域 4.读写锁 5.互斥锁,锁本身就是互斥的 6.悲观锁,不相 ...

  2. postgresql----网络地址类型和函数

    本人对网络这块实在是搞不清楚,要是能有人推荐一下资料就好了!不知道有没有跟我一样呢?!所以在这里先贴一点从其他地方搞来的一些IPv4的东东. IPv4主要包括一下5类地址 A类: 0 7位 网络号 2 ...

  3. ubuntu部署nginx

    先更新本机内置的程序. sudo apt-get updatesudo apt-get upgrade再判断系统是否内置了add-apt-repository命令,如果没有执行下列命令安装 sudo ...

  4. HDU 6447 - YJJ's Salesman - [树状数组优化DP][2018CCPC网络选拔赛第10题]

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6447 Problem DescriptionYJJ is a salesman who has tra ...

  5. SSH免密码登录配置方法详解

    1.测试主机配置信息: 192.168.100.236 db06.chavin.king db06 192.168.100.237 db07.chavin.king db07 2.创建测试用户: gr ...

  6. oracle11gR2 win7_32位客户端连接虚拟机中oracle11gR2 win7_32位服务器方法

    改写服务器中的监听文件(listener.ora和tnsnames.ora) “ora-12541:TNS:无监听程序”问题的解决 ora-12541:TNS:无监听程序,出现这种错误的时候,可以尝试 ...

  7. hmm前后向算法

    跟医生就医推导过程是一样的 隐马尔科夫模型HMM(一)HMM模型 隐马尔科夫模型HMM(二)前向后向算法评估观察序列概率 隐马尔科夫模型HMM(三)鲍姆-韦尔奇算法求解HMM参数 隐马尔科夫模型HMM ...

  8. JS 防止表单重复提交的方法

    第一种:用flag标识,下面的代码设置checkSubmitFlg标志: <script language="”JavaScript”"> var checkSubmi ...

  9. 优云软件助阵ArchSummit全球架构师峰会

    由极客邦科技与 InfoQ 中国主办的 ArchSummit 全球架构师峰会深圳站,于 7 月 7 日 - 8 日在深圳华侨城洲际酒店举办. 本次大会特邀 100 多位国内外技术专家,与1000余名技 ...

  10. WordPress跳过语言包加载提高效率

    WordPress 加载语言包是需要花费 0.1-0.5 秒不等的时间,所以如果 WordPress 前台可以不加载语言包,而主题中的一些文本直接写成中文,就可以加快网站的速度,并且又能保证后台的中文 ...