Java是一门强大的面向对象的语言,除了8种基本的数据类型,其他一切皆为对象。因此,在Java中定义函数或方法都离不开对象,也就意味着很难直接将方法或函数像参数一样传递,而Java8中的Lambda表达式解决了这个问题。

一、为什么需要Lambda

简单的来说,引入Lambda就是为了简化代码,允许把函数作为一个方法的参数传递进方法中。

1.1 真的简化了?

示例:如果想把某个接口的实现类作为参数传递给一个方法会怎么做?

  • Java8以前
public static void general() {
// 用匿名内部类的方式来创建线程
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("公众号:风尘博客!");
}
}).run();
}
  • Lambda 写法
public static void lambda() {
// 使用Lambda来创建线程
new Thread(() -> System.out.println("公众号:风尘博客!")).run();
}

1.2 Lambda表达式是什么?

Java中,将方法作为参数进行传递的方式被称为Lambda表达式

1.3 Lambda 表达式语法结构

Lambda其实是一个箭头函数,也可称为匿名函数:->

箭头操作符将Lambda表达式分成了两部分:

  1. 左侧:Lambda表达式的参数列表(接口中抽象方法的参数列表)
  2. 右侧:Lambda表达式中所需执行的功能(Lambda体,对抽象方法的实现)

1.4 语法格式

  • 无参,无返回值,Lambda 体只需一条语句。
public static void noParam() {
Runnable r1 = () -> System.out.println("noParam Test!");
r1.run();
}
  • Lambda 需要一个参数,参数的小括号可以省略。
public static void oneParam() {
// Consumer<String> con = (s) -> System.out.println(s);
// 参数的小括号可以省略。
Consumer<String> con = s -> System.out.println(s);
con.accept("oneParam Test!");
}
  • Lambda 需要多个参数,并且有返回值。
public static void params() {
Comparator<Integer> com = (x, y) -> {
System.out.println("函数式接口");
// 比较x/y的大小
return Integer.compare(x, y);
};
System.out.println(com.compare(1, 2));
}
  • Lambda 体只有一条语句时,return 与大括号可以省略。
public static void one() {
Comparator<Integer> com = (x, y) -> Integer.compare(x, y);
System.out.println(com.compare(1, 2));
}

上面几条示例好像有一个共性:参数列表的数据类型都没写,这是为什么呢?

1.5 类型推断

Lambda 表达式中的参数类型都是由编译器推断得出的。

public static void typeInference() {
//Integer 类型可以省略
Comparator<Integer> com = (Integer x,Integer y) -> {
System.out.println("函数式接口");
return Integer.compare(x, y);
};
// 类型推断
BinaryOperator<Long> addImplicit = (x, y) -> x + y;
}

Lambda 表达式中无需指定类型,程序依然可 以编译,这是因为 javac 根据程序的上下文,在后台 推断出了参数的类型。Lambda 表达式的类型依赖于上下文环境,是由编译器推断出来的。

1.6 小节

Lambda表达式使得Java拥有了函数式编程的能力,但在JavaLambda表达式是对象,它必须依附于一类特别的对象类型——函数式接口(functional interface)。

二、函数式接口

函数接口是只有一个抽象方法的接口,用作 Lambda 表达式的类型。使用@FunctionalInterface注解修饰的类,编译器会检测该类是否只有一个抽象方法或接口,否则,会报错。可以有多个默认方法,静态方法。

JDK8java.util.function 中定义了几个标准的函数式接口,供我们使用。

2.1 Java 内置四大核心函数式接口

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

void accept(T t);

consumerDemo(3, s -> System.out.println(s * 3));

public static void consumerDemo(Integer value, Consumer<Integer> consumer) {
consumer.accept(value);
}
  • 供给型接口

T get();

// 生成10个以内的随机书
List<Integer> numList = supplierDemo(10, () -> (int)(100 * Math.random()));
System.out.println(numList); public static List<Integer> supplierDemo(int num, Supplier<Integer> supplier) {
List<Integer> list = new ArrayList<>();
for (int i = 0; i < num; i++) {
Integer n = supplier.get();
list.add(n);
}
return list;
}
  • 函数型接口

R apply(T t);

// 处理字符串
String str1 = functionDemo("Hello!风尘博客", s -> s.substring(6));
System.out.println(str1);
String str2 = functionDemo("vanDusty", s -> s.toUpperCase());
System.out.println(str2); public static String functionDemo(String str, Function<String, String> function) {
return function.apply(str);
}
  • 断言型接口

boolean test(T t);

