什么是lambda
Lambda 表达式,也可称为闭包,它是推动 Java 8 发布的最重要新特性。
Lambda 允许把函数作为一个方法的参数(函数作为参数传递进方法中)。
使用 Lambda 表达式可以使代码变的更加简洁紧凑。
语法
(parameters) -> expression或(parameters) ->{ statements; }
以下是lambda表达式的重要特征:
- 可选类型声明:不需要声明参数类型,编译器可以统一识别参数值。
- 可选的参数圆括号:一个参数无需定义圆括号,但多个参数需要定义圆括号。
- 可选的大括号:如果主体包含了一个语句,就不需要使用大括号。
- 可选的返回关键字:如果主体只有一个表达式返回值则编译器会自动返回值,大括号需要指定表达式返回了一个数值。
Lambda表达式示例
// 1. 不需要参数,返回值为 5() -> 5// 2. 接收一个参数(数字类型),返回其2倍的值x -> 2 * x// 3. 接受2个参数(数字),并返回他们的差值(x, y) -> x – y// 4. 接收2个int型整数,返回他们的和(int x, int y) -> x + y// 5. 接受一个 string 对象,并在控制台打印,不返回任何值(看起来像是返回void)(String s) -> System.out.print(s)
以上方法仅说明Lambda基础语法,不能独立运行。
简单运算方式的Lambda实现示例:
public class Java8Tester {public static void main(String args[]){Java8Tester tester = new Java8Tester();// 类型声明MathOperation addition = (int a, int b) -> a + b;// 不用类型声明MathOperation subtraction = (a, b) -> a - b;// 大括号中的返回语句MathOperation multiplication = (int a, int b) -> { return a * b; };// 没有大括号及返回语句MathOperation division = (int a, int b) -> a / b;System.out.println("10 + 5 = " + tester.operate(10, 5, addition));System.out.println("10 - 5 = " + tester.operate(10, 5, subtraction));System.out.println("10 x 5 = " + tester.operate(10, 5, multiplication));System.out.println("10 / 5 = " + tester.operate(10, 5, division));// 不用括号GreetingService greetService1 = message -> System.out.println("Hello " + message);// 用括号GreetingService greetService2 = (message) -> System.out.println("Hello " + message);greetService1.sayMessage("Runoob");greetService2.sayMessage("Google");}interface MathOperation {int operation(int a, int b);}interface GreetingService {void sayMessage(String message);}private int operate(int a, int b, MathOperation mathOperation){return mathOperation.operation(a, b);}}
10 + 5 = 1510 - 5 = 510 x 5 = 5010 / 5 = 2Hello RunoobHello Google
使用 Lambda 表达式需要注意以下两点:
- Lambda 表达式主要用来定义行内执行的方法类型接口,例如,一个简单方法接口。在上面例子中,我们使用各种类型的Lambda表达式来定义MathOperation接口的方法。然后我们定义了sayMessage的执行。
- Lambda 表达式免去了使用匿名方法的麻烦,并且给予Java简单但是强大的函数化的编程能力。
变量作用域
lambda 表达式只能引用标记了final的外层局部变量,这就是说不能在 lambda 内部修改定义在域外的局部变量,否则会编译错误。
public class Java8Tester {final static String salutation = "Hello! ";public static void main(String args[]){GreetingService greetService1 = message -> System.out.println(salutation + message);greetService1.sayMessage("Runoob");}interface GreetingService {void sayMessage(String message);}}
Hello! Runoob
public class Java8Tester {public static void main(String args[]) {final int num = 1;Converter<Integer, String> s = (param) -> System.out.println(String.valueOf(param + num));s.convert(2); // 输出结果为 3}public interface Converter<T1, T2> {void convert(int i);}}
lambda 表达式的局部变量可以不用声明为final,但是必须不可被后面的代码修改(即隐性的具有final的语义)
int num = 1;Converter<Integer, String> s = (param) -> System.out.println(String.valueOf(param + num));s.convert(2);num = 5;//报错信息:Local variable num defined in an enclosing scope must be final or effectively final
在 Lambda 表达式当中不允许声明一个与局部变量同名的参数或者局部变量。
String first = "";Comparator<String> comparator = (first, second) -> Integer.compare(first.length(), second.length()); //编译会出错
真实项目中的应用
Thread
使用Lambda表达式来简化匿名内部类创建线程案例:
public class Demo_Thread {public static void main(String[] args) {// 普通方式thread_old("old_thread");thread_new();}public static void thread_old(String tName) {// 开启一个新线程new Thread(new Runnable() {@Overridepublic void run() {System.out.println("老方式实现创建线程:" + Thread.currentThread().getName());}}, tName).start();}public static void thread_new() {new Thread(() -> System.out.println("这是新线程的创建方式:" + "thread_new")).start();}}
排序
下面是两种利用Comparator来排序的案例:
String[] players = {"Rafael Nadal", "Novak Djokovic","Stanislas Wawrinka", "David Ferrer","Roger Federer", "Andy Murray","Tomas Berdych", "Juan Martin Del Potro","Richard Gasquet", "John Isner"};// 1.1 使用匿名内部类根据 name 排序 playersArrays.sort(players, new Comparator<String>() {@Overridepublic int compare(String s1, String s2) {return (s1.compareTo(s2));}});// 1.2 使用 lambda expression 排序 playersComparator<String> sortByName = (String s1, String s2) -> (s1.compareTo(s2));Arrays.sort(players, sortByName);// 1.3 也可以采用如下形式:Arrays.sort(players, (String s1, String s2) -> (s1.compareTo(s2)));
同样的,根据第二个名字排序:
// 1.1 使用匿名内部类根据 surname 排序 playersArrays.sort(players, new Comparator<String>() {@Overridepublic int compare(String s1, String s2) {return (s1.substring(s1.indexOf(" ")).compareTo(s2.substring(s2.indexOf(" "))));}});// 1.2 使用 lambda expression 排序,根据 surnameComparator<String> sortBySurname = (String s1, String s2) ->( s1.substring(s1.indexOf(" ")).compareTo( s2.substring(s2.indexOf(" ")) ) );Arrays.sort(players, sortBySurname);// 1.3 或者这样Arrays.sort(players, (String s1, String s2) -> ( s1.substring(s1.indexOf(" ")).compareTo( s2.substring(s2.indexOf(" ")))));// 1.4 最简单方式Arrays.sort(players, Comparator.comparing((String s) -> s.substring(s.indexOf(" "))));
根据名称长度排序:
// 2.1 使用匿名内部类根据 name lenght 排序 playersArrays.sort(players, new Comparator<String>() {@Overridepublic int compare(String s1, String s2) {return (s1.length() - s2.length());}});// 2.2 使用lambda简化Comparator<String> sortByNameLength = Comparator.comparingInt(String::length);Arrays.sort(players, sortByNameLength);
高版本idea编译器对简化lambda有着更好的提示!
