Lambda 简介
Lambda 表达式本质上是匿名方法,其底层还是通过 invokedynamic 指令来生成匿名类来实现。它提供了更为简单的语法和写作方式,允许你通过表达式来代替函数式接口。
Lambda 表达式,可以让我们的代码变得简洁,并且可以通过数据流的方式处理集合。
函数式接口的概念
因为 Lambda 表达式应用了函数接口,我们先了解一下函数式接口的概念。
函数式接口即只有一个抽象方法的接口。函数式接口可通过注解 @FunctionalInterface 定义。
Java8 中内置了四大核心函数式接口:
1)消费型接口:
interface Consumer{
void accept(T t);
}
2)供给型接口:
interface Supplier{
T get();
}
3)函数型接口:
interface Function{
R apply(T t);
}
4)断言型接口:
interface Predicate{
boolean test(T t);
}
我们可以根据参数的个数以及返回值类型,选用合适的函数式接口。
Lambda 基本语法
(parameters) -> expression
或
(parameters) ->{ statements; }
parameters:参数,这里的参数指的是函数接口里的参数。可明确表示,也可以由 JVM 进行腿短。
->:指的是参数用于右边的 expression 和 statements。(方法体)
expression 或 statements:可以理解为方法体
下面通过几个示例看一看如何使用:
//示例1:不接受参数,直接返回5
() -> 5
//示例2:接受两个String类型参数,并打印出来
(String x, String y) -> System.out.println( x + y);
//示例3:接受x,y两个参数,参数类型由JVM根据上下文推断出来,并返回两个参数的和
(x, y) -> x+y;
Lambda 的接口使用
Lambda 表达式的目标类型是函数性接口——每一个 Lambda 都能通过一个特定的函数式接口与一个给定的类型进行匹配。
自定义函数接口使用示例
@Functional
Interface interface Converter{
T convert(F from);
}
使用传统方式使用该接口:
Converter converter=new Converter() {
@Override
public Integer convert(String from) {
return Integer.valueOf(from); }
};
Integer result = converter.convert("200");
System.out.println(result);
但是如果使用 Lambda 表达式的话,代码就可以变的很简洁:
Converter converter=(param) -> Integer.valueOf(param); //将overrid写在表达式中
Integer result = converter.convert("101");
System.out.println(result);
实现 Runnable 接口
1.1 使用匿名内部类
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Hello world !");
}
}).start();
1.2 使用 lambda expression
new Thread(() -> System.out.println("Hello world !")).start();
2.1 使用匿名内部类
Runnable race1 = new Runnable() {
@Override
public void run() {
System.out.println("Hello world !");
}
};
2.2 使用 lambda expression
Runnable race2 = () -> System.out.println("Hello world !");
// 直接调用 run 方法(没开新线程)
race1.run();
race2.run();
Lambda 循环使用
String[] names={"Mike","Mary","Eric","Lucy","Rose"};
List<String> students=Arrays.asList(names);
// 以前的循环方式
for (String student: students) {
System.out.print(student+ "; ");
}
//方式一: 使用 lambda 表达式以及函数操作
students.forEach((student) -> System.out.print(student+ "; "));
// 方式二:使用双冒号操作符(Lambda表达式方法引用的第一种语法格式,下面会讲 )
students.forEach(System.out::println);
方法引用
方法引用,有三种语法格式:
1)对象::实例方法名
例:Consumer con=(x)->System.out.println(x);可以写为:
Consumer con=System.out::println;//System.out 即 PrintWriter 对象
2)类::静态方法名
例:Comparator com=(x,y)->Integer.compare(x,y);可以写为:
Comparator com=Integer::compare;
3)类::实例方法名
例:BiPredicate bp=(x,y)->x.equals(y);
BiPredicate bp=String::equals;
静态方法使用
传统情况下,我们可能写出这种代码:
public class User {
public static void main(String[] args) {
Converter<String ,Integer> converter=new Converter<String, Integer>() {
@Override
public Integer convert(String from) {
return User.String2Int(from);
}
};
converter.convert("120");
}
@FunctionalInterface
interface Converter<F,T>{
T convert(F from);
}
static int String2Int(String from) {
return Integer.valueOf(from);
}
}
如果使用 Lambda 表达式,静态方法引用的话:
public class User {
public static void main(String[] args) {
Convert<String, Integer> convert = User::string2Int;
System.out.println(convert.convert("120"));
}
@FunctionalInterface
interface Convert<F, T> {
T convert(F from);
}
static int string2Int(String from) {
return Integer.valueOf(from);
}
}
将 Override 改为一句话:
Converter converter = User::String2Int;
使用 User 类里的静态方法 string2Int。
实例方法使用*
传统代码中:
public class User {
public static void main(String[] args) {
Converter<String, Integer> converter = new Converter<String, Integer>() {
@Override
public Integer convert(String from) {
return new Helper().String2Int(from);
}
};
converter.convert("120");
}
@FunctionalInterface
interface Converter<F, T> {
T convert(F from);
}
static class Helper {
public int String2Int(String from) {
return Integer.valueOf(from);
}
}
}
使用 Lambda 表达式中的方法引用的话,十分简洁:
Helper helper = new Helper();
Converter<String, Integer> converter = helper::String2Int;
converter.convert("120");
Stream 方面的应用,因为目前没有怎么看流,所以准备等熟悉了解流之后再完整笔记。
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于