// 将满足条件的字符串放入集合
List<String> list = Arrays.asList("hello", "van", "function", "predicate");
List<String> newList = predicateDemo(list, s -> s.length() > 5);
System.out.println(newList); public static List<String> predicateDemo(List<String> list, Predicate<String> predicate) {
List<String> newList = new ArrayList<>();
for (String s : list) {
if (predicate.test(s)) {
newList.add(s);
}
}
return newList;
}

2.2 自定义函数式接口

我们可以在任意函数式接口上使用 @FunctionalInterface 注解, 这样做可以检查它是否是一个函数式接口,同时 javadoc 也会包含一条声明,说明这个接口是一个函数式接口。

// 字符串转大写
String newStr = selfFunctionalInterface((str) -> str.toUpperCase(), "abc");
System.out.println(newStr); public static String selfFunctionalInterface(SelfFunctionalInterface<String> selfFunctionalInterface, String str) {
return selfFunctionalInterface.getValue(str);
}

三、方法引用和构造器引用

3.1 方法引用

方法引用是指通过方法的名字来指向一个方法。

3.1.1 方法引用使用的前提条件是什么呢?

  1. 方法引用所引用的方法的参数列表必须要和函数式接口中抽象方法的参数列表相同(完全一致);
  2. 方法引用所引用的方法的的返回值必须要和函数式接口中抽象方法的返回值相同(完全一致)。

3.1.2 方法引用三种格式

  • 实例对象名::实例方法名
private static void instanceMethod() {
UserDomain user = new UserDomain(1L, "Van"); Supplier<String> sup = () -> user.getUserName();
System.out.println(sup.get()); // 等同于
Supplier<String> supplier = user::getUserName;
System.out.println(supplier.get());
}
  • 类名::静态方法名
private static void staticMethod() {
Comparator<Integer> com = (x, y) -> Integer.compare(x, y);
System.out.println(com.compare(3,9)); // 等同于
Comparator<Integer> com2 = Integer::compare;
System.out.println(com2.compare(3,9));
}
  • 类名::实例方法名
private static void instanceMethodObject() {
UserDomain user = new UserDomain(1L, "Van"); Function<UserDomain, String> fun = (e) -> e.getUserName();
System.out.println(fun.apply(user)); // 等同于
Function<UserDomain, String> fun2 = UserDomain::getUserName;
System.out.println(fun2.apply(user));
}

3.2 构造器引用

  1. 前提:构造器参数列表要与接口中抽象方法的参数列表一致!
  2. 语法格式:类名 :: new
  • 构造器引用
private static void object() {
// UserDomain 中必须有一个 UserDomain(String userName) 的构造器,下同
Function<String,UserDomain> fun = (n) -> new UserDomain(n);
fun.apply("Van"); System.out.println("===等价于===");
Function<String,UserDomain> function = UserDomain::new;
function.apply("Van"); // 带两个参数的构造器引用就要用BiFunction,多个参数的话,还可以自定义一个这样的函数式接口
BiConsumer<Long, String> biConsumer = UserDomain :: new;
biConsumer.accept(1L,"Van");
}
  • 数组引用
private static void array() {
//传统Lambda实现
Function<Integer,int[]> function = (i) -> new int[i];
int[] apply = function.apply(10);
System.out.println(apply.length); //数组类型引用实现
function = int[] ::new;
apply = function.apply(100);
System.out.println(apply.length);
}

四、 总结

Github 示例代码

Lambda表达式是Java对于函数式编程的温和转变,面向对象编程和函数式编程不是互相对立的,结合使用能够更加有效地帮助我们管理程序的复杂性。

技术交流

  1. 风尘博客
  2. 风尘博客-掘金
  3. 风尘博客-博客园
  4. Github

