认识Lambda表达式
  首先来引入一个示例
 new Thread(new Runnable() {
     @Override
     public void run() {
         System.out.println("Hello World!");
         }
     });

  使用Lambda表达式则只需要使用一句话就可代替上面使用匿名类的方式。
       new Thread(() -> System.out.println("Hello World!"));

  在这个例子中,传统的语法规则,是将一个匿名内部类作为参数进行传递,实现了Runnable接口,并将其作为参数传递给Thread类,这实际上传递的是一段代码,也即将代码作为了数据进行传递,这就带来许多不必要的“样板代码”。
  Lambda表达式一共有三部分组成:左边代表参数 -> 右边代表函数主体
  后面的示例中会详解这个结构,包括有无参数,有无返回值的问题。 那么这个看起来奇奇怪怪的不太像Java的语法规则,其本身含义到底什么呢?这也是开始困扰的问题,什么时候在什么场景下可以使用Lambda表达式。
  能够接收Lambda表达式的参数类型,是一个只包含一个方法的接口。只包含一个方法的接口称之为“函数接口”。
  例如上面创建一个线程的示例,Runnable接口只包含一个方法,所以它被称为“函数接口”,所以它可以使用Lambad表达式来代替匿名内部类。根据这个规则,试着来写一个函数接口,并使用Lambda表达式作为参数传递。

package com.coderbuff.custom;
 
 /**
  * 函数接口:只有一个方法的接口。作为Lambda表达式的类型
  */
 public interface FunctionInterface {
     void test();
 }

  测试:
  package com.coderbuff.custom;
 
  import org.junit.Test;
 
  /**
   * 函数接口测试
   */
  public class FunctionInterfaceTest {
 
     @Test
     public void testLambda() {
         func(new FunctionInterface() {
             @Override
             public void test() {
                 System.out.println("Hello World!");
             }
         });
         //使用Lambda表达式代替上面的匿名内部类
         func(() -> System.out.println("Hello World"));
     }
 
     private void func(FunctionInterface functionInterface) {
         functionInterface.test();
     }
 }

  可以看到,只要是一个接口中只包含一个方法,则可以使用Lambda表达式,这样的接口称之为“函数接口”。
  上面的函数接口比较简单不包含参数,也不包含返回值。
  再来修改FunctionInterface函数接口逐步加大Lambda表达式的难度——包含参数,不包含返回值。

package com.coderbuff.custom;
 
 /**
  * 函数接口:只有一个方法的接口。作为Lambda表达式的类型
  */
 public interface FunctionInterface {
     void test(int param);
 }

  测试:
  package com.coderbuff.custom;
 
  import org.junit.Test;
 
  /**
   * 函数接口测试
   */
  public class FunctionInterfaceTest {
 
     @Test
     public void testLambda() {
         //使用Lambda表达式代替匿名内部类
         func((x) -> System.out.println("Hello World" + x));
     }
 
     private void func(FunctionInterface functionInterface) {
         int x = 1;
         functionInterface.test(x);
     }
 }

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

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

package com.coderbuff.custom;
 
 /**
  * 函数接口:只有一个方法的接口。作为Lambda表达式的类型
  */
 public interface FunctionInterface<T> {
     void test(T param);
 } 

  测试:
  package com.coderbuff.custom;
 
  import org.junit.Test;
 
  /**
   * 函数接口测试
   */
  public class FunctionInterfaceTest {
 
     @Test
     public void testLambda() {
         //使用Lambda表达式代替匿名内部类
         func((Integer x) -> System.out.println("Hello World" + x));
     }
 
     private void func(FunctionInterface<Integer> functionInterface) {
         int x = 1;
         functionInterface.test(x);
     }
 }

  上面的示例提到了Lambda表达式的两种情况:
  无参数,无返回值;
  有参数,无返回值。
  接下来就是有参数,有返回值这种较为复杂的情况。

package com.coderbuff.custom;
 
 /**
  * 函数接口:只有一个方法的接口。作为Lambda表达式的类型
  */
 public interface FunctionInterface<T> {
     boolean test(T param);
 }

  测试:
  package com.coderbuff.custom;
 
  import org.junit.Test;
 
  /**
   * 函数接口测试
   */
  public class FunctionInterfaceTest {
 
     @Test
     public void testLambda() {
         //使用Lambda表达式代替匿名内部类
         func((Integer x) -> true);
     }
 
     private void func(FunctionInterface<Integer> functionInterface) {
         int x = 1;
         functionInterface.test(x);
     }
 }

  此时的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通过流的方式则可以望文生义且代码量大大减小。

