Lambda 表达式是 Java 8 最受欢迎的功能。人们将函数式编程的概念引入了 Java 这门完全面向对象的命令式编程语言。

关于函数式编程是如何运作的,这个话题超出了本文的范围,不过我们会提炼出它一个明显有别于我们所经常使用的 OOP (面向对象编程)的功能来加以讨论。

在本文中, 我们将了解到 lambda 表达式具体是什么东西,还有就是它们是如何将自己融入整个 Java 生态系统的。我们也会对没有使用 lambda 表达式的代码以及后面使用 lambda 进行重构的示例代码进行一下观察和比较。

了解 Lambda 表达式

Lambda 表达式是一块我们可以将其传入并执行的代码。对于作为 Java 程序员的我们而言,并不会怎么习惯将一块代码传入一个函数这样的方式。我们的习惯是将定义的代码封装到方法体里面,然后通过对象引用来加以执行,如下所示:

public class LambdaDemo {
    public void printSomething(String something) {
        System.out.println(something);
    }

    public static void main(String[] args) {
        LambdaDemo demo = new LambdaDemo();
        String something = "I am learning Lambda";
        demo.printSomething(something);
    }
}

 这是经典 OOP 开发范式的风格,将方法实现对调用者隐藏。调用者只是简单地向方法传入一个变量,然后方法拿这个变量会执行一些操作,并返回另外一个变量值,或者如我们的示例所示,会产生一些副作用效果。

  现在我们要来看看一种使用了行为传递方式,而不是变量传递的等效实现。为此,我们要创建一个函数式的接口,里面定义的是对行为,而不是对方法的抽象。一个函数式接口是一种只有一个方法的接口:

public class LambdaDemo {
    interface Printer {
        void print(String val);
    }

    public void printSomething(String something, Printer printer) {
        printer.print(something);
    }
}

在上面的代码实现中, Printer 接口负责所有的打印操作。printSomething 方法不再对行为进行定义,而是执行由 Printer 定义的行为:

public static void main(String[] args) {
    LambdaDemo demo = new LambdaDemo();
    String something = "I am using a Functional interface";
    Printer printer = new Printer() {
        @Override
        public void print(String val) {
            System.out.println(val);
        }
    };
    demo.printSomething(something, printer);
}

读者中比较有观察能力的可能已经注意到,我并没有在这里做什么新的事情。的确是这样的,因为我还没有应用到 lambda 表达式。我们只是简单地创建了一个 Printer 接口的具体实现,并将它传入了 printSomething 方法。

  上面的示例旨在给我们带来一个将 Lambda 表达式引入到 Java 中的关键目标:

    Lambda 表达式原被用于定义一个函数式接口的内联实现。


在我们使用 lambda 表达式对上面的示例进行重构之前,先来学习一下必要的语法知识:

(param1,param2,param3...,paramN) - > {//代码块;}

一个 lambda 表达式的组成,是一个我们通常会定义在方法声明中的,以括弧封闭起来并以逗号分隔的参数列表,后面跟上一个箭头标记指向要执行的代码。现在,让我们来使用 lambda 对上面的代码进行重构:

public static void main(String[] args) {
    LambdaDemo demo = new LambdaDemo();
    String something = "I am learning Lambda";
    /**/
    Printer printer = (String toPrint)->{System.out.println(toPrint);};
    /**/
    demo.printSomething(something, printer);
}

看上去非常紧凑且美观。因为函数式接口只声明了一个方法,所以在 lambda 的第一部分中传入的参数被自动地映射到了方法的参数列表上,而箭头右边的代码则被当做是方法的具体实现了。

 为什么要使用 Lambda 表达式

如同前面的示例, lambda 表达式能让我们拥有更加紧凑的代码,更加易于阅读和跟踪。这个在性能和多核处理方法还有其它的一些好处,不过它们得在你了解了 Streams API 以后才有用,而这个超出了本文的范围。

  通过比较使用和没使用 lambda 的 main 方式实现,当它一下子把代码变得简短的时候,我们切实地看到了 lambda 表达式的能力:

public static void main(String[] args) {
    LambdaDemo demo = new LambdaDemo();
    String something = "I am learning Lambda";
    /**/
    Printer printer = (String toPrint)->{System.out.println(toPrint);};
    /**/
    demo.printSomething(something, printer);
}

我们还可以让代码比这里所展示的更简洁。这样的事情发生时,你甚至无需指定箭头左边参数的类型,而其类型会由编译器根据接口方法的形参推断出来。

Printer printer = (toPrint)->{System.out.println(toPrint);};

我们还可以做得更好。lambda 的另外一个特性就是: 如果只有一个参数, 就可以将括弧完全消除掉。同样的,如果在箭头右边只有一条语句,也可以将大括号去掉:

Printer printer = toPrint -> System.out.println(toPrint);

现在的代码看起来真正变得可爱起来,不过我们才刚刚开始而已。如果我们的接口方法并不要任何参数,那就可以将生命用一对空的括弧替换掉:

() -> System.out.println("anything");

如果我们只是内联一个 lambda 进去,而不去首先创建一个对象然后将其传入到 saySomething 方法,会如何呢:

public static void main(String[] args) {
    LambdaDemo demo = new LambdaDemo();
    String something="I am Lambda";
    /**/
    demo.printSomething(something, toPrint -> System.out.println(toPrint));
}

现在我们才是真的在谈论函数式编程了。我们的 main 函数体从一开始的 9 行代码减少到了 3 行。这样紧凑的代码使得 lambda 表达式对于 Java 程序员非常有吸引力。

Java 8 中的 Lambda 表达式的更多相关文章

  1. java函数式编程之lambda表达式

    作为比较老牌的面向对象的编程语言java,在对函数式编程的支持上一直不温不火. 认为面向对象式编程就应该纯粹的面向对象,于是经常看到这样的写法:如果你想写一个方法,那么就必须把它放到一个类里面,然后n ...

  2. Lambda 表达式,Java中应用Lambda 表达式

    一.Lambda 表达式 简单来说,编程中提到的 lambda 表达式,通常是在需要一个函数,但是又不想费神去命名一个函数的场合下使用,也就是指匿名函数. 链接:知乎 先举一个普通的 Python 例 ...

  3. Java中的Lambda表达式简介及应用

    在接触Lambda表达式.了解其作用之前,首先来看一下,不用Lambda的时候我们是怎么来做事情的. 我们的需求是,创建一个动物(Animal)的列表,里面有动物的物种名,以及这种动物是否会跳,是否会 ...

  4. C#中的Lambda表达式和表达式树

    在C# 2.0中,通过方法组转换和匿名方法,使委托的实现得到了极大的简化.但是,匿名方法仍然有些臃肿,而且当代码中充满了匿名方法的时候,可读性可能就会受到影响.C# 3.0中出现的Lambda表达式在 ...

  5. [Java 8] (5) 使用Lambda表达式进行设计

    使用Lambda表达式进行设计 在前面的几篇文章中,我们已经见识到了Lambda表达式是怎样让代码变的更加紧凑和简洁的. 这一篇文章主要会介绍Lambda表达式怎样改变程序的设计.怎样让程序变的更加轻 ...

  6. Java 高级特性: Lambda 表达式

    本文源代码上传到了码云,请点击 LambdaExpression 获取.Lambda 表达式是 java 8 的新特性,本文讲解了 lambda 表达式的所有知识.内容涉及到 lambda 表达式是什 ...

  7. Java函数式编程和lambda表达式

    为什么要使用函数式编程 函数式编程更多时候是一种编程的思维方式,是种方法论.函数式与命令式编程的区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉代码要怎么做.说白了,函数式编程是基于 ...

  8. Android中使用Lambda表达式开发

    参考文章:ImportNew 要在Android开发中使用lambda表达式,首先需要在 Module 的build.gradle中加入: compileOptions { targetCompati ...

  9. Java核心技术-接口、lambda表达式与内部类

    本章将主要介绍: 接口技术:主要用来描述类具有什么功能,而并不给出每个功能的具体实现.一个类可以实现一个或多个接口. lambda表达式:这是一种表示可以在将来的某个时间点执行的代码块的简洁方法. 内 ...

随机推荐

  1. Oracle 11g R2 for Win10(64位)的安装步骤

    下载 官网下载地址: win64_11gR2_database_1of2.zip win64_11gR2_database_2of2.zip 将两个压缩包解压到同一个目录下,即"databa ...

  2. hibernate多表操作

    一.表之间的关系 1.一对一 2.一对多 3.多对多 二.表之间关系建表原则 1.一对多:在多的一方创建一个外键,指向一的一方的主键 2.多对多:创建一个中间表,中间表至少有两个字段,分别作为外键指向 ...

  3. Django学习手册 - csrf

    CSRF csrf原理 无csrf时存在隐患 Form提交 Ajax提交 默认为全局都csrf Form表单提交方式: <div> <form action="/login ...

  4. python模块-----time

    说明 time模块提供各种时间相关的功能 与时间相关的模块有:time,datetime,calendar 这个模块的功能不是适用于所有的平台 这个模块中定义的大部分函数是调用C平台上的同名函数实现 ...

  5. NDT 算法和一些常见配准算法

    原文链接:http://ghx0x0.github.io/2014/12/30/NDT-match/ 目前三维配准中用的较多的是ICP迭代算法,需要提供一个较好的初值,同时由于算法本身缺陷,最终迭代结 ...

  6. 2018 Multi-University Training Contest 9 杭电多校第九场 (有坑待补)

    咕咕咕了太久  多校博客直接从第三场跳到了第九场orz 见谅见谅(会补的!) 明明最后看下来是dp场 但是硬生生被我们做成了组合数专场…… 听说jls把我们用组合数做的题都用dp来了遍 这里只放了用组 ...

  7. 【python网络爬虫】之requests相关模块

    python网络爬虫的学习第一步 [python网络爬虫]之0 爬虫与反扒 [python网络爬虫]之一 简单介绍 [python网络爬虫]之二 python uillib库 [python网络爬虫] ...

  8. seo 优化排名 使用总结

    SEO 的优化技巧 随着百度对竞价排名位置的大幅减少,SEO优化将自己的网站在首页上有更好的展示有了更多的可能. 本文将系统阐述SEO优化原理.优化技巧和优化流程. 搜索引擎的优化原理是蜘蛛过来抓取网 ...

  9. 【转】Python数据类型之“文本序列(Text Sequence)”

    [转]Python数据类型之“文本序列(Text Sequence)” Python中的文本序列类型 Python中的文本数据由str对象或字符串进行处理. 1.字符串 字符串是Unicode码值的不 ...

  10. Unity-Rigidbody碰撞穿透

    首先,说说碰撞的条件:1.rigidbody(刚体),一般用在主动移动的物体上,比如角色.2.collider,碰撞器,一般用于受力物体上,比如障碍块. 发生概率即触发方式: 1.刚体速度足够快,被撞 ...