函数式接口

Lambda表达式需要接口的支持,并且接口的抽象方法还只能有一个,要么没法区分实现的是哪一个抽象方法了。因此Lambda表达式需要函数式接口的支持

什么是函数式接口

接口中只有一个抽象方法的接口称为函数式接口。

函数式接口可以使用一个注解@FunctionalInterface修饰,此注解可以检查是否是函数式接口

函数式接口的使用

假设我们现在有一个需求:对一个数进行运算,什么运算都可以。如果我们想用Lambda表达式来实现的话,我们就需要一个接口来支持,下面我们先写一个MyFunction接口

  1. @FunctionalInterface
  2. public interface MyFunction {
  3. public Integer getValue(Integer num);
  4. }

然后我们需要一个运算的方法

  1. //运算方法
  2. public Integer operation(Integer num, MyFunction fun){
  3. return fun.getValue(num);
  4. }

测试类

  1. @Test
  2. public void test06(){
  3. //平方运算
  4. Integer num = operation(10,(x) -> x*x);
  5. System.out.println(num);
  6. //加和运算
  7. Integer num2 = operation(100,(y) -> y+200);
  8. System.out.println(num2);
  9. }

执行结果

  1. 100
  2. 300

不管是什么运算,我们只需要关注Lambda表达式的方法体如何实现运算就可以了。

通过以上例子发现,如果我们想用Lambda表达式来实现的话,我们就需要写一个接口来支持,是不是也挺麻烦的,其实,一般我们常用的接口,Java8里基本都已经给我们提供了,下面来看下Java8内置的核心接口。

Java8内置的四大核心函数式接口

一、Consumer:消费型接口(void accept(T t))

我们先看一下Java8中给我定义好的Consumer接口

  1. @FunctionalInterface
  2. public interface Consumer<T> {
  3. /**
  4. * Performs this operation on the given argument.
  5. *
  6. * @param t the input argument
  7. */
  8. void accept(T t);
  9. default Consumer<T> andThen(Consumer<? super T> after) {
  10. Objects.requireNonNull(after);
  11. return (T t) -> { accept(t); after.accept(t); };
  12. }
  13. }

然后我们自己写个简单的例子看一下,写个开开森森去消费的方法

  1. @Test
  2. public void test01(){
  3. happy("一消费就很开森啦啦啦",str -> System.out.println(str));
  4. }
  5. public void happy(String str,Consumer<String> consumer){
  6. consumer.accept(str);
  7. }

运行结果

  1. 一消费就很开森啦啦啦

以上就是消费型接口,消费即是有去无回,有参数无返回值类型的接口。

二、Supplier:供给型接口(T get())

我们先看一下Java8中给我定义好的Supplier接口

  1. @FunctionalInterface
  2. public interface Supplier<T> {
  3. /**
  4. * Gets a result.
  5. *
  6. * @return a result
  7. */
  8. T get();
  9. }

我们再来写个例子,写个方法产生指定个数的整数并放入集合中

  1. @Test
  2. public void test02(){
  3. List<Integer> numList = getValue(10,() -> (int)(Math.random()*100));
  4. for (Integer num :
  5. numList) {
  6. System.out.println(num);
  7. }
  8. }
  9. //产生指定个数的整数并放入集合中
  10. public List<Integer> getValue(int num,Supplier<Integer> supplier){
  11. List<Integer> list = new ArrayList<>();
  12. for (int i = 0; i < num; i++) {
  13. list.add(supplier.get());
  14. }
  15. return list;
  16. }

执行结果

  1. 84
  2. 22
  3. 60
  4. 29
  5. 52
  6. 34
  7. 9
  8. 95
  9. 34
  10. 6

以上就是供给型接口,只有产出,没有输入,即只有返回值没有入参,给你产生一些对象,并返回给你。

三、Function : 函数型接口(R apply(T t))

Java8中函数式接口的定义

  1. @FunctionalInterface
  2. public interface Function<T, R> {
  3. /**
  4. * Applies this function to the given argument.
  5. *
  6. * @param t the function argument
  7. * @return the function result
  8. */
  9. R apply(T t);
  10. ····后面代码省略了····
  11. }

