Lambda表达式说白了就是一种匿名方法,不需要方法名,修饰符,和返回值类型。
使用方法
首先在moudle的build.gradle文件中添加配置:app/build.gradle添加。
android {
...
defaultConfig {
...
jackOptions.enabled=true;
}
compileOptions{
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
...
}
就这样配置就OK了;
接下来我们来看看怎么使用;
首先拿一个点击事件来比对一下:
一般写法
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
}
});
解说:这是一种匿名内部类的写法,较之外部类的写法已经算是很简洁了。
Lambda表达式*
button.setOnClickListener(view ->{
});
解说:不难发现,Lambda的这种写法连匿名内部类都不用写了,更加简洁了,直接将参数暴露在外,在方法体中可以直接调用该参数;如果没有参数就用()就行,例如开启线程的写法:
new Thread(() -> {
});
到这里大家可能就要说了:不就是少了两行代码,有什么了不起,学的时候还不是要从匿名内部类写起?
那再用在点击事件中开启一个线程的:
一般写法
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
new Thread(new Runnable() {
@Override
public void run() {
}
});
}
});
看起来是不是有点眼花缭乱,很明显的onClick等方法有点抢镜,让run方法不是那么容易被发现;
Lambda表达式
button.setOnClickListener( view-> new Thread(() -> {
}));
怎么样,是不是被惊艳到了,服不服??Lambda就是这么简洁,简洁到没朋友。
可能有朋友就奇怪怎么写一起了,不该是两个分开的Lambda表达式吗?
button.setOnClickListener(view -> {
new Thread(() -> {
});
});
带着这个疑问我们从源头找起,来看看Lambda表达式到底简化了什么?
从上面我们不难发现,Lambda表达式的简化,其实是简化了接口的匿名内部类的实现:
//一般写法
View.OnClickListener listener=new View.OnClickListener() {
@Override
public void onClick(View view) {
}
};
//Lambda表达式
View.OnClickListener listener= view -> {
};
我们发现Lambda简化了OnClickListener这个接口的匿名实现和一个必须实现的方法,只留下一个参数。
那么 现在我们有了如下疑问?
Lambda表达式只能简化接口的匿名实现吗?
Lambda表达式能简化多个方法的接口的匿名实现吗?
Lambda表达式能简化方法里有多个参数的接口吗?
Lambda表达式能简化方法带有返回值的接口吗?
下面来一一解答:
Lambda表达式只能简化接口的匿名实现吗?
答案是肯定的。
就目前而言Lambda表达式只能简化接口的匿名内部类实现。
原因大概是:接口是没有构造方法,而抽象类和一般的类是有构造方法的;接口里的方法没有方法体等等。因为接口的特殊性,Lambda表达式就是只针对接口而已。
Lambda表达式能简化多个方法的接口的匿名实现吗?
答案是不能:这里从Lambda的表达式就可以看出来,已经简化到没有没有一丝多余的代码,多个方法怎么写呢。
Lambda只能简化单一方法接口的匿名内部类实现;
Lambda表达式能简化多个参数的单一方法接口吗?
这个答案是肯定的。
前面我在例子中已经用到了没有参数和一个参数的接口匿名实现。
没有参数一个空的小括号;
一个参数在括号里面添加一个参数,
多个参数就直接添加就OK了;
一个参数时小括号是可以省略的;
另外参数类型是可以省略的,当然也就可以写的;
省略括号时是不能写参数类型的。
//没有参数
Runnable runnable=() -> {
};
//一个参数
View.OnClickListener listener= (View view) -> {
};
View.OnClickListener listener= (view) -> {
//参数类型可省略
};
View.OnClickListener listener= view -> {
//括号可省略,但不能添加参数
};
//两个参数
public interface JackListener{
void check(String string,String check);
};
doWhat("jack666", (string, check) -> {
//这里的两个参数指的是接口里方法的参数,而不是doWhat的参数哦
});
Lambda表达式能简化方法带有返回值的接口吗?
答案是肯定的,其实返回值和简化前的写法是一样,return一下就可以了。示例如下;
//两个参数
public interface JackListener{
boolean check(String string,String check);
};
doWhat("jack666", (string, check) -> {
boolean result=string.contains(check);
Toast.makeText(MainActivity.this, ""+result, Toast.LENGTH_SHORT).show();
return result;
});
现在咱们再来看看前面的两个Lambda嵌套时发生了化学反应:
button.setOnClickListener(view -> {
new Thread(() -> {
});
});
//两个Lambda表达式嵌套,极简模式
button.setOnClickListener( view-> new Thread(() -> {
//前提是Lambda表达式中的方法体内部只有一个单纯Lambda表达式
}));
//四个Lambda表达式嵌套,CRAZY!!!
button.setOnClickListener(view -> new Thread(() -> button.setOnClickListener(view1 -> new Thread(() -> {
}))));
从Lambda表达式开始等于开启了Java的极简模式,期待下个版本会有更多的简化,祝福Java在简化这条路上越走越远,让嘲笑Java臃肿的人去屎!
欢迎加入学习交流群569772982,大家一起学习交流。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。