jdk8已经发布4年,其中有一个特性:Lambda,它是一个令开发者便捷开发的一种方式,Lambda Expression (Lambda表达式)是为了让java提供一种面向函数编程,原本在jdk8之前只支持面向对象编程,

而函数式编程则是对行为的抽象(将行为作为一个参数进行传递),接下来通过例子来说明:

这是一个线程对象,传入匿名内部类的例子:

new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Hello World!");
}
});

IDEA会给出提示可以使用Lambda表达式替换。

通过使用Lambda表达式则只需要使用一句话就可代替上面使用匿名类的方式。

new Thread(() -> System.out.println("Hello World!"));

  

在这个例子中,传统的语法规则,我们是将一个匿名内部类作为参数进行传递,我们实现了Runnable接口,并将其作为参数传递给Thread类,这实际上我们传递的是一段代码,也即我们将代码作为了数据进行传递,这就带来许多不必要的“样板代码”。

  Lambda表达式一共有三部分组成:

  后面的示例中我们会详解这个结构,包括有无参数,有无返回值的问题。 那么这个看起来奇奇怪怪的不太像Java的语法规则,其本身含义到底什么呢?这也是开始困扰我的问题,什么时候在什么场景下可以使用Lambda表达式。

  能够接收Lambda表达式的参数类型,是一个只包含一个方法的接口。只包含一个方法的接口称之为“函数接口”。

  例如上面创建一个线程的示例,Runnable接口只包含一个方法,所以它被称为“函数接口”,所以它可以使用Lambad表达式来代替匿名内部类。根据这个规则,我们试着来写一个函数接口,并使用Lambda表达式作为参数传递。

1 package com.coderbuff.custom;
2
3 /**
4 * 函数接口:只有一个方法的接口。作为Lambda表达式的类型
5 * Created by Kevin on 2018/2/17.
6 */
7 public interface FunctionInterface {
8 void test();
9 }

  测试:

 1 package com.coderbuff.custom;
2
3 import org.junit.Test;
4
5 /**
6 * 函数接口测试
7 * Created by Kevin on 2018/2/17.
8 */
9 public class FunctionInterfaceTest {
10
11 @Test
12 public void testLambda() {
13 func(new FunctionInterface() {
14 @Override
15 public void test() {
16 System.out.println("Hello World!");
17 }
18 });
19 //使用Lambda表达式代替上面的匿名内部类
20 func(() -> System.out.println("Hello World"));
21 }
22
23 private void func(FunctionInterface functionInterface) {
24 functionInterface.test();
25 }
26 }

  可以看到,只要是一个接口中只包含一个方法,则可以使用Lambda表达式,这样的接口称之为“函数接口”。

  上面的函数接口比较简单不包含参数,也不包含返回值

  我们再来修改FunctionInterface函数接口逐步加大Lambda表达式的难度——包含参数,不包含返回值

1 package com.coderbuff.custom;
2
3 /**
4 * 函数接口:只有一个方法的接口。作为Lambda表达式的类型
5 * Created by Kevin on 2018/2/17.
6 */
7 public interface FunctionInterface {
8 void test(int param);
9 }

  测试:

 1 package com.coderbuff.custom;
2
3 import org.junit.Test;
4
5 /**
6 * 函数接口测试
7 * Created by Kevin on 2018/2/17.
8 */
9 public class FunctionInterfaceTest {
10
11 @Test
12 public void testLambda() {
13 //使用Lambda表达式代替匿名内部类
14 func((x) -> System.out.println("Hello World" + x));
15 }
16
17 private void func(FunctionInterface functionInterface) {
18 int x = 1;
19 functionInterface.test(x);
20 }
21 }

  关注Lambda表达式“(x) -> Sysout.out.println("Hello World" + x)”,左边传递的是参数,此处并没有指明参数类型,因为它可以通过上下文进行类型推导,但在有些情况下不能推导出参数类型(在编译时不能推导通常IDE会提示),此时则需要指明参数类型。我个人建议,任何情况下指明函数的参数类型

  哪种情况不能推导出参数类型呢?就是函数接口是一个泛型的时候。

 
1 package com.coderbuff.custom;
2
3 /**
4 * 函数接口:只有一个方法的接口。作为Lambda表达式的类型
5 * Created by Kevin on 2018/2/17.
6 */
7 public interface FunctionInterface<T> {
8 void test(T param);
9 } 
 

  测试:

 
 1 package com.coderbuff.custom;