下面写个处理字符串方法的例子

  1. @Test
  2. public void test03(){
  3. String returnStr = strHandler("\t\t\t我是一串有空格\t的字符串\t\t",(str) -> str.trim());
  4. System.out.println(returnStr);
  5. }
  6. public String strHandler(String str,Function<String,String> function){
  7. return function.apply(str);
  8. }

运行结果:trim()方法只是去除首尾的字符串

  1. 我是一串有空格 的字符串

以上就是一个函数型接口,一个参数,一个返回值,当然参数和返回值的类型可以相同

四、Predicate:断言型接口(boolean test(T t))

Java8中断言型接口的定义如下

  1. @FunctionalInterface
  2. public interface Predicate<T> {
  3. /**
  4. * Evaluates this predicate on the given argument.
  5. *
  6. * @param t the input argument
  7. * @return {@code true} if the input argument matches the predicate,
  8. * otherwise {@code false}
  9. */
  10. boolean test(T t);
  11. ····后面到吗省略····
  12. }

下面我们写一个断言的例子,写个方法将满足条件的字符串放入集合中

  1. @Test
  2. public void test04(){
  3. List<String> list = Arrays.asList("aa","bbb","ccccc","dddddd");
  4. //长度大于3的字符串放入集合中
  5. List<String> reList = filterStr(list,(s) -> s.length() >3);
  6. for (String str: reList) {
  7. System.out.println(str);
  8. }
  9. }
  10. public List<String> filterStr(List<String> strList,Predicate<String> pre){
  11. List<String> list = new ArrayList<>();
  12. for (String str: strList) {
  13. if(pre.test(str)){
  14. list.add(str);
  15. }
  16. }
  17. return list;
  18. }

运行结果

  1. ccccc
  2. dddddd

以上是断言型接口,输入一个参数,输出一个boolean类型的返回值

五、其他类型的一些函数式接口

除了上面四种类型的核心接口还有一些其他的不常用的接口,我们可以了解一下

  1. BiFunction<T, U, R>
  • 类型:参数类型T,U返回类型R
  • 用途:Function的子接口,对类型为T和U的参数应用操作,返回R类型的结果。包含有方法R apply(T t,U u)
  1. UnaryOperator(Function子接口)
  • 类型:参数类型T,返回类型T
  • 用途:Function的子接口,对类型为T的对象进行一元运算,并返回T类型的结果。包含有方法T apply(T t)
  1. BinaryOperator(BiFunction子接口)
  • 类型:参数类型T,T,返回类型T
  • 用途:BiFunction的子接口,对类型为T的对象进行二元运算,并返回T类型的结果。包含有方法T apply(T t1,T t2)
  1. BiConsumcr(T, U)
  • 类型:参数类型T, U,返回类型void
  • 用途:Consumcr的子接口,对类型为T,U的参数应用操作。包含有方法void accept(T t,U u)
  1. ToIntFunction、ToLongFunction、ToDoubleFunction
  • 类型:参数类型T,返回类型int long double
  • 用途:分别计算int long double值的函数
  1. IntFunction、LongFunction、DoubleFunction
  • 类型:参数类型int long double,返回类型T
  • 用途:参数分别为int long double类型的函数

总结

以上就是Java8内置的核心接口,我们在使用的过程中,按需所取就ok啦。