JDK 8 中Lambda表达式的使用的更多相关文章

  1. Java中lambda表达式详解

    原文地址:http://blog.laofu.online/2018/04/20/java-lambda/ 为什么使用lambda 在java中我们很容易将一个变量赋值,比如int a =0;int ...

  2. Java8中Lambda表达式的10个例子

    Java8中Lambda表达式的10个例子 例1 用Lambda表达式实现Runnable接口 //Before Java 8: new Thread(new Runnable() { @Overri ...

  3. VS编译环境中TBB配置和C++中lambda表达式

    TBB(Thread Building Blocks),线程构建模块,是由Intel公司开发的并行编程开发工具,提供了对Windows,Linux和OSX平台的支持. TBB for Windows ...

  4. Java8 Collections.sort()及Arrays.sort()中Lambda表达式及增强版Comparator的使用

    摘要:本文主要介绍Java8 中Arrays.sort()及Collections.sort()中Lambda表达式及增强版Comparator的使用. 不废话直接上代码 import com.goo ...

  5. JDK中Lambda表达式的序列化与SerializedLambda的巧妙使用

    前提 笔者在下班空余时间想以Javassist为核心基于JDBC写一套摒弃反射调用的轻量级的ORM框架,过程中有研读mybatis.tk-mapper.mybatis-plus和spring-boot ...

  6. Java中Lambda表达式的使用

    简介(译者注:虽然看着很先进,其实Lambda表达式的本质只是一个"语法糖",由编译器推断并帮你转换包装为常规的代码,因此你可以使用更少的代码来实现同样的功能.本人建议不要乱用,因 ...

  7. Java中Lambda表达式的使用(转)

    https://www.cnblogs.com/franson-2016/p/5593080.html 简介(译者注:虽然看着很先进,其实Lambda表达式的本质只是一个"语法糖" ...

  8. Java中Lambda表达式基础及使用详解

    概述 Lambda 是JDK 8 的重要新特性.它允许把函数作为一个方法的参数(函数作为参数传递进方法中),使用 Lambda 表达式可以使代码变的更加简洁紧凑,使Java代码更加优雅. 标准格式 三 ...

  9. Java中Lambda表达式的进化之路

    Lambda表达式的进化之路 为什么要使用Lambda表达式 可以简洁代码,提高代码的可读性 可以避免匿名内部类定义过多导致逻辑紊乱 在原先实现接口抽象方法的时候,需要通过定义一个实现接口的外部类来实 ...

随机推荐

  1. 启动代码和Bootloader区别

    很多人都会把两者混淆,所以这里稍微说一下. 启动代码和Bootloader有点类似于类和结构体的关系,也就是说结构体只是一种很特殊很特殊的类,但不能说类是一种结构体. 也就是说可以说启动代码是Boot ...

  2. WPF Demo19 命令、UC

    命令系统的基本元素和关系WPF命令系统的组成要素:A.命令(command):WPF命令实际上就是实习了ICommand接口的类.平时使用最多的就是RoutedCommand类.B.命令源(comma ...

  3. Azure ARM (18) 将Azure RM Manage Disk托管磁盘的Image,跨订阅迁移

    <Windows Azure Platform 系列文章目录> 先挖一个坑,以后再埋. 最近遇到一个客户需求,客户使用了Azure RM Manage Disk托管磁盘,然后捕获镜像做成了 ...

  4. ALGO-14_蓝桥杯_算法训练_回文数

    问题描述 若一个数(首位不为零)从左向右读与从右向左读都一样,我们就将其称之为回文数. 例如:给定一个10进制数56,将56加65(即把56从右向左读),得到121是一个回文数. 又如:对于10进制数 ...

  5. C++进阶--结构体和类

    // 单纯从语言上来说,两者唯一的区别是,默认成员是公有还是私有 // 从使用习惯上 // 小的消极对象,包含公有数据,没有或仅有很少的基本的成员函数 -- 数据容器 struct Person_t ...

  6. U3d 注意

    设置layer :不要使用go.layer= intMM;要使用Nguitool.Setlayer(go,intLayer); static public void SetLayer (GameObj ...

  7. vc++使用cookie登录网站

    以http://www.idc816.com为例 效果图: 1)先登录获取cookie,再使用cookie获取用户信息 2)记录cookie,关闭程序后重新运行程序,直接获取用户信息 使用Fiddle ...

  8. appium工具 录制的方法

    参考: https://www.jianshu.com/p/5eef1bfa42ae

  9. Java学习——方法中传递参数分简单类型与复杂类型(引用类型)编程计算100+98+96+。。。+4+2+1的值,用递归方法实现

    package hello; public class digui { public static void main(String[] args) { // TODO Auto-generated ...

  10. go语言学习--指针数组和数组指针

    数组指针(也称行指针)定义 int (*p)[n];()优先级高,首先说明p是一个指针,指向一个整型的一维数组,这个一维数组的长度是n,也可以说是p的步长.也就是说执行p+1时,p要跨过n个整型数据的 ...