Java8新特性 ----- Lambda表达式和方法引用/构造器引用详解

news2024/9/17 7:43:27

前言

在讲一下内容之前,我们需要引入函数式接口的概念

什么是函数式接口呢?

函数式接口:有且仅有一个抽象方法的接口

java中函数式编程的体现就是Lambda表达式,你可以认为函数式接口就是适用于Lambda表达式的接口.

也可以加上注解来在编译层次上限制函数式接口

  • @Functionallnterface
  • 放在 接口定义的上方:如果接口是函数式接口,编译通过;如果不是,编译失败
  • 注:自定义的函数式接口可以加上这个表示是函数式接口,不加也可以,但是建议添加

常见的函数式接口有如下四种

接口                                 抽象方法
Consumer<T> 消费型接口           void accept(T t)
Supplier<T> 供给型接口           T get()
Function<T>函数型                R apply(T t)
Predicted<T>判断型接口           boolean test(T t)

以下对Lambda表达式的说明都是基于函数式接口的

为什么需要Lambda表达式?

本身我们最原始的方法定义类来实现接口,在用类来实例化对象调用方法显得冗余且重,我们简化到匿名内部类的时候也显得冗余,主要目的是为了简化代码,并提供更加简洁和灵活的函数式编程方式。也与后面要说的stream api有关,这里不做过多赘述.

1.Lambda表达式

Lambda表达式的本质其实是一个接口实现类的对象,也是一个匿名函数.

下面我们就谈谈几种lambda表达式的应用场景

1.实现Runnable接口,注意此处不涉及到线程!!

Lambda表达式的思想就是能省略的就省略,不产生歧义即可

于是就有了这样的写法

 @Test
    public void test1()
    {
        //语法格式1:无参数,无返回值
        Runnable r1 = new Runnable() {
            @Override
            public void run() {
                System.out.println("我爱北京天安门");
            }
        };
        r1.run();
        System.out.println("*******************");
        Runnable r2 = ()->{ System.out.println("我爱北京天安门");};
        r2.run();
    }

2.当lanbda表达式中的参数类型确定时,参数类型也可以省略

 @Test
    public void test3()
    {
        //数据类型可以省略,因为可以由编译器进行推断
        Consumer<String> con1 = (String s)->{
            System.out.println(s);
        };
        con1.accept("如果大学可以重来,你最想重来的事是啥?");
        System.out.println("****************");

        Consumer<String> con2 = (s)->{
            System.out.println(s);
        };
        con2.accept("如果大学可以重来,你最想重来的事是啥?");

    }

3.只有一个参数的时候,小括号可以省略

 @Test
    public void test5()
    {
        //当只有一个参数的时候,参数的小括号可以省略
        Consumer<String> con1 = s->{System.out.println(s);};
        con1.accept("世界那么大,我想去看看");
    }

4.只有一条语句时,return语句和大括号可以省略(必须一起省略)

         Comparator<Integer> com1 = (o1,o2) ->{
            return o1.compareTo(o2);
        };
        System.out.println(com1.compare(12,6));
        System.out.println("***********************");


        Comparator<Integer> com2 = (o1,o2) -> o1.compareTo(o2);

总结:

格式

->:箭头操作符
->的左边:Lambda形参列表:对应着要重写的接口中要重写的形参列表
->的右边:Lambda体,对应着接口的实现类要重写的方法体
Lambda形参列表 ->Lambda体

细节注意

->的左边 :lambda 形参列表 :参数类型可以省略,如果形参列表只有一个,小括号也可以省略
->的右边:lambda体: 对应着重写方法的方法体,如果方法体中只有一条执行语句,则大括号可以省略,有return关键字,则return需要一并省略

2.方法引用

可以看做是Lambda表达式的进一步延伸

使用说明

情况1: 对象 :: 实例方法(非静态方法)
要求:函数式接口的抽象方法a与其内部实现时调用的某个方法b的形参列表和返回值类型都相同(或一致),
我们就可以考虑用方法b对方法a进行替换,此替换或覆盖称为方法引用
注:此时b是非静态的方法,需要对象来调用

