JDK 8的新特性-Lambda表达式 精品文章总结
文章目录
- 一. 前言
- 1.2 认识Lambda表达式
- 二. Lambda 表达式的格式
- 2.1 语法格式一: 无参数,无返回值,Lambda体只有一条语句
- 2.2 语法格式二: 有一个参数,并且无返回值
- 2.3 语法格式三: 有两个以上的参数,并且有返回值,并且Lambda体有多条语句
- 2.4 语法格式四:若Lambda体中只有一条语句,`return`和大括号`{}`都可以省略
- 2.5 语法格式五:Lambda表达式的参数列表数据类型可以省略不写,因为JVM编译器通过上下文推断出,数据类型,即:"类型推断"
- 三. 函数式接口
- 3.1 什么是函数式接口?
- 3.2 自定义的函数式接口
- 3.3 Java内置的四大核心函数式接口
- 四. 方法引用与构造器的引用
- 4.1 方法引用
- 4.2 构造器引用
- 4.3 数组引用
一. 前言
JDK8已经发布快4年的时间了,现在来谈它的新特性显得略微的有点“不合时宜”。尽管JDK8已不再“新”,但它的重要特性之一——Lambda
表达式依然是不被大部分开发者所熟练运用,甚至不被开发者所熟知。
国内的开发环境大家都知道,有各种的老项目,有各种各样的发布风险,让公司以及项目组对新的技术往往望而却步,有公司甚至时至今日还在使用JDK6来进行项目开发,这导致了在很多技术的选择上受到了很大限制,进而不能跟随时代的脚步使得项目甚至公司一步一步走向衰落。
本文简单认识JDK8的重要新特性之一——Lambda表达式。 在JDK8之前,Java是不支持函数式编程的,所谓的函数编程,即可理解是将一个函数(也称为“行为”)作为一个参数进行传递。通常我们提及得更多的是面向对象编程,面向对象编程是对数据的抽象(各种各样的POJO类),而函数式编程则是对行为的抽象(将行为作为一个参数进行传递)。在JavaScript中这是很常见的一个语法特性,但在Java中将一个函数作为参数传递这却行不通,好在JDK8的出现打破了Java的这一限制。
1.2 认识Lambda表达式
@Testpublic void test01(){// JDK 1.8之前用法Runnable r1 = new Runnable() {@Overridepublic void run() {System.out.println("JDK 8之前用法");}};r1.run();// Lambda表达式用法Runnable r2 = ()->System.out.println("JDK 8 用法--Lambda表达式用法");r2.run();}
说明:
- 在这个例子中,传统的语法规则,我们是讲一个匿名内部内作为参数进行传递,我们实现了Runnable接口,并且将其作为参数传递给Thread类. 实际上我们传递的是一段代码, 即 我们将代码作为数据进行传递,这就带来了需要不必要的"样板代码"
- Lambda表达式一共分为三个部分:
- 左边: 代表参数列表
- 右边:表示Lambda体
- ‘->’ 箭头操作符
二. Lambda 表达式的格式
2.1 语法格式一: 无参数,无返回值,Lambda体只有一条语句
()->System.out.println(“hello Lambda!”);
/*** 语法格式一: 无参数,无返回值*/@Testpublic void test01(){int num = 0 ; // JDK 1.7 前, 必须是finalRunnable r = new Runnable() {@Overridepublic void run() {System.out.println("Hello world! "+ num);}};r.run();System.out.println("-----------------------");Runnable r1 = ()->System.out.println("hello World");System.out.println(r1);}
2.2 语法格式二: 有一个参数,并且无返回值
(x)->System.out.println(x);
只有一个参数时,小括号"()" 可以省略
/*** 语法格式二: 有一个参数,无返回值*/@Testpublic void test02(){Consumer<String> con = (x)->System.out.println(x);con.accept("hello Consumer");System.out.println("-----------------------");Consumer<String> con1 = x->System.out.println(x);con1.accept("hello Consumer");}
2.3 语法格式三: 有两个以上的参数,并且有返回值,并且Lambda体有多条语句
- 两个以上的参数,左侧的"小括号
()
" 不能省略- Lambda 体有多条语句是,
{}
不能省略- Lambda 体存在多条语句是
return
不能省略
/*** 语法格式四: 有两个以上的参数,有返回值,并且Lambda体有多条语句时* 大括号"{}" 不可以省略,并且参数的小括号"()"也不能省略*/@Testpublic void test04(){Comparator<Integer> com = (x,y)->{System.out.println("函数式接口");return x.compareTo(y);};int compare = com.compare(3, 4);System.out.println(compare);}
2.4 语法格式四:若Lambda体中只有一条语句,return
和大括号{}
都可以省略
2.5 语法格式五:Lambda表达式的参数列表数据类型可以省略不写,因为JVM编译器通过上下文推断出,数据类型,即:“类型推断”
(Integer x, Integer y) -> Integer.compare(x, y);
总结:
上联:左右遇一括号省
下联:左侧推断类型省
横批:能省则省
三. 函数式接口
3.1 什么是函数式接口?
- 只包含一个包含一个抽象方法的接口,称为函数式接口
- 可以通过Lambda表达式来创建爱你改接口的对象.(若Lambda表达式抛出一个受检异常,那么该异常需要在目标接口的抽象方法上声明).
- 可以在任意函数式接口上使用
@FunctionIntrerace
注解,这样做可以检查他是否是一个函数式接口,同时,Javadoc也会包含一条声明,说明这个接口是一个函数式接口
3.2 自定义的函数式接口
@FunctionInterface
public interface MyNumber{public double getValue();
}// 函数式接口中使用泛型
@FunctionInterface
public interface MyFunc<T>{public T getValue(T t);
}// 作为参数传递Lambda表达式
public String toUpperString(MyFunc<String> mf, String str){return mf.getValue(str);
}// 作为参数传递给Lambda表达式:
String newStr = toUpperString((str) -> str.toUpperCase(), "abcdef");
System.out.println(newStr);/*
作为参数传递Lambda表达式: 为了将Lambda表达式作为参数传递,接受Lambda 表达式的参数类型必须与该Lambda表达式兼容的函数接口的类型
*/
3.3 Java内置的四大核心函数式接口
函数式接口 | 参数类型 | 返回类型 | 用途 |
---|---|---|---|
Consumer 消费型接口 | T | void | 对类型为T的对象应用操作,包含方法: void accept(T t) |
Supplier 供给型接口 | 无 | T | 返回类型为T的对象,包含方法:T get(); |
Function<T,R> 函数式接口 | T | R | 对类型为T的对象应用操作,并返回结果. 结果是R类型的对象. 包含的方法有: R apply (T t) |
Predicate 断定型接口 | T | boolean | 确定类型为T的对象是否满足某约束,并且返回boolean值. 包含方法: boolean test(T t) ; |
package com.atguigu.java8;import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;import org.junit.Test;/** Java8 内置的四大核心函数式接口* * Consumer<T> : 消费型接口* void accept(T t);* * Supplier<T> : 供给型接口* T get(); * * Function<T, R> : 函数型接口* R apply(T t);* * Predicate<T> : 断言型接口* boolean test(T t);* */
public class TestLambda3 {//Predicate<T> 断言型接口:@Testpublic void test4(){List<String> list = Arrays.asList("Hello", "Dreamhai", "Lambda", "www", "ok");List<String> strList = filterStr(list, (s) -> s.length() > 3);for (String str : strList) {System.out.println(str);}}//需求:将满足条件的字符串,放入集合中public List<String> filterStr(List<String> list, Predicate<String> pre){List<String> strList = new ArrayList<>();for (String str : list) {if(pre.test(str)){strList.add(str);}}return strList;}//Function<T, R> 函数型接口:@Testpublic void test3(){String newStr = strHandler("\t\t\t 我大青海威武 ", (str) -> str.trim());System.out.println(newStr);String subStr = strHandler("我大青海威武", (str) -> str.substring(2, 4));System.out.println(subStr);}//需求:用于处理字符串public String strHandler(String str, Function<String, String> fun){return fun.apply(str);}//Supplier<T> 供给型接口 :@Testpublic void test2(){List<Integer> numList = getNumList(10, () -> (int)(Math.random() * 100));for (Integer num : numList) {System.out.println(num);}}//需求:产生指定个数的整数,并放入集合中public List<Integer> getNumList(int num, Supplier<Integer> sup){List<Integer> list = new ArrayList<>();for (int i = 0; i < num; i++) {Integer n = sup.get();list.add(n);}return list;}//Consumer<T> 消费型接口 :@Testpublic void test1(){happy(10000, (m) -> System.out.println("你们海哥喜欢大宝剑,每次消费:" + m + "元"));} public void happy(double money, Consumer<Double> con){con.accept(money);}
}
函数式接口 | 参数类型 | 返回类型 | 用途 |
---|---|---|---|
BiFunction<T,U,R> | T,U | R | 对类型为T,U 参数应用操作,返回R类型的结果. 包含的方法为: R apply(T t , U u); |
UnaryOperator (Function子接口) | T | T | 对类型为T的对象进行一元运算, 并返回T类型的结果. 包含方法: T apply(T t); |
BinaryOperator (BiFunction子接口) | T,T | T | 对类型为T 的对象进行二元运算,并且返回T类型的结果. 包含方法为: T apply(T t1 , T t2) ; |
BiConsumer<T,U> | T,U | void | 对类型为T,U 参数应用操作. 包含方法为 void accept(T t,U u); |
ToIntFunction | T | int | 计算int值的函数 |
ToLongFunction | T | long | 计算long值的函数 |
ToDoubleFunction | T | double | 计算double值的函数 |
IntFunction | int | R | 参数为int类型的函数 |
LongFunction | long | R | 参数为long类型的函数 |
DoubleFunction | double | R | 参数为double类型的函数 |
四. 方法引用与构造器的引用
4.1 方法引用
- 当要传递给Lambda体的操作,已经有实现的方法了,可以试用方法引用!(实现抽象方法的参数列表,必须与方法引用方法的参数列表保持一致!)
- 方法引用: 使用操作符"
::
"将方法名和对象或类的名字分割开来 - 主要有以下三种情况:
3.1 对象::
实例方法
3.2 类::
静态方法
3.3 类::
实例方法
// 例如:
(x)->System.out.println(x); <=> System.out::println// 例如:
BinaryOperator<Double> bo = (x,y)-> Math.pow(x,y);
<=>
BinaryOperator<Double> bo = Math::pow;// 例如:
Compare((x,y)->x.equals(y),"abcdef","abcdef");
<=>
Compare(String::equals,"abcdef","abcdef");
// 注意: 当需要引入方法的第一个参数是调用对象,并且第二个参数是需要引用方法的第二个参数(或无参数)时: ClassName::methodName
4.2 构造器引用
格式: ClassName::new
与函数式接口相结合,自动与函数式接口中的方法兼容.
可以吧构造器引用赋值给定义的方法. 与构造器参数列表要与接口中抽象方法的参数列表一直!
Function<Integer,MyClass> fun = (n)->new MyClass(n);
<=>
Function<Integer,MyClass> fun = MyClass::new ;
4.3 数组引用
格式: type[] :: new
Function<Integer,Integer[]> fun = (n)->new Integer[n];
<=>
Function<Integer,Integer[]> fun = MyClass[]::new ;
JDK 8的新特性-Lambda表达式 精品文章总结相关推荐
- 【Java】jdk 1.8 新特性——Lambda表达式
Lambda表达式 jdk 1.8 新加入的特性,简化了简单接口的实现 函数式接口 函数式中只有一个待实现的方法,可以使用@FunctionalInterface注解标注函数式接口.这个接口中只能有一 ...
- java lambda表达式详解_Java8新特性Lambda表达式详解
课程目标: 通过本课程的学习,详细掌握Java8新特性之Lambda表达式: 适用人群:有Java基础的开发人员: 课程概述:从Java 8出现以来lambda是最重要的特性之一,它可以让我们用简洁流 ...
- Java 8 新特性 lambda表达式
/ Created by Manager on 2021/4/1. Java 8 新特性 lambda表达式 StreamAPI 新日期 新注解 */ 视频连接 1https://www.bilibi ...
- Java8新特性----Lambda表达式详细探讨
Java8新特性 Lambda表达式 入门演示 案例1 如何解决 cannot be cast to java.lang.Comparable问题? 案例2 优化方式一 : 策略设计模式 优化方式二: ...
- Java 8 新特性Lambda 表达式
Java 8 新特性Lambda 表达式 一.常用循环 二.匿名内部类 三.排序集合 四.循环打印对象 五.根据条件修改 六.排序 七.求和 八.统计方法 九.材料 一.常用循环 public cla ...
- C++11新特性——λ(lambda)表达式详解
C++11新特性--λ(lambda)表达式 C++11中引入了λ表达式,它可以用来定义一个内联(inline)的函数,作为一个本地的对象或者一个参数.有了λ表达式,我们可以很方便的使用stl标准库. ...
- java新特性lambda表达式快速入门
文章目录 序 常规写法 写法一:新建类并实现该接口 写法二:使用匿名函数 lambda写法 写法一:lambda常规写法 写法二:lambda简写 中场疑问 lambda的方法引用 写法一 写法二 练 ...
- jdk8新特性 lambda表达式详解
本文主要讲到的内容有: 一- 前言 二- 背景 三- lambda表达式的语法 四- Lambda程序例子 4-1 Runnable Lambda 4-2 Comparator Lambda 4-3 ...
- java thread lambda_Java8新特性--Lambda表达式
从java8出现以来lambda是最重要的特性之一,它可以让我们用简洁流畅的代码完成一个功能. 很长一段时间java被吐槽是冗余和缺乏函数式编程能力的语言,随着函数式编程的流行java8种也引入了 这 ...
最新文章
- 架构师之路 — 数据库设计 — 关系型数据库的迁移与版本控制
- 京东也不甘当当后!买书4折优惠!
- 计算机科学概论(2)数据的操控和程序的执行
- sql练习(针对Mysql)
- 键盘输入Scanner类方法属性使用
- mysql5.6.24怎么打开_mysql 5.6.24 安装配置方法图文教程
- Unreal Engine 4 中的 UI 优化技巧
- 傅立叶变换,时域,频域二
- STM32CubeMX 安装提示需要java环境(1.8.0_45)
- matlab不能radon变换,Radon变换的理解
- 山西最新五大姓氏排名发布,排名第一的是王,第二的竟是……
- Oracle9i数据库Data Guard实施及维护手册
- The exception was not handled due to missing onError handler in the subscribe() method call
- 远程桌面“无法验证此计算机的身份“
- java 发起HTTPS请求-SSL客户端
- Vue-纯前端导出word文档 Can‘t find end of central directory:is this a zip file?
- VVC帧间预测(四)仿射运动补偿预测
- 服务器硬防和软防是什么意思?服务器软件防火墙
- 大数据时代对存储发展的要求有哪些
- 如何在UnrealEngine虚幻引擎中进行版本管理
热门文章
- 【资讯】铂链加盟软银ARM人工智能联盟,与海尔、商汤科技同门
- 国外计算机研究生申请ddl,美国大学研究生申请ddl简介及具体时间安排
- 百度地图电子围栏判断 地图定位
- 谷歌智能眼镜回归,针对商业用户售价999美元
- pyautogui微信小程序自动化操作
- AS/400 初级培训课程
- ibatis学习以及与mybatis的不同
- php中注册数字插不进去,进php数字
- outlook邮箱邮件内容乱码_邮件标题乱码问题解决一例
- python数据分析比较好的书籍_python数据分析比较好的书籍推荐|陇川制作项目盈利能力分析...