环境配置
android stuido配置
在bulidp(project)中配置
classpath 'me.tatarka:gradle-retrolambda:3.2.0'
接着在app modle中应用
apply plugin: 'me.tatarka.retrolambda'
最后在
android {
...
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
...
}
用Lambda表达式开启线程
我们知道,开启一个子线程。可以这么写:
new Thread(new Runnable() {
@Override
public void run() {
buffer.append("普通的开启一个子线程\n");
}
}).start();
那么Lambda表达式的写法是什么呢
new Thread(() -> {
buffer.append("通过lambda表达式开启一个子线程\n");
}).start();
是不是简单多了,我们不需要去new Runnable什么的再去重写它的run方法。而是直接通过() -> 方法替代了这么多代码。
当然我们还是先看看结果,看是不是这两个子线程都被开启了,期望的结果,buffer中拼接了两段字符串,当然我会把它显示在TextView上面。
结果很明显,两个线程都被开启了。
那么,什么情况下我们可以使用Lambda表达式呢?
函数式接口:Functional Interface。
定义的一个接口,接口里面必须有且只有一个抽象方法,但是可以有多个被default修饰的方法,这样的接口就称为函数是接口。
都知道接口可以有多个抽象方法,所以我们可以用@FunctionalInterface annotation注解,这样当我们的接口中只能由一个抽象方法,如果有多个时,就不提示错误。
在这里,因为java8的新特性,接口中可以有被实现的方法,但是该方法需要被default修饰。
所有,当我们对函数式的接口创建对象时,如上面的匿名内部类的创建,
普通情况我们可以这么写 :
Runnable runnable = new Runnable() {
@Override
public void run() {
//具体过程
}
};
用Lambda则是这么写:
Runnable runnable1 = () ->{
//具体过程
};
接着我们看看自定义的接口,如何用Lambda表达式来实现。
public interface MyListener{
String doSomething(String a,String b);
}
我们定义一个MyListener接口,当然如果里面写两个方法,那么Lambda表达式就会报错。
接着像Runnable一样创建一个对象。
那么问题来了,方法里面的参数并且还有返回值怎么办。机智的我们当然知道,返回值肯定直接return就可以里,至于参数呢!!!仔细想想,Runnable中的run方法是没有参数的,所以我们只用了(),那么面对有参数的方法,当然是在方法里面把参数带上。这样就可以return了啊。
所以我们的Lambda表达式可以这么写
MyListener myListener = (String a,String b) ->{
return a+b;
};
是不是超简单。
不过还有更简单的写法!
java会自动推断出Lambda表达式中的参数类型,没错,那就是连类型都不用写
最终上面的代码可以这么写:
MyListener myListener = (a,b) ->{
return a+b;
};
接口的方法被调用后的情况我们写好了,那么我们还需要一个能调用该接口的方法,我们才能看到结果。
那么我们随便写个方法吧,需要传入一个
private void helloLambda(String a, String b, MyListener listent){
result.setText(listent.doSomething(a,b));
}
然后在主线程中调用下
helloLambda("Hello ","Lambda ^ ^",myListener);
看看结果,没问题。我们自定义的接口也成功的使用了Lambda表达式
android中可以使用Lambda表达式的情况,除了上面的Thread,还有像设置点击事件之类的功能也非常适合
b.setOnClickListener(
(v) ->{
//具体点击后的处理过程
}
);
到这里Lambda的使用就介绍完了。