情况2: 类 :: 静态方法
要求:函数式接口的抽象方法a与其内部实现时调用的某个方法b的形参列表和返回值类型都相同,
我们就可以考虑用b对方法a进行替换,此替换或覆盖称为方法引用
注:此时b是静态的方法,需要类来调用


情况3: 类 :: 实例方法

要求:函数式接口的抽象方法a与其内部实现时时调用的对象的某个方法b的返回值类型相同
同时,抽象方法a中有n个参数,方法b中有n-1个参数,且抽象方法a的第一个参数作为方法b的调用者,且抽象方法a
的后n-1个参数与方法b的n-1个参数类型相同或一致,则可以使用方法引用

注意:此方法b是静态方法,需要对象调用,但是形式上,写出a所属的类.

举例说明

1.对象::方法类型

此时我们发现get方法是空参方法,返回值是String,emp.getName()方法返回值是String形参也为空,这样就可以用这个实现的方法来覆盖原有的get方法,于是可以写作

emp::getName()

注意:这里的相同可以理解为满足多态即可.

 @Test
    public void test2()
    {
        //供给型 Supplier中的T() get
        //Employee 中的String getName()
        Employee emp = new Employee(1001,"马化腾",34,6000.38);

        Supplier<String> sup1 = new Supplier<String>() {
            @Override
            public String get() {
                return emp.getName();
            }
        };
        System.out.println(sup1.get());

        //Lambda表达式写法
        Supplier<String> sup2 = ()-> emp.getName();
        System.out.println(sup2.get());


        //3.方法引用
        Supplier<String> sup3 = emp :: getName;
        System.out.println(sup3.get());
    }

2.类::静态方法举例

这里我们发现compare方法和实现中的Integer的compare方法参数和返回值一直,就可以使用方法引用,只不过这里的compare方法是静态方法,要使用类来调用,看做对原本抽象方法的一个覆盖,写作Integer :: compare;

@Test
    public void test3()
    {
        //类 :: 静态对象
        Comparator<Integer> com1 = new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return Integer.compare(o1,o2);
            }

        };

        Comparator<Integer> com2 = (o1,o2) ->Integer.compare(o1,o2);
        System.out.println(com2.compare(12,21));
    }

    Comparator<Integer> com3 = Integer :: compare;

3.类 :: 实例方法

这个的理解就想对困难一点点,本质和之前一样

这里我么假设抽象方法的形参有n个,实现的语句是形参1为调用者的语句

这里就可以把形参1抽象为其对应的类,剩余的返回值和形参都与原抽象方法一致

这样就可以用这种形式的方法引用代替

@Test
    public void test5()
    {
        //情况3 类 :: 实例方法(难)
        Comparator<String> com1 = new Comparator<String>() {
            @Override
            public int compare(String o1, String o2) {
                return o1.compareTo(o2);
            }
        };


        //2.Lambda表达式
        Comparator<String> com2 = (o1,o2) ->o1.compareTo(o2);

        //满足参数是 重写的函数的参数是需要调用的函数的参数的n+1个,也可以使用方法引用的方式
        Comparator<String> com3 = String :: compareTo;
    }

3.构造器引用/数组引用

实际上是对方法引用的一种特殊操作

就是抽象方法里面返回一个构造器,如果把构造器看做一个方法,其实本质上就一样了

 Supplier<Employee> sup1 = new Supplier<Employee>() {
            @Override
            public Employee get() {
                return new Employee();
            }
        };

Supplier<Employee> sup2 = Employee :: new;

也可以是多参数的构造器引用,因为前面的泛型参数可以直接推断你的构造器类型

public void test2()
    {
        Function<Integer,Employee> func1 = new Function<Integer, Employee>() {
            @Override
            public Employee apply(Integer id) {
                return new Employee(id);
            }
        };
        System.out.println(func1.apply(12));

        //构造器引用
        Function<Integer,Employee> func2 = Employee :: new;
        //调用的是Employee类中参数是Integer类型的构造器
        func2.apply(11);
    }

三个,四个也是可以的

数组引用

和上面类似,不做过多解释

 Function<Integer,Employee[]> func1 = new Function<Integer, Employee[]>() {
            @Override
            public Employee[] apply(Integer length) {
                return new Employee[length];
            }
        };
        Function<Integer,Employee[]> func2 = Employee[] :: new;

总结 

Lambda表达式可以对函数式接口的实现代码进行精简,(满足一定条件)进一步引出了方法引用/构造器引用/数组引用等...

秋秋语录:今日事今日毕,语法基础一定要打扎实,大处着眼,小处着手,多看多练.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1238779.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

关于2023年11月25日PMI认证考试有关事项的通知

PMP项目管理学习专栏https://blog.csdn.net/xmws_it/category_10954848.html?spm1001.2014.3001.54822023年8月PMP考试成绩出炉|微思通过率95%以上-CSDN博客文章浏览阅读135次。国际注册项目管理师(PMP) 证书是项目管理领域含金量最高的职业资格证书&#xff0c;获得该资质是项…

美团四年、字节三年,我的软件测试之路

前言 时间回到8年前&#xff0c;我人生中的第一份实习工作&#xff0c;是在某互联网公司做一个自动化测试工程师。当时的我可谓意气风发&#xff0c;想要大干一场&#xff0c;结果第一次做测试就出现了事故。由于对某些地方的不了解&#xff0c;把某一个地方侧漏了&#xff0c…

基于C#实现树状数组

有一种数据结构是神奇的&#xff0c;神秘的&#xff0c;它展现了位运算与数组结合的神奇魅力&#xff0c;太牛逼的&#xff0c;它就是树状数组&#xff0c;这种数据结构不是神人是发现不了的。 一、概序 假如我现在有个需求&#xff0c;就是要频繁的求数组的前 n 项和&#x…

2018年全国硕士研究生入学统一考试管理类专业学位联考数学试题——解析版

文章目录 2018 年考研管理类联考数学真题一、问题求解&#xff08;本大题共 5 小题&#xff0c;每小题 3 分&#xff0c;共 45 分&#xff09;下列每题给出 5 个选项中&#xff0c;只有一个是符合要求的&#xff0c;请在答题卡上将所选择的字母涂黑。真题&#xff08;2018-01&a…

JavaScript实现右键菜单

