美文网首页
java8-05-回顾

java8-05-回顾

作者: hylexus | 来源:发表于2017-12-31 14:51 被阅读26次

1 函数式接口的分类

常见的函数式接口可以大致分为以下几类:

  • 单输入,无输出
  • 单输入,单输出
  • 无输入,单输出
  • 两个不同类型的输入,第三种类型的输出
  • 两个不同类型的输入,其中一种类型的输出
  • ……

可以发现,无非以下三种大的分类:

  • 供给型:生产数据,一般无输入
  • 消费型:消费数据,一般无输出
  • 转换型:TypeA —> TypeB的类型转换

在实际使用中,没必要每次都新建这些函数式接口来支持lambda,JDK对各种类型的函数式接口已经基本都内置了。这些内置函数式接口都在 java.util.function 包下。

1.1 供给型

Supplier

1.2 消费型

Consumer

1.3 转换型

Function

1.4 Predicate (特殊的转换型接口)

Predicate

2 Stream

java8最吸引人的地方之一就是牛逼哄哄的 Stream-API 了。简单理解他就是升级版的 Iterator,比迭代器强大多了。

Stream

函数式编程关注的是对纯数据的处理,对于 Stream 的操作大致就是三个过程:

StreamProcess

本节注重于 Stream收集 操作,对于数据源和中间的处理过程略过。

2.1 Stream的获取

略过。

2.2 Stream的中间操作

略过。

2.3 Stream.collect()

对于Stream最终的收集操作有两个重载的版本:

<R, A> R collect(Collector<? super T, A, R> collector);

<R> R collect(Supplier<R> supplier,
              BiConsumer<R, ? super T> accumulator,
              BiConsumer<R, R> combiner);

要理解这两个方法,先看看 Collector 的几个方法。

3 Collector

/**
 * @param <T> Stream中的元素类型
 * @param <A> 收集过程中的临时中间类型
 * @param <R> 收集完成后输出结果的类型
 **/
public interface Collector<T, A, R> {
    /**
     * 提供一个结果容器。
     */
    Supplier<A> supplier();

    /**
     * 两个参数的消费型接口。将迭代中的当前元素添加到结果容器中。
     */
    BiConsumer<A, T> accumulator();

    /**
     * 转换型接口,两个相同输入,同类型的输出。
     * 在并行处理中,将各个子Stream返回结果合并。
     */
    BinaryOperator<A> combiner();

    /**
     * 收集操作的最后一步。
     * 转换型接口,用于将收集过程中的临时中间类型元素(A)转换为结果类型(R)。
     */
    Function<A, R> finisher();

    /**
     * 返回一个集合。描述了被收集的流的一些特性。
     * 比如:是否是顺序相关的、支不支持并行等。
     * 详情见:3.4 Characteristics
     */
    Set<Characteristics> characteristics();
}

3.1 Collector.collect()源码

源码中是这样的:

public final <R, A> R collect(Collector<? super P_OUT, A, R> collector) {
    A container;
    if (
      // 并行流(stream().parallel()方法被调用)的情况
      isParallel()
            && 
      // 优化提示中包含 CONCURRENT 属性
      (collector.characteristics().contains(Collector.Characteristics.CONCURRENT))
            && 
      // 流是无序的
      (!isOrdered() 
        || 
       // 优化提示中包含 UNORDERED 属性
       collector.characteristics().contains(Collector.Characteristics.UNORDERED))) {
        container = collector.supplier().get();
        BiConsumer<A, ? super P_OUT> accumulator = collector.accumulator();
        forEach(u -> accumulator.accept(container, u));
    }
    else {
        container = evaluate(ReduceOps.makeRef(collector));
    }
    return 
      // 优化提示中包含 IDENTITY_FINISH(恒等函数) 属性?
      collector.characteristics().contains(Collector.Characteristics.IDENTITY_FINISH)
           ? 
      // 直接将中间临时元素(A)当做最终结果(R)
      (R) container
           : 
        // 使用 finisher 转换成最终结果
        collector.finisher().apply(container);
}

从以上源码可以大致得出下面的两个流程图:

3.2 串行流的执行过程

sequential-stream

3.3 并行流的执行过程

parallel-stream

3.4 Characteristics

该类主要是给 Collector 接口在 collect() 的时候提供一些优化参数。

enum Characteristics {
    /**
     * 支持多线程调用,并行收集。
     */
    CONCURRENT,

    /**
     * 流中元素是无序的,收集过程不受元素先后顺序的影响。
     */
    UNORDERED,

    /**
     * 恒等函数。
     * 此时不再使用finisher再转换一次元素。直接将中间元素(A)当做最终结果(R)。
     */
    IDENTITY_FINISH
}

这三个特性的常用组合在 Collectors 工具类中也有定义:

static final Set<Collector.Characteristics> CH_CONCURRENT_ID
            = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.CONCURRENT,
                                                     Collector.Characteristics.UNORDERED,
                                                     Collector.Characteristics.IDENTITY_FINISH));
static final Set<Collector.Characteristics> CH_CONCURRENT_NOID
            = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.CONCURRENT,
                                                     Collector.Characteristics.UNORDERED));
static final Set<Collector.Characteristics> CH_ID
            = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.IDENTITY_FINISH));
static final Set<Collector.Characteristics> CH_UNORDERED_ID
            = Collections.unmodifiableSet(EnumSet.of(Collector.Characteristics.UNORDERED,
                                                     Collector.Characteristics.IDENTITY_FINISH));
static final Set<Collector.Characteristics> CH_NOID = Collections.emptySet();

对于流的收集操作先介绍这点,下次分享自定义Collector。

相关文章

  • java8-05-回顾

    1 函数式接口的分类 常见的函数式接口可以大致分为以下几类: 单输入,无输出 单输入,单输出 无输入,单输出 两个...

  • 周回顾回顾什么?

    关于周回顾,也是2018年常常遇到的一个时间管理概念。 下面是我总结出来18年遇到周回顾的几个地方: 有感兴趣的小...

  • 读书笔记|规划最好的一年|目标回顾

    目标回顾分为三个层次:每日回顾、每周回顾、季度回顾。 每日回顾 避免目标中途遗忘,每日目标回顾的作用就是建立总体目...

  • 回顾

    我今年26岁,我的儿时记忆已经记不起多少了,能回忆起来的事情大概要从03年非典了。03年我上六年级升初中 板蓝根卖...

  • 回顾

    今天AgileCommunity的话题是“如何提升迭代回顾会议的效率和生产力?” 这个话题在上次问题收集中排在首位...

  • 回顾

    坐在车上,望着窗外熟悉的风景,时光荏苒,不知不觉间,我在这座城市呆了已有十个年头了,曾经也有离开过,过了一段时间却...

  • 回顾

    我固执的认为我的父母没办法在一起生活,这个想法不仅害了我父母,也害了我自己。 为什么会有这个想法,源自于童年的恐惧...

  • 回顾

    时间过得好快,没想到这半年就过去了,回顾总是让我容易生发出感慨,还觉得有点难以想象。这一两个月,和G先生的...

  • 回顾

    看以前的字,想笑。感觉自己进步了。加油

  • 回顾

    没什么好说的一天,认真回顾了之前的笔记 笔记做的有点复杂,抓不住重点,需要改进 学习总是枯燥,希望自己能够坚持下去...

网友评论

      本文标题:java8-05-回顾

      本文链接:https://www.haomeiwen.com/subject/xdfagxtx.html