Java-8函数式编程设计-Functional-Interface

Java 8函数式编程设计(Functional interfaces)

我自己的理解,函数式编程对用户最大的价值是促使开发者养成模块化编程的习惯,代码可读性和维护性提高很多。

通过阅读JDK 8的 java.util.function
java.util.stream 包源码,意在理解Java的函数式接口设计。

读后自己的理解:Java函数式编程的核心是将最基础的数学函数抽象成接口对象,可在已有的接口上进行积木拼插组合,形成完整地类型转换系统。最基础的数学函数包括一元函数、谓词、二元函数、运算符计算,对应的Java接口分别是Function、Predicate、BiFunction、BinaryOperator。

源码阅读笔记:jdk-8-Functional-Interface


函数式接口API

  • @FunctionalInterface
  • Functional interfaces
    • Function
    • Consumer
    • Predicate
    • Supplier
  • 数据流特性和操作
    • BaseStream
    • Stream
  • 使用示例

@FunctionalInterface

  • 函数式接口注解
  • 从概念上讲,函数式接口都只有一个抽象方法
  • 请注意,可以使用Lambda表达式、方法引用和构造器引用创建函数式接口的实例。

Functional interfaces

  • 函数式接口提供lambda表达式和方法引用的目标类型
  • 每个函数式接口有一个单一的抽象方法,称为函数方法
  • 函数式接口可以匹配或适配为lambda表达式的参数和返回类型
  • 函数式接口可以在多个上下文中提供一个目标类型,如赋值上下文、方法调用上下文、转换上下文
  • 函数式接口往往代表抽象的概念,如函数、操作、谓词
  • 可扩展的命名约定
  • 基本的一元函数:Function、Consumer、Predicate、Supplier
  • 二元函数:BiFunction、BiConsumer、BiPredicate
  • 扩展的运算符:UnaryOperator、BinaryOperator
  • 泛型->基本类型:int、long、double

一元函数

Function<T, R>

  • 从T到R的一元映射函数,接受一个参数并产生一个结果的一元函数 (类型转换函数)
  • R apply(T t)
  • 组合函数
  • compose(before):V -> T -> R
  • andThen(after):T -> R -> V
  • 基本类型的参数:IntFunction<R>、LongFunction<R>、DoubleFunction<R>
  • 基本类型的结果:ToIntFunction<T>、ToLongFunction<T>、ToDoubleFunction<T>
  • 基本类型的参数和结果:IntToLongFunction、IntToDoubleFunction

Consumer<T>

  • 从T到void的一元函数,接受一个入参但不返回任何结果的操作
  • void accept(T t)
  • andThen(after):N个消费者模式,多次消费的场景
  • 基本类型的参数:IntConsumer、LongConsumer、DoubleConsumer

Predicate<T>

  • 一个参数的谓词(返回布尔值的函数)
  • boolean test(T t)
  • 谓词函数:and、negate、or
  • 基本类型的参数:IntPredicate、LongPredicate、DoublePredicate

Supplier<T>

  • 表示结果的供应商
  • T get()
  • 基本类型的结果:BooleanSupplier、IntSupplier、LongSupplier、DoubleSupplier

使用图表方式总结如下:

一元函数 方法 类型转换 组合方法 基本类型专用类
Function<T, R> R apply(T t) T->R compose(before)、andThen(after) IntFunction<R>、ToIntFunction<T>、IntToLongFunction
Consumer<T> void accept(T t) T->void andThen(after) IntConsumer
Predicate<T> boolean test(T t) T->boolean and、negate、or IntPredicate
Supplier<T> T get() *->T IntSupplier、BooleanSupplier

二元函数

BiFunction<T, U, R>

  • 从(T、U)到R的二元函数,接受两个参数并产生一个结果的二元函数
  • R apply(T t, U u)
  • 组合函数
  • andThen(Function<? super R, ? extends V> after):(T, U) -> R -> V
  • 基本类型的结果:ToIntBiFunction<T, U>、ToLongBiFunction<T, U>、ToDoubleBiFunction<T, U>

BiConsumer<T, U>

  • 从(T、U)到void的二元函数,接受两个入参但不返回任何结果的操作
  • void accept(T t, U u)
  • 基本类型的参数:ObjIntConsumer<T>、ObjLongConsumer<T>、ObjDoubleConsumer<T>

BiPredicate<T, U>

  • 两个参数的谓词(返回布尔值的函数)
  • boolean test(T t, U u)
  • 谓词函数:and、negate、or

使用图表方式总结如下:

二元函数 方法 类型转换 组合方法 基本类型专用类
BiFunction<T, U, R> R apply(T t, U u) (T、U)->R andThen(Function after) ToIntBiFunction<T, U>
BiConsumer<T, U> void accept(T t, U u) (T、U)->void andThen(after) ObjIntConsumer<T>
BiPredicate<T, U> boolean test(T t, U u) (T、U)->boolean and、negate、or

扩展的运算符

UnaryOperator<T>

  • 一元运算符(单个操作数,生产与操作数类型相同的结果)
  • 继承自Function<T, T>
  • 基本类型的运算符:IntUnaryOperator、LongUnaryOperator、DoubleUnaryOperator

BinaryOperator<T>

  • 二元运算符(两个相同类型的操作数,生产与操作数类型相同的结果)
  • 继承自BiFunction<T, T, T>
  • 基本类型的运算符:IntBinaryOperator、LongBinaryOperator、DoubleBinaryOperator
运算符 父类 组合方法 基本类型专用类
UnaryOperator<T> Function<T, T> IntUnaryOperator: int applyAsInt(int operand)
BinaryOperator<T> BiFunction<T, T, T> minBy、maxBy IntBinaryOperator: int applyAsInt(int left, int right)

数据流操作

BaseStream<T,S extends BaseStream<T,S>>

数据流操作特性

  • 顺序流
  • 并行流
  • 拆分器
  • 关闭数据流(数据流管道)

Stream<T>

数据流

  • 中间操作,返回Stream<T>
    • filter
    • distinct
    • map
    • flatMap
    • peek
  • 终结操作,返回结果/Optional<T>
    • collect
    • forEach
    • reduce
    • anyMatch
    • findAny
    • findFirst
    • count
    • max
    • min

使用示例

Optional<T>

单值数据流

  • 可能包含null值的容器对象,包装方法的返回结果
  • empty()、of(T value)、ofNullable(T value)
  • isPresent()、get()
  • void ifPresent(Consumer<? super T> consumer)
  • Optional<T> filter(Predicate<? super T> predicate):过滤操作
  • Optional<U> map(Function<? super T, ? extends U> mapper)
  • Optional<U> flatMap(Function<? super T, Optional<U>> mapper)
  • T orElse(T other)、T orElseGet(Supplier<? extends T> other)、T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X
  • 基本类型实现:OptionalInt、OptionalLong、OptionalDouble
  • 使用模式:
    • Optional<T>.ifPresent(consumer)
    • Optional<T>.filter(predicate).flatMap(mapper)
    • Optional<T>.filter(predicate).map(mapper)

祝玩得开心!ˇˍˇ
云舒,2017.7.26,杭州

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351

推荐阅读更多精彩内容