1、代码实现 window.onload function () {(function () {// 自定义右键菜单内容并插入到body最后一个节点前let dom <div id"rightMenuBars"><div class"rightMenu-group rightMenu-small"><div class"rightMenu-item"><…

C#开发的OpenRA游戏之属性RenderSprites(8)

C#开发的OpenRA游戏之属性RenderSprites(8) 本文开始学习RenderSprites属性,这个属性是跟渲染有关的,因此它就摄及颜色相关的内容,所以我们先来学习一下调色板,这是旧游戏的图片文件保存的格式,如果放在现代来看,不会再采用这种方法,毕竟现在存储空间变大,便宜了,并…

RubbleDB: CPU-Efficient Replication with NVMe-oF

RubbleDB: CPU-Efficient Replication with NVMe-oF 前言 这是ATC2023的文章&#xff0c;作者来自哥伦比亚大学这篇工作在LSM-tree多副本存储的场景下&#xff0c;利用NVMe-oF技术避免了LSM-tree副本上的重复合并&#xff0c;减少了CPU开销。 Introduction 为了提供高可用性…

练习九-利用状态机实现比较复杂的接口设计

练习九-利用状态机实现比较复杂的接口设计 1&#xff0c;任务目的&#xff1a;2&#xff0c;RTL代码3&#xff0c;RTL原理框图4&#xff0c;测试代码5&#xff0c;波形输出 1&#xff0c;任务目的&#xff1a; &#xff08;1&#xff09;学习运用状态机控制的逻辑开关&#xff…

vue history路径编码

记录今天遇到的一个问题&#xff1a; 问题现状 有一个需要前端伪造302进行重定向的需求&#xff0c;我们需要将这样的一个路径&#xff1a;http://xxx.com/system-name/#/index&#xff0c;拼接在跳转地址的后面&#xff0c;进行重定向。拼接的方式是这样的&#xff1a; htt…

暗物质探测器认知教学VR元宇宙平台打破传统束缚

“飞船正在上升&#xff0c;马上就冲出大气层了!”这是一位在1&#xff1a;1还原的神舟飞船返回舱内借助VR设备置身元宇宙世界&#xff0c;沉浸式体验升空全过程的游客兴奋地说道。不仅如此&#xff0c;在载人飞船训练期&#xff0c;元宇宙技术为航天员虚拟一个逼真的太空世界&…

口碑好的猫罐头有哪些?宠物店受欢迎的5款猫罐头推荐!

快到双十二啦&#xff01;铲屎官们是时候给家里猫主子囤猫罐头了。许多铲屎官看大促的各种品牌宣传&#xff0c;看到眼花缭乱&#xff0c;不知道选哪些猫罐头好&#xff0c;胡乱选又怕踩坑。 口碑好的猫罐头有哪些&#xff1f;作为一个经营宠物店7年的老板&#xff0c;活动期间…

Windows平台如何实现RTSP流二次编码并添加动态水印后推送RTMP或轻量级RTSP服务

技术背景 我们在对接RTSP播放器相关的技术诉求的时候&#xff0c;遇到这样的需求&#xff0c;客户做特种设备巡检的&#xff0c;需要把摄像头拍到的RTSP流拉下来&#xff0c;然后添加动态水印后&#xff0c;再生成新的RTSP URL&#xff0c;供平台调用。真个流程需要延迟尽可能…

LL(1)语法分析程序设计与实现

制作一个简单的C语言词法分析程序_用c语言编写词法分析程序-CSDN博客文章浏览阅读322次。C语言的程序中&#xff0c;有很单词多符号和保留字。一些单词符号还有对应的左线性文法。所以我们需要先做出一个单词字符表&#xff0c;给出对应的识别码&#xff0c;然后跟据对应的表格…

FlinkCDC实现主数据与各业务系统数据的一致性(瀚高、TIDB)

文章末尾附有flinkcdc对应瀚高数据库flink-cdc-connector代码下载地址 1、业务需求 目前项目有主数据系统和N个业务系统,为保障“一数一源”,各业务系统表涉及到主数据系统的字段都需用主数据系统表中的字段进行实时覆盖,这里以某个业务系统的一张表举例说明:业务系统表Ta…

社群乱象,社群玩法正解

社群乱象&#xff0c;社群玩法正解 越来越不喜欢混社群了&#xff0c;从原有的好几百社群&#xff0c;精简到剩两三个。就这两三个&#xff0c;也貌似奄奄一息&#xff0c;命不久矣的感觉。 现在的社群&#xff0c;妥妥的已经被各路妖魔鬼怪给玩坏了。童话觉得这里面还有不少…

vue3 iconify 图标几种使用 并加载本地 svg 图标

iconify iconify 与 iconify/vue 使用 下载 pnpm add iconify/vue -D使用 import { Icon } from "iconify/vue";<template><Icon icon"mdi-light:home" style"color: red; font-size: 43px" /><Icon icon"mdi:home-flo…

Spring Boot 邮件发送(五种类型的邮件)

邮件协议&#xff1a; SMTP、POP3、IMAP SMTP 协议全称为 Simple Mail Transfer Protocol&#xff0c;译作简单邮件传输协议&#xff0c;它定义了邮件客户端软件与 SMTP 服务器之间&#xff0c;以及 SMTP 服务器与 SMTP 服务器之间的通信规则。 用户先将邮件投递到腾讯的 SMT…

.nvmrc 文件使用详解

文章目录 1. 前言2. .nvmrc 是什么3. 创建 .nvmrc 文件4. 使用 .nvmrc 文件5. 终端自动切换版本 1. 前言 当开发多个项目时&#xff0c;每个项目运行环境要求的 node 版本不一样&#xff0c;那么我们就需要给每个项目指定 node 版本&#xff0c;也就是通过终端执行 nvm install…

虚拟摇杆OnJoystickMove未被调用,角色不移动

更改interaction type 为 event notification