什么是函数式编程

函数式编程是java8的一大特色,也就是将函数作为一个参数传递给指定方法。别人传的要么是基本数据类型,要么就是地址引用 ,我们要穿一个“动作”。

Stream

说到函数式编程,就不得不提及Stream,Stream跟我们熟知的io流可不是同一个东西,泛指可以顺序执行或者并行执行的元素序列,主要是针对集合,可以将多个函数通过“.”串起来执行,其特点如下:

  • stream不会存储数据,只是将集合流化,比如说 声明一个stream之后,往集合里面扔东西,stream可以取到新扔到集合里面的数据,你可以理解为操作时stream会实时从堆中的集合对象提取数据。
  • stream不会改变原集合,我的理解是stream 是一堆元素顺序或者并行执行我们串起来的函数,改变后并不会对集合中的元素造成影响。
  • steam是延迟执行的,也就是说在聚合操作之前 的其他操作,都会阻塞,直到执行聚合函数,其他的函数才开始一并执行。

解析相关接口

我们看看跟函数式编程相关的接口

java.long.FunctionalInterface是一个注解接口,函数接口都会实现它,看看它有什么特别的

从这里我们可以看出来一个函数接口只有一个抽象方法,但是如果要加一切其他的功能怎么办呢?接口中添加功能的话相当麻烦,接口相关实现类都需要修改,接下来“default”就登场了

default方法只能在接口出现,它不是抽象方法,可以通过methodInstance.isDefault()辨认,咦,弄啥勒,接口还能有普通方法?是的你没看错,添加default方法是为了方便修改接口,不至于每次修改过后都要将他的“儿子,孙子”等实现一一修改;如果在接口中default修饰的方法不加body会怎么样呢?   没错, 会编译失败!贴图:

接口中的default方法不用强制在实现类中 出现,当然业务需要的话,也可以重写来进行多态;

下面是java.util.function中的函数接口描述

序号 接口 & 描述
1 BiConsumer<T,U>

代表了一个接受两个输入参数的操作,并且不返回任何结果

2 BiFunction<T,U,R>

代表了一个接受两个输入参数的方法,并且返回一个结果

3 BinaryOperator<T>

代表了一个作用于于两个同类型操作符的操作,并且返回了操作符同类型的结果

4 BiPredicate<T,U>

代表了一个两个参数的boolean值方法

5 BooleanSupplier

代表了boolean值结果的提供方

6 Consumer<T>

代表了接受一个输入参数并且无返回的操作

7 DoubleBinaryOperator

代表了作用于两个double值操作符的操作,并且返回了一个double值的结果。

8 DoubleConsumer

代表一个接受double值参数的操作,并且不返回结果。

9 DoubleFunction<R>

代表接受一个double值参数的方法,并且返回结果

10 DoublePredicate

代表一个拥有double值参数的boolean值方法

11 DoubleSupplier

代表一个double值结构的提供方

12 DoubleToIntFunction

接受一个double类型输入,返回一个int类型结果。

13 DoubleToLongFunction

接受一个double类型输入,返回一个long类型结果

14 DoubleUnaryOperator

接受一个参数同为类型double,返回值类型也为double 。

15 Function<T,R>

接受一个输入参数,返回一个结果。

16 IntBinaryOperator

接受两个参数同为类型int,返回值类型也为int 。

17 IntConsumer

接受一个int类型的输入参数,无返回值 。

18 IntFunction<R>

接受一个int类型输入参数,返回一个结果 。

19 IntPredicate

:接受一个int输入参数,返回一个布尔值的结果。

20 IntSupplier

无参数,返回一个int类型结果。

21 IntToDoubleFunction

接受一个int类型输入,返回一个double类型结果 。

22 IntToLongFunction

接受一个int类型输入,返回一个long类型结果。

23 IntUnaryOperator

接受一个参数同为类型int,返回值类型也为int 。

24 LongBinaryOperator

接受两个参数同为类型long,返回值类型也为long。

25 LongConsumer

接受一个long类型的输入参数,无返回值。

26 LongFunction<R>

接受一个long类型输入参数,返回一个结果。

27 LongPredicate

R接受一个long输入参数,返回一个布尔值类型结果。