2
3 import org.junit.Test;
4
5 /**
6 * 函数接口测试
7 * Created by Kevin on 2018/2/17.
8 */
9 public class FunctionInterfaceTest {
10
11 @Test
12 public void testLambda() {
13 //使用Lambda表达式代替匿名内部类
14 func((Integer x) -> System.out.println("Hello World" + x));
15 }
16
17 private void func(FunctionInterface<Integer> functionInterface) {
18 int x = 1;
19 functionInterface.test(x);
20 }
21 }
 

  上面的示例提到了Lambda表达式的两种情况:

  无参数,无返回值;

  有参数,无返回值。

  接下来就是有参数,有返回值这种较为复杂的情况。

 
1 package com.coderbuff.custom;
2
3 /**
4 * 函数接口:只有一个方法的接口。作为Lambda表达式的类型
5 * Created by Kevin on 2018/2/17.
6 */
7 public interface FunctionInterface<T> {
8 boolean test(T param);
9 }
 

  测试:

 
 1 package com.coderbuff.custom;
2
3 import org.junit.Test;
4
5 /**
6 * 函数接口测试
7 * Created by Kevin on 2018/2/17.
8 */
9 public class FunctionInterfaceTest {
10
11 @Test
12 public void testLambda() {
13 //使用Lambda表达式代替匿名内部类
14 func((Integer x) -> true);
15 }
16
17 private void func(FunctionInterface<Integer> functionInterface) {
18 int x = 1;
19 functionInterface.test(x);
20 }
21 }
 

  此时的Lambda表达式“(Integer x) -> true”,右边是表达式的主体,直接返回true,如果有多行代码,则可以直接使用花括号表示,例如:

func((Integer x) -> {
System.out.println("Hello World" + x);
return true
;
}
);

  Lambda表达式基本的语法规则:

  无参数,无返回值;

  有参数,无返回值;

  有参数,有返回值。

  这三种基本情况已经大致清楚了,特别是需要弄清,什么时候可以使用Lambda表达式代替匿名内部类,也就是Lambda表达式的应用场景是函数接口。Lambda表达式这一新特性在JDK8中的引入,更大的好处则是集合API的更新,新增的Stream类库,使得我们在遍历使用集合时不再像以往那样不断地使用for循环。

JDK8使用集合的正确姿势

  示例:计算来自“chengdu”的学生数量有多少。

  在JDK8前的代码:

for (Student student : studentList) {
if (student.getCity().equals("chengdu")) {
count++;
}
}

  JDK8使用集合的正确姿势:

count = studentList.stream().filter((student -> student.getCity().equals("chengdu"))).count();

  API的使用“难度”恰似提高了,实际只是不熟悉而已。传统迭代的方式需要阅读完整个循环才能明白代码逻辑,JDK8通过流的方式则可以望文生义且代码量大大减小。

  其中最为重要的是——Stream流。Stream的是通过函数式编程方式实现的在集合类上进行复杂操作的工具。若要详细讲解Stream的实现方式我相信再写一篇博客也不为过,所以此处不再考查Stream的内部实现。这里是想告诉大家,如果有幸使用JDK8的开发环境进行开发,尽量学习使用新的集合操作API。

 有关Lambda表达式的应用太多,并发编程、响应式编程等等。

本文引用:http://www.cnblogs.com/yulinfeng/p/8452379.html

最新文章

  1. 从无到有实现登录功能以及thinkphp怎么配置数据库信息
  2. 两个Service之间相互监视的实现
  3. IE奇怪报错
  4. DataTable 怎样设置列宽? DataTable中已经有数据了怎样在现实的时候设置它的列宽?
  5. Centos安装wine等组件的问题
  6. shell中读写mysql数据库
  7. UIButton 详解
  8. jboss之mod_cluster集群
  9. params参数的使用方法
  10. Octave Tutorial(《Machine Learning》)之第一课《数据表示和存储》
  11. LABjs、RequireJS、SeaJS 哪个最好用?为什么?
  12. [sed]命令笔记
  13. Ubuntu下解压压缩文件
  14. linux ubuntu 安装后没有root密码
  15. PTA8
  16. mysql脚本转h2
  17. ceph 重启,停止,开始
  18. 转载 Unity Text 插入超链接
  19. [HTTP]_[C/C++]_[解析URL的转义字符百分比字符串]
  20. Oracle列自增实现(3)-DEFAULT Values Using Sequences

热门文章

  1. angular-Scope
  2. POJ——T1789 Truck History
  3. Wing IDE 怎样设置 python版本号
  4. zzulioj--1777--和尚特烦恼3——何时能下山(水题)
  5. angular4(3)angular脚手架引入scss
  6. 持久层框架Clone
  7. (转)PHP(其他语言类似)编码的规范性
  8. 关于jsp web项目,jsp页面与servlet数据不同步的解决办法(报错404、405等)即访问.jsp和访问web.xml中注册的/servlet/的区别
  9. JavaScript学习——使用JS完成页面定时弹出广告
  10. jquery获取焦点和失去焦点