必知必会之Lambda表达式的更多相关文章

  1. 必知必会之Java注解

    必知必会之Java注解 目录 不定期更新中-- 元注解 @Documented @Indexed @Retention @Target 常用注解 @Deprecated @FunctionalInte ...

  2. 《MySQL 必知必会》读书总结

    这是 <MySQL 必知必会> 的读书总结.也是自己整理的常用操作的参考手册. 使用 MySQL 连接到 MySQL shell>mysql -u root -p Enter pas ...

  3. SQL 必知必会

    本文介绍基本的 SQL 语句,包括查询.过滤.排序.分组.联结.视图.插入数据.创建操纵表等.入门系列,不足颇多,望诸君指点. 注意本文某些例子只能在特定的DBMS中实现(有的已标明,有的未标明),不 ...

  4. 《MySQL必知必会》[01] 基本查询

    <MySQL必知必会>(点击查看详情) 1.写在前面的话 这本书是一本MySQL的经典入门书籍,小小的一本,也受到众多网友推荐.之前自己学习的时候是啃的清华大学出版社的计算机系列教材< ...

  5. mysql必知必会

    春节放假没事,找了本电子书mysql必知必会敲了下.用的工具是有道笔记的markdown文档类型. 下面是根据大纲已经敲完的章节,可复制到有道笔记的查看,更美观. # 第一章 了解SQL## 什么是S ...

  6. 《MySQL必知必会》整理

    目录 第1章 了解数据库 1.1 数据库基础 1.1.1 什么是数据库 1.1.2 表 1.1.3 列和数据类型 1.1.4 行 1.1.5 主键 1.2 什么是SQL 第2章 MySQL简介 2.1 ...

  7. msql 必知必会笔记

    Edit Mysql 必知必会 第一章 理解SQL 什么是数据库 数据库(database) 保存有组织的数据的容器 什么是表  一组特定类型的数据的结构化清单 什么是模式  数据库和表的布局及特性的 ...

  8. 《SQL必知必会》笔记

    SQL必知必会(第4版) 作者:[美]Ben Forta 本书介绍了sql在不同数据库工具(Oracle.SQLite.SQL server.MySQL.MariaDB.PostgreSQL...)是 ...

  9. SQL 必知必会 总结(一)

    SQL必知必会 总结(一) 第 1 课 了解SQL 1.数据库(database): 保存有组织的数据容器(通常是一个文件或一组文件). 2.数据库管理系统(DBMS): 数据库软件,数据库是通过 D ...

  10. MySql必知必会实战练习(二)数据检索

    在上篇博客MySql必知必会实战练习(一)表创建和数据添加中完成了各表的创建和数据添加,下面进行数据检索和过滤操作. 1. Select子句使用顺序 select--->DISTINCT---& ...

随机推荐

  1. 阿里CTR预估:用户行为长序列建模

    本文将介绍Alibaba发表在KDD'19 的论文<Practice on Long Sequential User Behavior Modeling for Click-Through Ra ...

  2. nginx负载均衡动态自动更新(微博开源模块nginx-upsync-module使用)

    这几天项目有个需求:负载要求能根据节点健康状态动态的增减.nginx自带的upstram已经很强大,而且基于Nginx Upstream配置动态更新已经有很多开源方案,大多数都是基于生成配置文件后进行 ...

  3. python文档字符串(函数使用说明)

    关键字: 函数说明.help()函数 1.效果图: 2.代码: # 文档字符串( doc str) 是 函数使用说明 # 用法: 在函数第一行写一个字符串 def fn(*nums): ''' 函数的 ...

  4. 初探ASP.NET Core 3.x (1) - 关于ASP.NET Core

    I 什么是ASP.NET Core ASP.NET is an open source web framework, created by Microsoft, for building modern ...

  5. Nginx的一理解(2)

    1.静态HTTP服务器 首先,Nginx是一个HTTP服务器,可以将服务器上的静态文件(如HTML.图片)通过HTTP协议展现给客户端. 配置:

  6. Xcode10:library not found for -lstdc++.6.0.9 临时解决

    1.https://pan.baidu.com/s/1IkbZb6qaxgvghP1HEFQa6w?errno=0&errmsg=Auth%20Login%20Sucess&& ...

  7. mysql使用唯一索引避免插入重复数据

    使用MySQL 索引防止一个表中的一列或者多列产生重复值 一:介绍MYSQL唯一索引 如果要强烈使一列或多列具有唯一性,通常使用PRIMARY KEY约束. 但是,每个表只能有一个主键. 因此,如果使 ...

  8. 客户端TNSPING通 连接出现ORA-12514错误

    ORA-12514: TNS: 监听程序当前无法识别连接描述符中请求的服务,这是一个经常遇到的问题,可以按照以下步骤一步步解决 1.使用tnsping检测 tnsping可判断出以下两点(1)判断网络 ...

  9. Maven的学习之路1

    对于下列这段Maven的命令行有不懂之处,在zgmzyr的博客上得到了解释.非常感谢这位博友,写在自己的随笔上以便查看和修改. 1. 创建项目 $ mvn archetype:generate -Dg ...

  10. php--->查询超大文件(12G)

    今天遇到一个要在一个12G日志中查询数据的需求,手中暂时没有查询这种超大文件的工具,于是自己写了一个程度来读这个超大文件 其整体思路就是一行一行地去读取超大文件中的数据,然后将拿出的一行数据做相应的查 ...