28 LongSupplier

无参数,返回一个结果long类型的值。

29 LongToDoubleFunction

接受一个long类型输入,返回一个double类型结果。

30 LongToIntFunction

接受一个long类型输入,返回一个int类型结果。

31 LongUnaryOperator

接受一个参数同为类型long,返回值类型也为long。

32 ObjDoubleConsumer<T>

接受一个object类型和一个double类型的输入参数,无返回值。

33 ObjIntConsumer<T>

接受一个object类型和一个int类型的输入参数,无返回值。

34 ObjLongConsumer<T>

接受一个object类型和一个long类型的输入参数,无返回值。

35 Predicate<T>

接受一个输入参数,返回一个布尔值结果。

36 Supplier<T>

无参数,返回一个结果。

37 ToDoubleBiFunction<T,U>

接受两个输入参数,返回一个double类型结果

38 ToDoubleFunction<T>

接受一个输入参数,返回一个double类型结果

39 ToIntBiFunction<T,U>

接受两个输入参数,返回一个int类型结果。

40 ToIntFunction<T>

接受一个输入参数,返回一个int类型结果。

41 ToLongBiFunction<T,U>

接受两个输入参数,返回一个long类型结果。

42 ToLongFunction<T>

接受一个输入参数,返回一个long类型结果。

43 UnaryOperator<T>

接受一个参数为类型T,返回值类型也为T。

除了上面列出的,还有个别函数接口没列出来,例如Comparator(), 但总归是个函数接口都会基于FunctionalInterface注解

总结

这些函数主要还是分为几个主要函数,其余都是以其为基础的分化;

  • Supplier<T> 无参数,返回一个结果。
  • Consumer<T> 代表了接受一个输入参数并且无返回的操作。
  • Function<T,R> 接受一个输入参数,返回一个结果,返回的结果跟参数的类型无关。
  • Predicate<T> 接受一个输入参数,返回一个布尔值结果。
  • BinaryOperator<T> 代表了一个作用于于两个同类型操作符的操作,并且返回了操作符同类型的结果,xxxOperator 函数都是接受类型跟返回值类型相同。

而且,从上面五个主要函数接口命名也可以看出其作用,"Bi"作为前缀的都是属于二元操作,接受两个参数。

T1toT2是从接受T1类型参数,返回T2类型结果,比如IntToLongFunction 接受一个int类型参数,返回long类型结果。

实战1

下面我写了一个功能,拳击手有分组跟体重两个属性,然后给一组拳击手集合,根据小组或者体重级排序,用到的是Comparator(),也是一个函数接口

上码:

 
  1. /**

  2. * Created by zhanghe

  3. * 拳击手

  4. */

  5. public class Boxer {

  6. //分组

  7. private String group;

  8. //重量级

  9. private Integer weight;

  10. public Boxer(String group, Integer weight) {

  11. this.group = group;

  12. this.weight = weight;

  13. }

  14. public static <T> List<List<T>> divider(List<T> datas, Comparator<T> c) {

  15. //声明一个列表来接收各个分组

  16. List<List<T>> list = new ArrayList<>();

  17. for (T t : datas) {

  18. //通过isSameGroup 标识来判断分组是否已创建

  19. boolean isSameGroup = false;

  20. for (int i = 0; i < list.size(); i++) {

  21. //compare函数返回值为int,正数说明param1>param2,0说明param1=param2,负数说明param1<param2

  22. //这里用到的原理是将List列表datas中的各项与分组列表list中的元素比较(比较的规则由外面作为参数传递,这就是函数式编程),

  23. // 值为0表示规则相符即为同一组

  24. if (c.compare(t, list.get(i).get(0)) == 0) {

  25. isSameGroup = true;

  26. list.get(i).add(t);

  27. break;

  28. }

  29. }

  30. //比较完了发现没有规则相符的,即自成一系

  31. if (!isSameGroup) {

  32. List<T> e = new ArrayList<>();

  33. e.add(t);

  34. list.add(e);

  35. }

  36. }

  37. return list;

  38. }

  39. public static void main(String[] args) {

  40. List<Boxer> boxers = Arrays.asList(

  41. new Boxer("红队", 120),

  42. new Boxer("绿队", 180),

  43. new Boxer("蓝队", 200),

  44. new Boxer("绿队", 220),

  45. new Boxer("蓝队", 120),

  46. new Boxer("红队", 80),

  47. new Boxer("红队", 90),

  48. new Boxer("绿队", 240)

  49. );

  50. List<List<Boxer>> dividByGroup = divider(boxers, new Comparator<Boxer>() {

  51. @Override

  52. public int compare(Boxer o1, Boxer o2) {

  53. //分组一样 即认为 相同

  54. return o1.group.equals(o2.group) == true ? 0 : 1;

  55. }

  56. });

  57. System.out.println("根据小组区分:");

  58. dividByGroup.stream().forEach(e->{

  59. System.out.println(e);

  60. });

  61. List<List<Boxer>> dividByWeight = divider(boxers, new Comparator<Boxer>() {

  62. @Override

  63. public int compare(Boxer o1, Boxer o2) {

  64. //体重/100 即体重百分位相同 即认为 相同

  65. return (o1.weight / 100 - o2.weight / 100) == 0 ? 0 : 1;

  66. }

  67. });

  68. System.out.println("根据体重区分:");

  69. dividByWeight.stream().forEach(e->{

  70. System.out.println(e);

  71. });

  72. }

  73. @Override

  74. public String toString() {

  75. return "Boxer{" +

  76. "小组='" + group + '\'' +

  77. ", 体重=" + weight +

  78. '}';

  79. }

  80. }

