lambda使用
我们都知道如何使用匿名内部类。如果一个匿名内部类非常的简单,比如它只包含一个方法,那么我们以前使用匿名内部类的语法可能看起来比较笨拙不清晰。因此,我们可以试着将功能函数作为一个方法的参数。Lambda表达式就可以让你将函数作为方法的参数,将代码作为数据。
比如,有一个Person类,如下:
public class Person {
public enum Sex {
MALE, FEMALE
}
String name;
LocalDate birthday;
Sex gender;
String emailAddress;
public int getAge() {
// ...
}
public void printPerson() {
// ...
}
}
有一个人员列表,你想筛选出符合某些条件的人,你可以添加如下接口和方法
interface CheckPerson {
boolean test(Person p);
}
public static void printPersons(
List<Person> roster, CheckPerson tester) {
for (Person p : roster) {
if (tester.test(p)) {
p.printPerson();
}
}
}
如果你想筛选出年龄在18-25岁之间的男性,可以这样
class CheckPersonEligibleForSelectiveService implements CheckPerson {
public boolean test(Person p) {
return p.gender == Person.Sex.MALE &&
p.getAge() >= 18 &&
p.getAge() <= 25;
}
}
printPersons(
roster, new CheckPersonEligibleForSelectiveService());
或是这样
printPersons(
roster,
new CheckPerson() {
public boolean test(Person p) {
return p.getGender() == Person.Sex.MALE
&& p.getAge() >= 18
&& p.getAge() <= 25;
}
}
);
这是我们以前使用的语法,但是有了lambda之后我们可以这样
printPersons(
roster,
(Person p) -> p.getGender() == Person.Sex.MALE
&& p.getAge() >= 18
&& p.getAge() <= 25
);
进一步简化为
printPersons(
roster,
p -> p.getGender() == Person.Sex.MALE
&& p.getAge() >= 18
&& p.getAge() <= 25
);
lambda表达式构成
一个用逗号分割,圆括号包裹的参数列表。如(param1,param2),如果只有一个参数,圆括号可以省略。
箭头符号:->
-
实体:可以是只有一句代码,也可以是包含多句代码的代码块。如:一句代码
p.getGender() == Person.Sex.MALE && p.getAge() >= 18 && p.getAge() <= 25
如果只有一句代码,Java 会在运行时计算它的值,并作为返回值。你也可在使用return语法来声明返回值,如下
p -> { return p.getGender() == Person.Sex.MALE && p.getAge() >= 18 && p.getAge() <= 25;
}
return 并不是一个表达式,所以必须使用{}包裹起来,才能作为lambda表达式使用。 我们不需要使用{}包裹一个返回值为void的语句。如
java
email -> System.out.println(email)
```
类型问题
我们来看一个问题,假如有如下方法
void invoke(Runnable r) {
r.run();
}
<T> T invoke(Callable<T> c) {
return c.call();
}
String s = invoke(() -> "done");
哪个方法invoke()方法会被调用?
invoke(Callable<T>)会被调用,因为单个语句的lambda表达是默认将其值作为返回值,() -> "done"的类型是Callable<T>
Java 编译器会根据双下文的情景来决定lambda表达式的类型。只要其中下面所列信息的一个,java 编译器就可以推断出lambda表达式的类型
- 变量声明(Variable declarations)
- 分配(Assignments)
- 返回类型 (Return statements)
- 数组的初始化 (Array initializers)
- 方法或构造器的参数 (Method or constructor arguments)
- lambda表达式的内容 (Lambda expression bodies)
- 条件表达式 ?:(Conditional expressions, ?:)
- 强制转换 (Cast expressions)
gradle配置步骤
-
在project的build.gradle的buidle.dependcies中添加(也可以添加到module的build.gradle)
classpath 'me.tatarka:gradle-retrolambda:3.2.0'
-
在module的build.gradle中添加(必须添加在module中)
apply plugin: 'me.tatarka.retrolambda'
-
在module的build.gradle的android中添加
compileOptions { sourceCompatibility org.gradle.api.JavaVersion.VERSION_1_8 targetCompatibility org.gradle.api.JavaVersion.VERSION_1_8 }
常见错误
1.junit4.12,在单元测试中使用lambda,报错!(未解决)
com.sun.tools.javac.code.Symbol$CompletionFailure: 找不到java.lang.invoke.MethodType的类文件
Error:Execution failed for task ':app:compileDebugUnitTestJavaWithJavac'.
Compilation failed; see the compiler error output for details.