一文搞懂Java8 Lambda表达式(附带视频教程)
Lambda表达式介绍
Java 8的一个大亮点是引入Lambda表达式,使用它设计的代码会更加简洁。通过Lambda表达式,可以替代我们以前经常写的匿名内部类来实现接口。Lambda表达式本质是一个匿名函数;
体验Lambda表达式
我们通过一个小例子来体验下Lambda表达式;
我们定义一个计算接口 只有一个方法add;
- public class Program {
- public static void main(String[] args) {
- Cal c1=new Cal() {
- @Override
- public int add(int a, int b) {
- return a+b;
- }
- };
- int c=c1.add(,);
- System.out.println(c);
- }
- }
- interface Cal{
- int add(int a,int b);
- }
这个是我们以前的实现,匿名内部类,然后调用执行;
我们现在用Lambda表达式改写下:
- public class Program {
- public static void main(String[] args) {
- Cal c1=(int a,int b) ->{return a+b;};
- int c=c1.add(,);
- System.out.println(c);
- }
- int add(int a,int b){
- return a+b;
- }
- }
- interface Cal{
- int add(int a,int b);
- }
匿名内部类,直接改成了:
- Cal c1=(int a,int b) ->{return a+b;};
简洁多了;
是不是感觉Lambda表达式挺强大,
接下来我们来看看Lambda表达式的语法吧;
Lambda表达式语法
我们看下这个Lambda表达式:
(int a,int b) ->{return a+b;};
这个本质是一个函数;
一般的函数类似如下:
- int add(int a,int b){
- return a+b;
- }
有返回值,方法名,参数列表,方法体
Lambda表达式函数的话,只有参数列表,和方法体;
( 参数列表 ) -> { 方法体 }
说明:
( ) :用来描述参数列表;
{ } : 用来描述方法体;
-> :Lambda运算符,可以叫做箭头符号,或者goes to
Lambda表达式语法细讲
我们搞一个案例,接口方法参数,无参,单个参数,两个参数,有返回值,没有返回值,这六种情况都罗列下:
- interface If1{
- /**
- * 无参数无返回值
- */
- void test();
- }
- interface If2{
- /**
- * 单个参数无返回值
- * @param a
- */
- void test(int a);
- }
- interface If3{
- /**
- * 两个参数无返回值
- * @param a
- * @param b
- */
- void test(int a,int b);
- }
- interface If4{
- /**
- * 无参数有返回值
- * @return
- */
- int test();
- }
- interface If5{
- /**
- * 单个参数有返回值
- * @param a
- * @return
- */
- int test(int a);
- }
- interface If6{
- /**
- * 多个参数有返回值
- * @param a
- * @param b
- * @return
- */
- int test(int a,int b);
- }
我们用Lambda表达式实现:
- // 无参数无返回值
- If1 if1=()->{
- System.out.println("无参数无返回值");
- };
- if1.test();
- // 单个参数无返回值
- If2 if2=(int a)->{
- System.out.println("单个参数无返回值 a="+a);
- };
- if2.test();
- // 两个参数无返回值
- If3 if3=(int a,int b)->{
- System.out.println("两个参数无返回值 a+b="+(a+b));
- };
- if3.test(,);
- // 无参数有返回值
- If4 if4=()->{
- System.out.print("无参数有返回值 ");
- return ;
- };
- System.out.println(if4.test());
- // 单个参数有返回值
- If5 if5=(int a)->{
- System.out.print("单个参数有返回值 ");
- return a;
- };
- System.out.println(if5.test());
- // 多个参数有返回值
- If6 if6=(int a,int b)->{
- System.out.print("多个参数有返回值 ");
- return a+b;
- };
- System.out.println(if6.test(,));
运行输出:
- 无参数无返回值
- 单个参数无返回值 a=
- 两个参数无返回值 a+b=
- 无参数有返回值
- 单个参数有返回值
- 多个参数有返回值
Lambda表达式精简语法
那件语法注意点:
- 参数类型可以省略
- 假如只有一个参数,()括号可以省略
- 如果方法体只有一条语句,{}大括号可以省略
- 如果方法体中唯一的语句是return返回语句,那省略大括号的同事return也要省略
改写实例:
- /**
- * @author java1234_小锋
- * @site www.java1234.com
- * @company Java知识分享网
- * @create 2020-08-12 16:43
- */
- public class Program2 {
- public static void main(String[] args) {
- // 1,参数类型可以省略
- // 2,假如只有一个参数,()括号可以省略
- // 3,如果方法体只有一条语句,{}大括号可以省略
- // 4,如果方法体中唯一的语句是return返回语句,那省略大括号的同事return也要省略
- // 无参数无返回值
- If1 if1=()->System.out.println("无参数无返回值");
- if1.test();
- // 单个参数无返回值
- If2 if2=a->System.out.println("单个参数无返回值 a="+a);
- if2.test();
- // 两个参数无返回值
- If3 if3=(a,b)->{
- System.out.println("两个参数无返回值 a+b="+(a+b));
- };
- if3.test(,);
- // 无参数有返回值
- If4 if4=()->;
- System.out.println(if4.test());
- // 单个参数有返回值
- If5 if5=a->{
- System.out.print("单个参数有返回值 ");
- return a;
- };
- System.out.println(if5.test());
- // 多个参数有返回值 参数类型可以省略
- If6 if6=(a,b)->a+b;
- System.out.println(if6.test(,));
- }
- }
方法引用
有时候多个lambda表达式实现函数是一样的话,我们可以封装成通用方法,以便于维护;
这时候可以用方法引用实现:
语法是:对象::方法
假如是static方法,可以直接 类名::方法
实例如下:
- public class Program2 {
- public static void main(String[] args) {
- // 方法引用
- // 语法:
- // static方法 类名::方法名
- // 普通方法 对象名::方法名
- Program2 program2=new Program2();
- If5 if5=program2::test;
- If5 if52=Program2::test2;
- System.out.println(if5.test());
- System.out.println(if52.test());
- }
- public int test(int a){
- return a-;
- }
- public static int test2(int a){
- return a-;
- }
- }
构造方法引用
如果函数式接口的实现恰好可以通过调用一个类的构造方法来实现,
那么就可以使用构造方法引用;
语法:类名::new
实例:
先定义一个Dog实体,实现无参和有参构造方法;
- public class Dog {
- private String name;
- private int age;
- public Dog() {
- System.out.println("无参构造方法");
- }
- public Dog(String name, int age) {
- System.out.println("有参构造方法");
- this.name = name;
- this.age = age;
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public int getAge() {
- return age;
- }
- public void setAge(int age) {
- this.age = age;
- }
- @Override
- public String toString() {
- return "Dog{" +
- "name='" + name + '\'' +
- ", age=" + age +
- '}';
- }
- }
在定义两个接口:
- interface DogService{
- Dog getDog();
- }
- interface DogService2{
- Dog getDog(String name,int age);
- }
测试:
- public class Program3 {
- public static void main(String[] args) {
- // 普通方式
- DogService dogService=()->{
- return new Dog();
- };
- dogService.getDog();
- // 简化方式
- DogService dogService2=()->new Dog();
- dogService2.getDog();
- // 构造方法引用
- DogService dogService3=Dog::new;
- dogService3.getDog();
- // 构造方法引用 有参
- DogService2 dogService21=Dog::new;
- dogService21.getDog("小米",);
- }
- }
执行结果:
- 无参构造方法
- 无参构造方法
- 无参构造方法
- 有参构造方法
综合实例
下面我们通过一个lambda操作集合的综合实例,来深入体验下Lambda表达式用法;
- public class Program4 {
- public static void main(String[] args) {
- List<Dog> list=new ArrayList<>();
- list.add(new Dog("aa",));
- list.add(new Dog("bb",));
- list.add(new Dog("cc",));
- list.add(new Dog("dd",));
- list.add(new Dog("ee",));
- // 排序
- System.out.println("lambda集合排序");
- list.sort((o1,o2)->o1.getAge()-o2.getAge());
- System.out.println(list);
- // 遍历集合
- System.out.println("lambda遍历集合");
- list.forEach(System.out::println);
- }
- }
运行输出:
- lambda集合排序
- [Dog{name='aa', age=}, Dog{name='dd', age=}, Dog{name='cc', age=}, Dog{name='bb', age=}, Dog{name='ee', age=}]
- lambda遍历集合
- Dog{name='aa', age=}
- Dog{name='dd', age=}
- Dog{name='cc', age=}
- Dog{name='bb', age=}
- Dog{name='ee', age=}
我们来分析下集合的sort方法,
sort方法里有一个Comparator接口,再点进去看下:
我们通过lambda就可以轻松实现排序:
- (o1,o2)->o1.getAge()-o2.getAge()
再看下集合的forEach方法,点进去:
有个消费者Consumer接口,再点进去:
接口里有个接口参数的accept的方法;
所以我们直接方法引用 直接输出每次的遍历值即可;
- System.out::println
@FunctionalInterface注解
前面我们会发现Consumer接口,Comparator接口都有
@FunctionalInterface注解;
这个注解是函数式接口注解,所谓的函数式接口,当然首先是一个接口,然后就是在这个接口里面只能有一个抽象方法。
这种类型的接口也称为SAM接口,即Single Abstract Method interfaces
特点
接口有且仅有一个抽象方法
允许定义静态方法
允许定义默认方法
允许java.lang.Object中的public方法
该注解不是必须的,如果一个接口符合"函数式接口"定义,那么加不加该注解都没有影响。加上该注解能够更好地让编译器进行检查。如果编写的不是函数式接口,但是加上了@FunctionInterface,那么编译器会报错
实例
- // 正确的函数式接口
- @FunctionalInterface
- public interface TestInterface {
- // 抽象方法
- public void sub();
- // java.lang.Object中的public方法
- public boolean equals(Object var1);
- // 默认方法
- public default void defaultMethod(){
- }
- // 静态方法
- public static void staticMethod(){
- }
- }
- // 错误的函数式接口(有多个抽象方法)
- @FunctionalInterface
- public interface TestInterface2 {
- void add();
- void sub();
- }
系统内置函数式接口
Java8的推出,是以Lambda重要特性,一起推出的,其中系统内置了一系列函数式接口;
再jdk的java.util.function包下,有一系列的内置函数式接口:
比如常用的Consumer,Comparator,Predicate,Supplier等;
Lambda表达式视频教程
感谢各位兄弟姐妹关注,锋哥为了大伙能更深刻的掌握Lambda的原理和应用,专门录制了一期视频教程。主要以IDEA开发工具,来讲解lambda表达式,希望小伙伴们快速的掌握。
纸上得来终觉浅,绝知此事要躬行。
需要多实战,多思考
B站视频教程在线地址:
https://www.bilibili.com/video/bv1ci4y1g7qD
------------------------------------------------------------------------------------------------------------------------------
作者:java1234_小锋
出处:https://www.cnblogs.com/java688/p/13511720.html
版权:本站使用「CC BY 4.0」创作共享协议,转载请在文章明显位置注明作者及出处。
------------------------------------------------------------------------------------------------------------------------------
一文搞懂Java8 Lambda表达式(附带视频教程)的更多相关文章
- Java8 Lambda表达式详解手册及实例
先贩卖一下焦虑,Java8发于2014年3月18日,距离现在已经快6年了,如果你对Java8的新特性还没有应用,甚至还一无所知,那你真得关注公众号"程序新视界",好好系列的学习一下 ...
- java8 Lambda表达式的新手上车指南(1)
背景 java9的一再推迟发布,似乎让我们恍然想起离发布java8已经过去了三年之久,java8应该算的上java语言在历代版本中变化最大的一个版本了,最大的新特性应该算得上是增加了lambda表达式 ...
- 一文搞懂RAM、ROM、SDRAM、DRAM、DDR、flash等存储介质
一文搞懂RAM.ROM.SDRAM.DRAM.DDR.flash等存储介质 存储介质基本分类:ROM和RAM RAM:随机访问存储器(Random Access Memory),易失性.是与CPU直接 ...
- java8 Lambda表达式的新手上车指南(1)--基础语法和函数式接口
背景 java9的一再推迟发布,似乎让我们恍然想起离发布java8已经过去了三年之久,java8应该算的上java语言在历代版本中变化最大的一个版本了,最大的新特性应该算得上是增加了lambda表达式 ...
- 基础篇|一文搞懂RNN(循环神经网络)
基础篇|一文搞懂RNN(循环神经网络) https://mp.weixin.qq.com/s/va1gmavl2ZESgnM7biORQg 神经网络基础 神经网络可以当做是能够拟合任意函数的黑盒子,只 ...
- 一文搞懂 Prometheus 的直方图
原文链接:一文搞懂 Prometheus 的直方图 Prometheus 中提供了四种指标类型(参考:Prometheus 的指标类型),其中直方图(Histogram)和摘要(Summary)是最复 ...
- Web端即时通讯基础知识补课:一文搞懂跨域的所有问题!
本文原作者: Wizey,作者博客:http://wenshixin.gitee.io,即时通讯网收录时有改动,感谢原作者的无私分享. 1.引言 典型的Web端即时通讯技术应用场景,主要有以下两种形式 ...
- 一文搞懂vim复制粘贴
转载自本人独立博客https://liushiming.cn/2020/01/18/copy-and-paste-in-vim/ 概述 复制粘贴是文本编辑最常用的功能,但是在vim中复制粘贴还是有点麻 ...
- 三文搞懂学会Docker容器技术(中)
接着上面一篇:三文搞懂学会Docker容器技术(上) 6,Docker容器 6.1 创建并启动容器 docker run [OPTIONS] IMAGE [COMMAND] [ARG...] --na ...
随机推荐
- 高效C++:继承和实现
如何正确的使用继承和实现是本章说明的重点. 确定public继承的关系是is-a public继承等同于is-a 对public继承,所有base的特性,在derived上都适用 避免遮掩继承而来的名 ...
- presto和hive日期函数对比
时间格式转换 日期格式→Unix时间戳 转10位Unix时间戳 数据:2020-07-23 15:01:13 Presto:select to_unixtime(cast('2020-07-23 15 ...
- 谁能告诉我如何通过Jenkins完成分布式环境搭建并执行自动化脚本
今天我们接着昨天的内容,看一看如何完成Jenkins分布式环境的搭建和使用,因为我之前也是自己一个人摸索的,如果有不对的地方,请各位看官私信指出. 新增分布式部署节点 在系统管理/节点管理中点击新建 ...
- Windows下给PHP安装redis扩展
一.选择适合的版本 二.下载扩展 官网下载地址:http://pecl.php.net/package/redis ,选择合适的版本进行下载 三.解压后将下面两个文件复制到PHP扩展文件目录(ext文 ...
- vue同时安装element ui跟 vant
记一个卡了我比较久的问题,之前弄的心态爆炸各种问题. 现在来记录一下,首先我vant是已经安装成功了的. 然后引入element ui npm i element-ui -S 接着按需引入,安装插件 ...
- pandas巩固
导包 import pandas as pd 设置输出结果列对齐 pd.set_option('display.unicode.ambiguous_as_wide',True) pd.set_opti ...
- Python unichr() 函数
描述 unichr() 函数 和 chr() 函数功能基本一样, 只不过是返回 unicode 的字符.高佣联盟 www.cgewang.com 注意: Python3 不支持 unichr(),改用 ...
- PHP array_diff() 函数
实例 比较两个数组的值,并返回差集: <?php $a1=array("a"=>"red","b"=>"gree ...
- PHP pow() 函数
实例 pow() 的实例: <?phpecho(pow(2,4) . "<br>");echo(pow(-2,4) . "<br>" ...
- C/C++编程笔记:C++入门知识丨函数和函数模板
本篇要学习的内容和知识结构概览 函数的参数及其传递方式 1. 函数参数传递方式 传值: 传变量值: 将实参内存中的内容拷贝一份给形参, 两者是不同的两块内存 传地址值: 将实参所对应的内存空间的地址值 ...