什么是Lambda表达式
lambda表达式,它将允许我们将行为传到函数里。在Java 8之前,如果想将行为传入函数,仅有的选择就是匿名类,需要6行代码。而定义行为最重要的那行代码,却混在中间不够突出。Lambda表达式取代了匿名类,取消了模板,允许用函数式风格编写代码。这样有时可读性更好,表达更清晰。
Lamdba示例
用lambda表达式实现Runnable
使用lambda表达式可以替换匿名类,而实现
Runnable
接口是匿名类的最好示例。Java 8之前的runnable
实现方法,需要4行代码,而使用lambda
表达式只需要一行代码。只需要用() -> {}
代码块替代整个匿名类。
1
2
3
4
5
6
7
8
9
10// Java 8之前:
new Thread(new Runnable() {
输出:
1
2在Java8之前, 需要写很多代码
使用Java8, Lambda表达式一目了然
这个例子展示了Java 8 lambda表达式的语法,可以使用
lambda
写出如下代码:
1
2
3(params) -> expression
(params) -> statement
(params) -> { statements }
例如,如果你的方法不对参数进行修改、重写,只是在控制台打印点东西的话,那么可以这样写:
1
() -> System.out.println("Hello World");
如果你的方法接收两个参数,那么可以写成如下这样:
1
(int a, int b) -> a + b
使用lambda表达式进行事件处理
在Android日常开发中,我们常常会设置各种事件,比如
setOnClickListener
、setOnItemClickListener
等等,下面对比下前后的写法变化:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15//之前
viewA.setOnClickListener(new View.OnClickListener() {
这样一对比是不是简洁很多?那么对于多个参数的
setOnItemClickListener
怎么写呢?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19//之前
xxxListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
以上两种例子大概是日常Android开发中最为常用的
启用Lambda
启用
Lamdba
目前有两种方式,一个是使用Google官方的,一个是使用第三方Java8兼容插件,推荐使用第三方兼容插件 。基本要求如下:
- Android Studio 2.1+
- 安装好 JAVA 8
方式一:Google官方方式
要使用新的 Java 8 语言功能,还需使用新的 Jack 工具链。新的 Android 工具链将 Java 源语言编译成 Android 可读取的 Dalvik 可执行文件字节码,且有其自己的 .jack 库格式,在一个工具中提供了大多数工具链功能:重新打包、压缩、模糊化以及 Dalvik 可执行文件分包。
以下是构建 Android Dalvik 可执行文件可用的两种工具链的对比:
旧版 javac 工具链:
javac (.java --> .class) --> dx (.class --> .dex)
新版 Jack 工具链:Jack (.java --> .jack --> .dex)
配置 Gradle
如需为您的项目启用 Java 8 语言功能和 Jack,请在模块层级的 build.gradle 文件中输入以下内容:
1
2
3
4
5
6
7
8
9
10
11
12
13android {
...
defaultConfig {
...
jackOptions {
enabled true
}
}
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
}
已知问题
Instant Run 目前不能用于 Jack,在使用新的工具链时将被禁用。方式二:使用第三方Java8兼容插件
下面是插件的
ReadMe
的配置:添加下面的内容到项目的build.gradle
文件中
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17buildscript {
repositories {
mavenCentral()
}
dependencies {
classpath 'me.tatarka:gradle-retrolambda:3.2.5'
}
}
// Required because retrolambda is on maven central
repositories {
mavenCentral()
}
apply plugin: 'com.android.application' //or apply plugin: 'java'
apply plugin: 'me.tatarka.retrolambda'在本人的项目中,是如下配置,可以避免很多不必要的错误:
build.gradle
文件在项目根目录有一个,在Module下也有一个:/build.gradle
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19buildscript {
repositories {
jcenter()
}
dependencies {
classpath 'com.android.tools.build:gradle:2.1.2'
classpath 'me.tatarka:gradle-retrolambda:3.2.5'
classpath 'me.tatarka.retrolambda.projectlombok:lombok.ast:0.2.3.a2'
}
configurations.classpath.exclude group: 'com.android.tools.external.lombok'
}
allprojects {
repositories {
jcenter()
}
}/app/build.gradle
1
2
3
4
5
6
7
8
9
10apply plugin: 'com.android.application'
apply plugin: 'me.tatarka.retrolambda'
android {
...
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
}可以看到我在使用中多了几个配置,不过都是在这个插件的
issue
里查到的。建议仔细阅读下配置说明 ,可以适应一些实际项目中的一些特别需求。常见Error及解决方式
Couldnot find property ‘options’ on task ‘:app:compileDebugJavaWithJack’.
原因:使用Google官方的方式有一定的兼容性,使用Jack时不能同时使用APT,如果使用butterknife、Dagger等使用了APT的注解框架就不行了
解决方案:使用retrolambda的兼容插件的方式启用lambda
Doesn’t seem to correctly use modified lombok version
原因:参见这里
解决方案:按照我上面写的就不会出现这个问题了。
总结
对于想尝鲜的的开发者,启用Lamdba是个不错的选择,毕竟这个东西会慢慢普及的。下面是一些为想了解更多Lamdba使用方法的朋友搜集的资料以及本文参考的文章: