【Java 8】函数式接口(一)—— Functional Interface简介
什么是函数式接口(Functional Interface)
其实之前在讲Lambda表达式的时候提到过,所谓的函数式接口,当然首先是一个接口,然后就是在这个接口里面只能有一个抽象方法。
这种类型的接口也称为SAM接口,即Single Abstract Method interfaces。
函数式接口用途
它们主要用在Lambda表达式和方法引用(实际上也可认为是Lambda表达式)上。
如定义了一个函数式接口如下:
@FunctionalInterface
interface GreetingService
{
void sayMessage(String message);
}
那么就可以使用Lambda表达式来表示该接口的一个实现(注:JAVA 8 之前一般是用匿名类实现的):
GreetingService greetService1 = message -> System.out.println("Hello " + message);
关于@FunctionalInterface注解
Java 8为函数式接口引入了一个新注解@FunctionalInterface,主要用于编译级错误检查,加上该注解,当你写的接口不符合函数式接口定义的时候,编译器会报错。
正确例子,没有报错:
@FunctionalInterface
interface GreetingService
{
void sayMessage(String message);
}
错误例子,接口中包含了两个抽象方法,违反了函数式接口的定义,Eclipse报错提示其不是函数式接口。

提醒:加不加@FunctionalInterface对于接口是不是函数式接口没有影响,该注解知识提醒编译器去检查该接口是否仅包含一个抽象方法
函数式接口里允许定义默认方法
函数式接口里是可以包含默认方法,因为默认方法不是抽象方法,其有一个默认实现,所以是符合函数式接口的定义的;
如下代码不会报错:
@FunctionalInterface
interface GreetingService
{
void sayMessage(String message);
default void doSomeMoreWork1()
{
// Method body
}
default void doSomeMoreWork2()
{
// Method body
}
}
函数式接口里允许定义静态方法
函数式接口里是可以包含静态方法,因为静态方法不能是抽象方法,是一个已经实现了的方法,所以是符合函数式接口的定义的;
如下代码不会报错:
@FunctionalInterface
interface GreetingService
{
void sayMessage(String message);
static void printHello(){
System.out.println("Hello");
}
}
函数式接口里允许定义java.lang.Object里的public方法
函数式接口里是可以包含Object里的public方法,这些方法对于函数式接口来说,不被当成是抽象方法(虽然它们是抽象方法);因为任何一个函数式接口的实现,默认都继承了Object类,包含了来自java.lang.Object里对这些抽象方法的实现;
如下代码不会报错:
@FunctionalInterface
interface GreetingService
{
void sayMessage(String message);
@Override
boolean equals(Object obj);
}
JDK中的函数式接口举例
java.lang.Runnable,java.awt.event.ActionListener, java.util.Comparator,java.util.concurrent.Callable
java.util.function包下的接口,如Consumer、Predicate、Supplier等
java.util.function 提供的函数式接口汇总
java.util.function包中所有的接口整理:
Function
表示一个方法接收参数并返回结果。
接收单个参数
| Interface | functional method | 说明 |
|---|---|---|
| Function<T,R> | R apply(T t) | 接收参数类型为T,返回参数类型为R |
| IntFunction | R apply(int value) | 以下三个接口,指定了接收参数类型,返回参数类型为泛型R |
| LongFunction | R apply(long value) | |
| Double | R apply(double value) | |
| ToIntFunction | int applyAsInt(T value) | 以下三个接口,指定了返回参数类型,接收参数类型为泛型T |
| ToLongFunction | long applyAsLong(T value) | |
| ToDoubleFunction | double applyAsDouble(T value) | |
| IntToLongFunction | long applyAsLong(int value) | 以下六个接口,既指定了接收参数类型,也指定了返回参数类型 |
| IntToDoubleFunction | double applyAsLong(int value) | |
| LongToIntFunction | int applyAsLong(long value) | |
| LongToDoubleFunction | double applyAsLong(long value) | |
| DoubleToIntFunction | int applyAsLong(double value) | |
| DoubleToLongFunction | long applyAsLong(double value) | |
| UnaryOperator | T apply(T t) | 特殊的Function,接收参数类型和返回参数类型一样 |
| IntUnaryOperator | int applyAsInt(int left, int right) | 以下三个接口,指定了接收参数和返回参数类型,并且都一样 |
| LongUnaryOperator | long applyAsInt(long left, long right) | |
| DoubleUnaryOperator | double applyAsInt(double left, double right) |
接收两个参数
| interface | functional method | 说明 |
|---|---|---|
| BiFunction<T,U,R> | R apply(T t, U u) | 接收两个参数的Function |
| ToIntBiFunction<T,U> | int applyAsInt(T t, U u) | 以下三个接口,指定了返回参数类型,接收参数类型分别为泛型T, U |
| ToLongBiFunction<T,U> | long applyAsLong(T t, U u) | |
| ToDoubleBiFunction<T,U> | double appleyAsDouble(T t, U u) | |
| BinaryOperator | T apply(T t, T u) | 特殊的BiFunction, 接收参数和返回参数类型一样 |
| IntBinaryOperator | int applyAsInt(int left, int right) | |
| LongBinaryOperator | long applyAsInt(long left, long right) | |
| DoubleBinaryOperator | double applyAsInt(double left, double right) |
Consumer
表示一个方法接收参数但不产生返回值。
接收一个参数
| interface | functional method | 说明 |
|---|---|---|
| Consumer | void accept(T t) | 接收一个泛型参数,无返回值 |
| IntConsumer | void accept(int value) | 以下三个类,接收一个指定类型的参数 |
| LongConsumer | void accept(long value) | |
| DoubleConsumer | void accept(double value) |
接收两个参数
| interface | functional method | 说明 |
|---|---|---|
| BiConsumer<T,U> | void accept(T t, U u) | 接收两个泛型参数 |
| ObjIntConsumer | void accept(T t, int value) | 以下三个类,接收一个泛型参数,一个指定类型的参数 |
| ObjLongConsumer | void accept(T t, long value) | |
| ObjDoubleConsumer | void accept(T t, double value) |
Supplier
返回一个结果,并不要求每次调用都返回一个新的或者独一的结果
| interface | functional method | 说明 |
|---|---|---|
| Supplier | T get() | 返回类型为泛型T |
| BooleanSupplier | boolean getAsBoolean() | 以下三个接口,返回指定类型 |
| IntSupplier | int getAsInt() | |
| LongSupplier | long getAsLong() | |
| DoubleSupplier | double getAsDouble() |
Predicate
根据接收参数进行断言,返回boolean类型
| interface | functional method | 说明 |
|---|---|---|
| Predicate | boolean test(T t) | 接收一个泛型参数 |
| IntPredicate | boolean test(int value) | 以下三个接口,接收指定类型的参数 |
| LongPredicate | boolean test(long value) | |
| DoublePredicate | boolean test(double value) | |
| BiPredicate<T,U> | boolean test(T t, U u) | 接收两个泛型参数,分别为T,U |
【Java 8】函数式接口(一)—— Functional Interface简介的更多相关文章
- 函数式接口(Functional Interface)
原文链接:https://www.cnblogs.com/runningTurtle/p/7092632.html 阅读目录 什么是函数式接口(Functional Interface) 函数式接口用 ...
- JAVA 8 函数式接口 - Functional Interface
什么是函数式接口(Functional Interface) 其实之前在讲Lambda表达式的时候提到过,所谓的函数式接口,当然首先是一个接口,然后就是在这个接口里面只能有一个抽象方法. 这种类型的接 ...
- Java 8函数式接口functional interface的秘密
Java 8函数式接口functional interface的秘密 2014年10月29日 17:52:55 西瓜可乐520 阅读数:3729 目录 [−] JDK 8之前已有的函数式接口 新定 ...
- Java 8 函数式接口
函数式接口(Functional Interface)就是一个有且仅有一个抽象方法,但是可以有多个非抽象方法的接口. 函数式接口可以被隐式转换为 lambda 表达式. Lambda 表达式和方法引用 ...
- Java 8 新特性-菜鸟教程 (3) -Java 8 函数式接口
Java 8 函数式接口 函数式接口(Functional Interface)就是一个有且仅有一个抽象方法,但是可以有多个非抽象方法的接口. 函数式接口可以被隐式转换为lambda表达式. 函数式接 ...
- Java之函数式接口@FunctionalInterface详解(附源码)
Java之函数式接口@FunctionalInterface详解 函数式接口的定义 在java8中,满足下面任意一个条件的接口都是函数式接口: 1.被@FunctionalInterface注释的接口 ...
- Java之函数式接口
函数式接口 概述:接口中只有一个抽象方法 下面介绍的可能很抽象,理解不了,至少在我看来单独的这几个借口是没有用的,跟最下面说的 Stream流一起用才会有效果 函数式接口,即适用于函数式编程场景的接口 ...
- Java常用函数式接口--Consumer接口andThen()方法使用案例(二)
Java常用函数式接口--Consumer接口使用案例
- Java常用函数式接口--Predicate接口使用案例
Java常用函数式接口--Predicate接口使用案例 该方法可以使用and来优化: 调用:
随机推荐
- java读取大文件内容到Elasticsearch分析(手把手教你java处理超大csv文件)
现在需要快算分析一个2g的csv文件: 基于掌握的知识,使用java按行读取文件,批量导入数据到es, 然后利用es强大的聚合能力分析数据,2个小时搞定! package com.example.de ...
- 消息队列手动确认Ack
以RabbitMQ为例,默认情况下 RabbitMQ 是自动ACK机制,就意味着 MQ 会在消息发送完毕后,自动帮我们去ACK,然后删除消息的信息.这样依赖就存在这样一个问题:如果消费者处理消息需要较 ...
- 经过4次优化我把python代码耗时减少95%
背景交代 团队做大学英语四六级考试相关服务.业务中有一个care服务,购买了care服务考试不过可以全额退款,不过有一个前提是要完成care服务的任务,比如坚持背单词N天,完成指定的试卷. 在这个背景 ...
- ThreadPoolExecutor里面4种拒绝策略(详细)
ThreadPoolExecutor类实现了ExecutorService接口和Executor接口,可以设置线程池corePoolSize,最大线程池大小,AliveTime,拒绝策略等.常用构造方 ...
- 在同级路径下,SpringBoot两种类型的配置文件(.properties/.yml)同时存在时,配置优先级如何处理?
两类配置文件如果同时存在,若 key 相同则 properties 优先级高,若key不同则合并加载:
- mysql 数据库中 int(3) 和 int(11) 有区别么???
今天去面试的时候 面试官问到了这个问题:int(3) 和 int(11) 有什么区别?? 当时一听有点蒙,(不知道为什么蒙,后来回来想想可能是觉得考官怎么会问这么简单的问题呢,所以蒙了),当时我的回答 ...
- 菜鸡的Java笔记 第三十五 接口定义增强
接口定义增强 在java从一开始到现在接口之中的核心组成部分:抽象方法与全局常量,但是随着技术的不断发展,用户在使用过程之中发现了有一些问题 如果说现在有一个接口经过了长年 ...
- Hi3516开发笔记(三):Hi3516虚拟机基础环境搭建之交叉编译环境境搭建以及开机启动脚本分析
前言 前面进行了可以传输,那么写一个简单的C程序来交叉编译并传入运行. 虚拟机 上一篇搭建的虚拟机环境,包含了sftp传递文件,网络能ping通,基于上一篇的虚拟机继续搭建. 海思交叉 ...
- 未能加载文件或程序集“Microsoft.CodeDom.Providers.DotNetCompilerPlatform
"/"应用程序中的服务器错误. 未能加载文件或程序集"Microsoft.CodeDom.Providers.DotNetCompilerPlatform, Versio ...
- 31、下一个排列 | 算法(leetode,附思维导图 + 全部解法)300题
零 标题:算法(leetode,附思维导图 + 全部解法)300题之(31)下一个排列 一 题目描述 二 解法总览(思维导图) 三 全部解法 1 方案1 1)代码: // 方案1 "双指针法 ...