这个是结果

实战2

我用Predicate 写了个具有匹配功能的方法,当然 匹配的规则还是作为一个函数参数传递,撸码不是很优雅,只看功能就好~

闲话不多说,上码:

 
  1. List<Object> list = Arrays.asList(1, 2,"",100,"3", 'c');

  2. List<Object> matchList = match(list, a -> {

  3. return a instanceof Integer && (Integer) a > 10;

  4. });

  5. System.out.println(matchList);//100

  6. Set set = new HashSet<>();

  7. set.add("haha");

  8. set.add(1);

  9. set.add(100);

  10. Set matchSet = match(set, a -> a instanceof Integer);

  11. System.out.println(matchSet);//[1, 100]

  12. String matchString = match("java", a -> {

  13. return a instanceof String && ((String) a).startsWith("j");

  14. });

  15. System.out.println(matchString);//java

  16. Integer noMatch = match(100, a -> {

  17. return a instanceof String && ((String) a).startsWith("j");

  18. });

  19. System.out.println(noMatch);//null

  20. }

  21. /**

  22. *

  23. * @param t 入参 需要匹配规则的参数

  24. * @param p 函数接口 匹配的动作

  25. * @param <T>

  26. * @return 验证成功 是集合 则只留匹配元素,是String,Integer ..匹配成功返回本身,反之返回null

  27. */

  28. public static <T> T match(T t, Predicate p) {

  29. if (t instanceof Collection) {

  30. if (t instanceof List) {

  31. t = (T) ((List) t).stream().filter(a -> p.test(a)).collect(Collectors.toList());

  32. }

  33. if (t instanceof Set) {

  34. t = (T) ((Set) t).stream().filter(a -> p.test(a)).collect(Collectors.toSet());

  35. }

  36. } else {

  37. if (!p.test(t)) {

  38. return null;

  39. }

  40. }

  41. return t;

java8函数式编程实例的更多相关文章

  1. 关于Java8函数式编程你需要了解的几点

    函数式编程与面向对象的设计方法在思路和手段上都各有千秋,在这里,我将简要介绍一下函数式编程与面向对象相比的一些特点和差异. 函数作为一等公民 在理解函数作为一等公民这句话时,让我们先来看一下一种非常常 ...

  2. Java8 函数式编程详解

    Java8 函数式编程详解 Author:Dorae Date:2017年11月1日23:03:26 转载请注明出处 说起Java8,可能很多人都已经知道其最大的改进,就是引入了Lambda表达式与S ...

  3. Python进阶:函数式编程实例(附代码)

    Python进阶:函数式编程实例(附代码) 上篇文章"几个小例子告诉你, 一行Python代码能干哪些事 -- 知乎专栏"中用到了一些列表解析.生成器.map.filter.lam ...

  4. Java8函数式编程探秘

    引子 将行为作为数据传递 怎样在一行代码里同时计算一个列表的和.最大值.最小值.平均值.元素个数.奇偶分组.指数.排序呢? 答案是思维反转!将行为作为数据传递. 文艺青年的代码如下所示: public ...

  5. [2017.02.23] Java8 函数式编程

    以前学过Haskell,前几天又复习了其中的部分内容. 函数式编程与命令式编程有着不一样的地方,函数式编程中函数是第一等公民,通过使用少量的几个数据结构如list.map.set,以及在这些数据结构上 ...

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

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

  7. Java8函数式编程和lambda表达式

    文章目录函数式编程JDK8接口新特性函数接口方法引用函数式编程函数式编程更多时候是一种编程的思维方式,是一种方法论.函数式与命令式编程区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉 ...

  8. 漫漫人生路,学点Jakarta基础-Java8函数式编程

    接口默认方法 Java8版本以后新增了接口的默认方法,不仅仅只能包含抽象方法,接口也可以包含若干个实例方法.在接口内定义实例方法(但是注意需要使用default关键字) 在此定义的方法并非抽象方法,而 ...

  9. Java8函数式编程的宏观总结

    1.java8优势通过将行为进行抽象,java8提供了批量处理数据的并行类库,使得代码可以在多核CPU上高效运行. 2.函数式编程的核心使用不可变值和函数,函数对一个值进行处理,映射成另一个值. 3. ...

随机推荐

  1. 鬼斧神工:求n维球的体积

    原文地址:http://spaces.ac.cn/archives/3154/ 原文作者:苏剑林 标准思路 简单来说,\(n\)维球体积就是如下\(n\)重积分 \[V_n(r)=\int_{x_1^ ...

  2. 【10】DP青蛙跳台阶

    一只青蛙一次可以跳上1级台阶,也可以跳上2级台阶.求该青蛙跳上一个 n 级的台阶总共有多少种跳法. 答案需要取模 1e9+7(1000000007),如计算初始结果为:1000000008,请返回 1 ...

  3. 【Python实现图片验证码】

    "```python import base64 import random from PIL import Image from PIL import ImageDraw # 画笔对象 f ...

  4. maven设置指定jdk版本

    今天心血来潮准备折腾一下jeecg,去下载了一个maven版本的项目,既然下载了maven版的,当然就要配置好maven环境了. 因为之前简单学习过maven,当时使用的版本是3.3.9的,但是今天在 ...

  5. 解决docker: error pulling image configuration: Get https://registry-1.docker.io/v2/library/mysql/: TLS handshake timeout.

    出现这个问题,一般的原因是无法连接到 docker hub,通过: systemctl stop docker echo "DOCKER_OPTS=\"\$DOCKER_OPTS ...

  6. allegro 16.6 铜皮显示问题

    Setup-->User Preference-->display-->opengl-->staic-shapes_fill_solid打勾.可以将栅格铜皮改为实铜, Setu ...

  7. Caused by: com.fasterxml.jackson.databind.exc.InvalidDefinitionException

    实体类缺少无参构造方法,序列化对象需要无参构造方法 @NoArgsConstructor

  8. Laravel Vuejs 实战:开发知乎 (10)使用 Select2 优化话题选择

    1.添加选择Topic 使用Select2,如何安装Select2 ,具体使用实例 Select2 and Laravel: Ajax Autocomplete 及 Loading data remo ...

  9. react组件之间传值方式

    1.父向子(通过props传值) 2.父向更深层的子(通过context传值) 3.子向父(通过回调函数传值:在父组件中创建一个函数来接收子组件传过来的参数值,通过父组件将这个函数做为子组件的属性传递 ...

  10. 公用技术——面向对象领域——UML图——《The Unified Modeling Language User Guide》V2读书笔记——第一章节(建模的意义)

    第一章节到第三章节介绍UML的基本概念.第一章节主要介绍了UML语言的历史,介绍了建模的重要性(狗窝,房子,大厦),介绍了UML要实现哪些目标,在最后介绍了在使用UML语言时应该遵循的一些原则或者是规 ...