本文是Lambda的入门文章。主要向读者介绍Lambda的基本概念和写法。主要面向的读者是Java新手。
主要内容:
- Lambda到底是什么东西?
- 怎么通过Lambda简化代码?
背景知识:其中在逐步推导Lambda表达式时,需要用到并发编程中的Runnable接口。
目录
什么是Lambda
基本特性:
- 是JDK1.8开始之后的新技术,是一种代码的新语法。
- 目的是为了简化匿名内部类的代码写法
使用前提:
- 首先必须是接口。
- 接口中只能有一个抽象方法。
- 即,Lambda表达式只能简化接口中只有一个抽象方法的匿名内部类形式。
函数式接口:
- 接口中只有一个抽象方法的接口称为函数式接口。
- 函数式接口注解:@FunctionalInterface 一旦某个接口加上了这个注解,这个接口只能、有且仅有一个抽象方法。
Lambda 语法
- (params) -> expression
- (params) -> statement
- (params) -> { statements }
- 左侧:Lambda 表达式的参数列表;右侧:Lambda 表达式中所需执行的功能, 即 Lambda 体
第一个Lambda表达式
在并发编程中,我们可以通过实现Runnable接口来开辟一个新的任务线程。下面,通过任务线程的代码编写,来理解Lambda是如何简化写法的。
原始写法
首先明确,Runnable接口只有一个方法,满足函数式接口的条件:
以下是常规的写法:
public class Main {
public static void main(String[] args) {
Thread thread = new Thread(new MyThread()); //创建一个线程
thread.start(); //执行线程
}
}
//实现Runnable接口,实现任务线程
class MyThread implements Runnable {
@Override
public void run() {
System.out.println("线程"+Thread.currentThread().getName()+"执行了!");
}
}
静态内部类简化
将线程任务类放在要调用内的内部,作为静态内部类或者成员内部类:
public class Main {
//实现Runnable接口,实现任务线程
public static class MyThread implements Runnable {
@Override
public void run() {
System.out.println("线程"+Thread.currentThread().getName()+"执行了!");
}
}
public static void main(String[] args) {
Thread thread = new Thread(new MyThread()); //创建一个线程
thread.start(); //执行线程
}
}
局部内部类简化
把任务线程放在方法体内、局部变量的位置:
public class Main {
public static void main(String[] args) {
class MyThread implements Runnable { //局部内部类,就相当于申明一个变量
@Override
public void run() {
System.out.println("线程"+Thread.currentThread().getName()+"执行了!");
}
}
Thread thread = new Thread(new MyThread()); //创建一个线程
thread.start(); //执行线程
}
}
匿名内部类简化
省略实现类类名的方式,其中“new Runnable() {}”,等价于“class 名字 implements Runnable() {}”:
public class Main {
public static void main(String[] args) {
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("线程"+Thread.currentThread().getName()+"执行了!");
}
}); //申明任务线程
thread.start(); //执行线程
}
}
Lambda简化
Lambda的简化格式:
(匿名内部类被重写方法的形参列表) -> {
被重写方法的方法体代码。
}
Lambda简化:
public class Main {
public static void main(String[] args) {
//使用Lambda简化:
// 把"public void run()"简化为"()"。
// 之所以不用指定方法名,是因为函数式接口中有且仅有一个方法,只要一调用,只有那一个方法可以调用
// 方法体保持不变
Thread thread = new Thread( () -> {
System.out.println("线程"+Thread.currentThread().getName()+"执行了!");
});
thread.start();
}
}
再简化
再简化:
public class Main {
public static void main(String[] args) {
//方法体内只有一行代码,可以直接去掉方法体的花括号:{}
Thread thread = new Thread(() -> System.out.println("线程"+Thread.currentThread().getName()+"执行了!") );
thread.start();
}
}
匿名对象的简化:
public class Main {
public static void main(String[] args) {
//把Thread类转换为匿名对象
new Thread(() -> System.out.println("线程"+Thread.currentThread().getName()+"执行了!")).start();
}
}
Lambda实例
Lambda简化Comparator接口匿名内部类写法:
public static void main(String[] args) {
List<Student> lists = new ArrayList<>();
Student s1 = new Student("李铭",18,'男');
Student s2 = new Student("冯龙",23,'男');
Student s3 = new Student("王乐乐",21,'男');
Collections.addAll(lists , s1 , s2 , s3);
// 按照年龄进行升序排序!
Collections.sort(lists, new Comparator<Student>() {
@Override
public int compare(Student s1, Student s2) {
return s1.getAge() - s2.getAge();
}
});
// Lambda简化写法
Collections.sort(lists, (Student t1, Student t2) -> {
return t1.getAge() - t2.getAge();
});
// 如果Lambda表达式的方法体代码只有一行代码。可以省略大括号不写。
// 此时,如果这行代码是return语句,必须省略return不写,同时也必须省略";"不写
Collections.sort(lists,(Student t1, Student t2) -> t1.getAge() - t2.getAge());
// 参数类型可以省略
Collections.sort(lists, (t1, t2) -> t1.getAge() - t2.getAge());
System.out.println(lists);
}
forEach遍历简化
public static void main(String[] args) {
List<String> names = new ArrayList<>();
names.add("胡伟光");
names.add("甘挺");
names.add("洪磊");
//forEach遍历原理
names.forEach(new Consumer<String>() {
@Override
public void accept(String s) {
System.out.println(s);
}
});
//Lambda简化
names.forEach((String s) -> {
System.out.println(s);
});
//省略入参类型
names.forEach((s) -> {
System.out.println(s);
});
//如果只有一个参数,省略:入参括号
names.forEach(s -> {
System.out.println(s);
});
//Lambda表达式的方法体代码只有一行代码,省略花括号、分号
names.forEach(s -> System.out.println(s) );
names.forEach(System.out::println);
}
总结:
- Lambda的作用是简化代码的写法,减少代码量。
- 核心语法是:(参数1,参数2) -> { 方法体(如果方法体可以使用一句代码,则可以省略花括号) }