IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> 【硬核】源码 + 案例分享 JDK8 新特性中的四大函数式接口 -> 正文阅读

[Java知识库]【硬核】源码 + 案例分享 JDK8 新特性中的四大函数式接口

  • J3 - 白起
  • 技术(Lambda # 函数式接口)

交代一下这次分享技术的背景!

因为我是后端开发(会点前端),在工作中使用的开发语言就是 Java。然,技术的更新迭代速度太快了就拿 Java 来说都已经更新到 JDK17,而我相信大部分公司使用的还停留在 JDK8 这个阶段甚至 JDK7 的也不在少数。

而我现阶段开发的项目使用的版本是 JDK11 ,项目中有着大量的 JDK8 新特性特别是流式编程和 Lambda 表达式的应用。

而 Lambda 表达式要想写的好的话,那下面要谈的这四大函数式接口就是前提,所以废话不多说了,往下看吧,肯定能让你有所收获的。

在这里插入图片描述

一、Consumer:消费型接口

源码如下:

在这里插入图片描述

@FunctionalInterface 注解说明该接口是一个函数式接口(接口中只有一个抽象方法)。

参数 T 就是我们要处理的对象类型。

功能:实现 Consumer 接口中的 accept 方法,对类型 T 的对象进行任意操作(自己实现处理逻辑),所以这个接口也称之为消费性接口

使用

方式一

  1. 创建一个类,实现该接口

    class MyConsumer<T> implements Consumer<T> {
        @Override
        public void accept(T t) {
            // 自己的处理逻辑
            System.out.println("======开始做一下事情,对传入的 T======");
            System.out.println("正在对 T(" + t + ") 做事情...");
            System.out.println("======end======");
        }
    }
    
  2. 测试

    public class ConsumerTest {
        @Test
        public void test() {
            MyConsumer<String> stringMyConsumer = new MyConsumer<>();
            List<String> stringList = Arrays.asList("a", "b", "c", "d");
            // 遍历
            for (String s : stringList) {
                // 处理每一个元素
                stringMyConsumer.accept(s);
            }
        }
    }
    

这种方式完全没有利用到 JDK8 的特性,只是用了它提供的一个接口而已,那下面我们就换种方式。

方式二

直接一步到位:

public class ConsumerTest {
    @Test
    public void test2() {
        List<String> stringList = Arrays.asList("a", "b", "c", "d");
        // 利用 Lambda 表达式
        stringList.forEach(str -> {
            System.out.println("我可以对 str(" + str + ") 做任何事情");
        });
    }
}

如果上面不理解哪里用到了消费型接口,那我拆分一下

public class ConsumerTest {
    @Test
    public void test3() {
        List<String> stringList = Arrays.asList("a", "b", "c", "d");
        handle(stringList, s -> {
            // 内部类参数,实现处理逻辑
            System.out.println("我可以对 s(" + s + ") 做任何事情,而且我还知道他是什么类型");
        });
    }
    public <T> void handle(List<T> list, Consumer<T> consumer) {
        for (T t : list) {
            // 调用处理方法
            consumer.accept(t);
        }
    }
}

是不是非常清晰了!

其实我们在 JDK8 中经常使用的 forEach 遍历就是一种消费型接口的实现模式。

二、Supplier:供给型接口

源码:

在这里插入图片描述

功能:提供指定类型的数据出去,所以也称该接口为供给型接口。

使用

方式一

  1. 创建一个类,实现该接口

    @Data
    class MySupplier<T> implements Supplier<T>{
    
        private String name;
    
        @Override
        public T get() {
            // 我将提供一个指定 T 类型数据出去
            // 这里的 T 我先内定为 MySupplier 类型进行编写案例
            MySupplier<T> t = new MySupplier<>();
            t.setName("哈哈!");
            return (T) t;
        }
    }
    
  2. 测试

    public class SupplierTest {
        @Test
        public void test1(){
            MySupplier<MySupplier> stringMySupplier = new MySupplier<>();
            System.out.println(stringMySupplier.get());
        }
    }
    // SupplierTest.MySupplier(name=哈哈!)
    

还是老样子,这种方式也是没有利用 JDK8 特性,那我们改造一下

方式二

public class SupplierTest {
    @Test
    public void test2(){
        Supplier<String> stringSupp = () ->{
            // 你的逻辑
            return "调用get方法,我才给你数据";
        };
        System.out.println(stringSupp.get());
    }
}
// 调用get方法,我才给你数据

说一下我的理解,这个类型接口主要的就是提供数据,那么在项目中的话我可以事先定义好一组数据,在特定的时候通过事先定义好的对象去 get 就行。

场景:

用户购买商品,进入业务,判断用户购买的条件是否满足附送礼品的条件,如果满足,那么判断礼品种类(A,B,C等),然后根据 Supplier 类型接口实现类获取对应的礼品返回给用户购买的商品列表中。这样的好处就是用户不满足条件时是不会给项目创建礼品对象的,只有 get 方法调用,才会生成对象形成一个懒加载的效果。

三、Function:函数型接口

源码:

在这里插入图片描述

参数 R 就是该接口方法返回的类型。

功能:传入一个 T 类型对象,调用 apply 方法进行处理,最终于返回处理后的数据类型为 R

使用

方式一

  1. 编写一个类,实现该接口

    class MyFunction<T, R> implements Function<T, R> {
        @Override
        public R apply(T t) {
            System.out.println("处理逻辑....【" + t + "】");
            return null;
        }
    }
    
  2. 测试

    public class FunctionTest {
    
        @Test
        public void test1() {
            MyFunction<String, String> myFunction = new MyFunction<>();
            myFunction.apply("J3-白起...");
        }
    
    }
    // 处理逻辑....【J3-白起...】
    

老样子,我们用新特性改造一下。

方式二

public class FunctionTest {

    @Test
    public void test2() {
        Function<String, String> myFunction = (t) ->{
            System.out.println("处理逻辑....【" + t + "】");
            return "success";
        };
        myFunction.apply("J3-白起...");
    }

}
// 处理逻辑....【J3-白起...】

上面就是这个接口的简单使用,那我再来写个案例方便理解。

案例

需求:根据一个姓名列表,获取对应姓名长度的列表,实现代码如下:

public class FunctionTest {

    @Test
    public void test3() {
        List<String> nameList = Arrays.asList("J3-baiqi", "shaheshagn", "sunwukong");
        List<Integer> nameLengthList = functionHandle(nameList, String::length);
        System.out.println(nameLengthList);
        // [8, 10, 9]
    }

    /**
     * 获取列表中,姓名的长度
     *
     * @param nameList 名字列表
     * @param function 名字长度列表
     * @return
     */
    public List<Integer> functionHandle(List<String> nameList, Function<String, Integer> function) {
        List<Integer> nameLengthList = new ArrayList<>(nameList.size());
        nameList.forEach(name -> nameLengthList.add(function.apply(name)));
        return nameLengthList;
    }
}

讲解一下:

我分两个地方说明,一个是实参、一个是形参。

先说形参,对于该需求,我抽了一个方法出来,专门实现获取列表中元素的长度。功能很明确就是获取一个长度值,但具体是如何实现,我不管,我只要传入数据给你,你返回我需要的数字就行,所以我将形参定义为 Function 接口,在要获取数字的的时候,调用接口的 apply 方法即可。

对于实参,调用方法时,我需要传递 Function 接口类型对象进去,我可以写一个实现类,然后实现 apply 方法,接着通过各种新奇古怪的技术,获取传入的对象长度就行。但既然是 JDK8 那我就完全可以用新特性去实现(方法引用)。

场景:这个在项目中使用的场景是非常多的,我经常是需要抽取方法,然而方法上的形参我会定义成一个接口形成规范,因为我不管实现,我只管调用就行。以后有其他同事使用我的方法时,你自己去实现我形参中的接口就行,然后我的方法就会根据你实现的接口获取数据进而给你想要的结果。

四、Predicate:断定型接口

源码:

在这里插入图片描述

功能:传入指定类型对象,返回 true 或 false,所以也称该接口为断定型接口。

使用

方式一

  1. 编写一个类,并实现该接口

    class MyPredicate<T> implements Predicate<T> {
    
        @Override
        public boolean test(T t) {
            // 判断如果传入的类型 T 是 String 那么判断是否含有 "J3" 字样
            if (t instanceof String) {
                return ((String) t).contains("J3");
            }
            if (t instanceof Integer) {
                return ((Integer) t) == 18;
            }
            // 如果是 Integer 类型,判断是否等于 18
            return false;
        }
    }
    
  2. 测试

    public class PredicateTest {
        @Test
        public void test1(){
            MyPredicate<String> stringMyPredicate = new MyPredicate<>();
            System.out.println(stringMyPredicate.test("J3-白起"));
            System.out.println(stringMyPredicate.test("白起"));
            MyPredicate<Integer> integerMyPredicate = new MyPredicate<>();
            System.out.println(integerMyPredicate.test(18));
            System.out.println(integerMyPredicate.test(28));
        }
    }
    //true
    //false
    //true
    //false
    // 本人依旧是 J3 依旧是18,哈哈!
    

老样子,看一下下面改造的样子

方式二

public class PredicateTest {
    @Test
    public void test2() {
        Predicate<String> myStringPredicate = (t) -> t.contains("J3");
        System.out.println(myStringPredicate.test("J3-白起"));
        System.out.println(myStringPredicate.test("白起"));
        Predicate<Integer> myIntegerPredicate = (t) -> t == 18;
        System.out.println(myIntegerPredicate.test(18));
        System.out.println(myIntegerPredicate.test(28));
    }
}   

这个类型接口关键点在于返回的结果是一个 boolean 类型,那它的定位就非常清楚了,用于判断而已,具体你要判断什么,这就取决于你了。

需求案例:

现在我们有个系统是分 pc 和 app 端的,用户注册后信息进入业务中,就要区别出来是哪个端的用户注册。这就可以用到 Predicate 接口了,将用户信息传给 test,然后就可以编写判断逻辑根据对应的信息区别出是 pc 还是 app 了。

五、最后

最后,画一张表简单归纳一下上面分析的四个接口如下:

函数式接口参数类型返回类型用途
Consumer
消费型接口
Tvoid对类型为T的对象应用操作,包含方法:void accept(T t)
Supplier
供给型接口
T返回类型为T的对象,包含方法:T get()
Function<T, R>
函数型接口
TR对类型为T的对象应用操作,并返回结果。结果是R类型的对象。包含方法:R apply(T t)
Predicate
断定型接口
Tboolean确定类型为T的对象是否满足某约束,并返回boolean 值。包含方法:boolean test(T t)

好了,今天的内容到这里就结束了,关注我,我们下期见


  • 由于博主才疏学浅,难免会有纰漏,假如你发现了错误或偏见的地方,还望留言给我指出来,我会对其加以修正。

  • 如果你觉得文章还不错,你的转发、分享、点赞、留言就是对我最大的鼓励。

  • 感谢您的阅读,十分欢迎并感谢您的关注。

^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

CSDN:J3 - 白起

掘金:J3-白起

知乎:J3-白起

这是一个技术一般,但热衷于分享;经验尚浅,但脸皮够厚;明明年轻有颜值,但非要靠才华吃饭的程序员。

^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2021-10-26 12:05:05  更:2021-10-26 12:06:21 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 23:51:39-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码