Java8的新特性--函数式接口的更多相关文章

  1. Java8 新特性 函数式接口

    什么是函数式接口   函数式接口是Java8引用的一个新特性,是一种特殊的接口:SAM类型的接口(Single Abstract Method).但是它还是一个接口,只是有些特殊罢了.  函数式接口的 ...

  2. Java8 新特性----函数式接口,以及和Lambda表达式的关系

    这里来讲解一下Java8 新特性中的函数式接口, 以及和Lambda 表达式的关系.看到过很多不少介绍Java8特性的文章,都会介绍到函数式接口和lambda表达式,但是都是分别介绍,没有将两者的关系 ...

  3. java8新特性-函数式接口详细讲解及案例

    一.函数式接口 1.1 概念 函数式接口在Java中是指:有且仅有一个抽象方法的接口.函数式接口,即适用于函数式编程场景的接口.而Java中的函数式编程体现就是Lambda,所以函数式接口就是可 以适 ...

  4. jdk8新特性--函数式接口的使用

    函数式接口的概念: 函数式接口的格式: 示例: 函数式接口的使用: 简化lambda表达式:

  5. java8的新特性以及用法简介

    1. 介绍 2 接口的默认方法 2 lambda表达式 2.1 函数式接口 2.2 方法与构造函数引用 2.3 访问局部变量 2.4 访问对象字段与静态变量 3. 内建函数式接口 3.1 Predic ...

  6. Java8一Lambda与函数式接口

    关于Lambda表示在工作学习中会经常用到,但并没有全面的去了解.在这里做一个较为详细的记录供以后学习查阅.主要参考Java 8 Lambda 表达式 引言 Java8之前,我们在使用Runnale创 ...

  7. 简单了解JAVA8的新特性

    JAVA8新特性会颠覆整个JAVA程序员的编程习惯 甚至如果您坚守JAVA7之前的编程习惯,今后你看比较年轻的程序员写的JAVA代码都会无法理解 所以为了保证不脱钩,我觉得有必要学习JAVA8的新特性 ...

  8. Java 8 特性 —— 函数式接口

    函数式接口 概述:接口中只有一个抽象方法. 函数式接口,即适用于函数式编程场景的接口.而 Java 中的函数式编程体现就是 Lambda,所以函数式接口就是可以适用于 Lambda 使用的接口.只有确 ...

  9. Java8内置的函数式接口

    JDK 1.8 API 包含了很多内置的函数式接口.其中就包括我们在老版本中经常见到的 Comparator 和 Runnable,Java 8 为他们都添加了 @FunctionalInterfac ...

随机推荐

  1. Codeforces Round #575 (Div. 3) D2. RGB Substring (hard version)

    传送门 题意: 给你一个长为n的仅由'R','G','B'构成的字符串s,你需要在其中找出来一个子串.使得这个子串在"RGBRGBRGBRGB........(以RGB为循环节,我们称这个串 ...

  2. Codeforces Round #681 (Div. 2, based on VK Cup 2019-2020 - Final) B. Saving the City (贪心,模拟)

    题意:给你一个\(01\)串,需要将所有的\(1\)给炸掉,每次炸都可以将一整个\(1\)的联通块炸掉,每炸一次消耗\(a\),可以将\(0\)转化为\(1\),消耗\(b\),问将所有\(1\)都炸 ...

  3. 国产网络损伤仪SandStorm -- 主界面简介

    国产网络损伤仪SandStorm可以模拟出带宽限制.时延.时延抖动.丢包.乱序.重复报文.误码.拥塞等网络状况,在实验室条件下准确可靠地测试出网络应用在真实网络环境中的性能,以帮助应用程序在上线部署前 ...

  4. Scanner用户交互

    Scanner用户交互 Scanner对象 引入语法: Scanner scanner=new Scanner(System.in);(固定的) 小写scanner为定义的名称 scanner.clo ...

  5. Kubernets二进制安装(17)之安装部署Dashboard

    1.下载dashboard镜像 在运维主机(mfyxw50.mfyxw.com)上执行命令 [root@mfyxw50 ~]# docker pull registry.cn-hangzhou.ali ...

  6. μC/OS-III---I笔记7---消息队列

    消息队列 任务之间仅仅靠信号量进行"沟通"是不够的,信号量可以标志事件的发生,却无法传递更多的数据,在需要任务间的数据信息传递时就绪要用到消息队列,传统我们一般在前后太系统中都是通 ...

  7. 翻译:《实用的Python编程》01_03_Numbers

    目录 | 上一节 (1.2 第一个程序) | 下一节 (1.4 字符串) 1.3 数字 本节讨论数学计算. 数字类型 Python 有 4 种类型的数字: 布尔型 整型 浮点型 复数(虚数) 布尔型( ...

  8. Android 如何设置 WebView 的屏幕占比

    Android 如何设置 WebView 的屏幕占比 由于 Android 适用于具有各种屏幕尺寸和像素密度的设备,因此您在设计网页时应将这些因素纳入考虑范围,以便您的网页始终以合适的尺寸显示. We ...

  9. WebXR All in One

    WebXR All in One VR / WebVR WebXR https://www.w3.org/TR/webxr/ WebXR Device API https://immersiveweb ...

  10. css dark theme & js theme checker

    css dark theme & js theme checker live demo https://codepen.io/xgqfrms/pen/GRprYLm <!DOCTYPE ...