Osheep

时光不回头,当下最重要。

不知道 Lambda 表达式 ,怎么写 Java?

Java 8 中为我们提供了一系列新特性,其中被称之为「语法糖」的 Lambda 表达式可谓是无数开发者的「装逼利器」,他简短的语法可以让我们写出更加优雅朴素高逼格的代码。

之所以被称之为「语法糖」,是因为他本质就是为简化代码而生,由编译器推断并帮你转换包装为常规的代码,因此你可以使用更少的代码来实现同样的功能。如果对没接触过的人来说,这就和某些很高级的黑客写的代码一样,简洁,难懂,难以调试,维护人员想骂娘,所以不建议乱用!

java 8 中,当然还有别的强大特性,比如 stream API、接口默认实现等等!因为 stream API 和接口默认实现等特性都只支持 Android 7.0 以上的系统,有兴趣的同学可以自己下去了解,在这里给大家着重介绍已兼容到 Android 2.3 的 Lambda 表达式!

Android 中使用 Java 8 属性

如果要想在 Android 项目中使用 Lambda 表达式或 Java 8 的其他新特性,这里以 Android Studio 为例,我们需要在 app/build.gradle 中做简单配置,具体的步骤请参考我博客上篇的文章 Android Studio 中配置 Java 8 属性

什么是 Lambda 表达式

Lambda 表达式本质上是一种匿名方法,它既没有方法名,也没有访问修饰符和返回值类型,使用它编写代码将会更加简洁,也更加易读!

lambda表达式允许你通过表达式来代替功能接口。 lambda表达式就和方法一样,它提供了一个正常的参数列表和一个使用这些参数的主体(body,可以是一个表达式或一个代码块)。

现在,我们已经简单知道什么是lambda表达式,让我们先从一些基本的例子开始。 在本节中,我们将看到lambda表达式如何影响我们编码的方式。
Lambda 表达式在 Android 中的具体应用

1. 线程中的写法
说了这么多,还是直接上代码的爽快!比如所传统开启一个子线程的写法如下:

new Thread(new Runnable() {
           @Override
           public void run()
              // 处理具体的逻辑
           }       
}).start();

而使用 Lambda 表达式则可以这样写:

new Thread(() -> {
 // 处理具体的逻辑
}).start();

这种高端的写法是不是觉得很神奇,很爽快,那么我很负责任的告诉你,如果习惯了你会爱上这种方式,且让我们接着往下看!

我们为什么可以使用这么爽快的写法呢?

这是因为 Thread 类的构造函数接受的参数是一个 Runnable 接口,并且接口中只有一个待实现方法,也就是 run 方法,我们来看一下极其简单的 Runnable 接口的源码,简单到让你尴尬!

public interface Runnable {
   void run();
}

那么结论一来了,凡是这种只有一个待实现方法的接口,都可以使用 Lambda 表达式的写法。比如说,通常创建一个类似于上述接口的匿名类实现需要这样写:

Runnable runnable = new Runnable() {
   @Override   public void run() {
       // 添加接口具体的实现   
} };

而有了 Lambda 就可以这样写了:

Runnable runnable = () -> {
 // 添加接口具体的实现
};

了解了 Lambda 表达式的基本写法,接下来我们用 Lambda 表达式 来实现我们自定义接口

// 新建一个 MyListener 接口
public interface MyListener {
   String doSomething(String a, int b);
}

和 Runnable 接口一样只有一个待实现方法,唯一不同的便是此方法中有参数和返回值,我们可以轻松的用 Lanbda 表达式这样实现:

MyListener listener = (String a, int b) -> {
   String result = a + b;   return result;
}

可以看到,doSomething () 方法的参数直接写在了括号后面,返回值写法与往常一致!

你以为这样既有简化完了?Java 可以根据上下文自动推断出 Lambda 表达式中的参数类型,因此进一步简化成如下方式:

MyListener listener = (a, b) -> {
   String result = a + b;   return result;
}

2. 有方法接收带参数接口(类似于上述 Runnable,MyListener 接口只有一个待实现方法)的参数时

举个例子,上代码!比如现在有一个方法是接受 MyListener 参数的:

public void SingleDog(MyListener listener) {
   String a = "Hello SingleDog";
   String b = "今年势必脱单";
   String result = listener.doSomething(a, b);
   Log.d("TAG", result);}

被问我为什么起名单身狗,因为我就是所以我骄傲!哈哈,傲娇脸 (ˇˍˇ)

我们在调用 SingleDog() 这个方法的时候可以这样写:

SingleDog((a, b) -> {
   String result = a + b;   return result;
});

那么 doSomething() 方法就会将 a 和 b 两个参数进行相加,从而最终打印结果就会是 “Hello SingleDog 今年势必脱单” ,单身狗共勉!

现在我们已经将 Lambda 表达式的写法基本都掌握了,接下来我们着重看一下在 Android 中有哪些常用的功能是可以用 Lambda 表达式进行替换的。

3. Lambda 表达式 对 Android 中常用功能的替换

现在我们知道只有一个待实现方法的接口都可以用 Lambda 表达式来编写,其实除了刚才说明的开启子线程的例子之外,还有我们时常会用到的设置点击事件,也是非常适合使用 Lambda 表达式的!

传统情况下我们给一个按钮设置点击事件的写法:

Button button = (Button) findViewById(R.id.button);
button.setOnClickListener(new View.OnClickListener() {
   @Override
   public void onClick(View v){
     // 处理点击事件
   }});

而使用了 Lambda 表达式后就可以将代码简化成这个样子了!

Button button = (Button) findViewById(R.id.button);
button.setOnClickListener((v) -> {
   // 处理点击事件
});

别急,还没「脱光」,如上当接口的待实现方法只有一个参数的时候,则可以直接将参数外面的括号去掉,如下:

Button button = (Button) findViewById(R.id.button);
button.setOnClickListener(v -> {
   // 处理点击事件
});

有没有一种当你以后用这种方式(ButterKnife 除外),写点击事件的时候会被人膜拜的情景映入脑海,哈哈!

总结

这样我们就基本已经将 Lambda 表达式的主要内容全都掌握了,是不是很 easy , 而且又觉得掌握了一手高大上的小技能。当然了,是否喜欢和接受这种极简主义的写法完全在个人,Java 8 对于哪一种写法都是支持的,多一种选择不是坏事,长个小见识也不至于见到后当场懵逼!

点赞