Java 8 发布带来的一个主要特性就是对函数式编程的支持。

而 Lambda 表达式就是一个新的并且很重要的一个概念。

它提供了一个简单并且很简洁的编码方式。

首先从几个简单的 Lambda 表达式的例子开始了解 Java 中的函数式编程。

Lambda 表达式初识:

首先定义一个 Lambda 表达式:

x -> x + 1

这个表达式输入参数是一个 x,然后对这个参数 x 的操作是加 1,然后将这个结果返回,即返回值。

从这个简单的 Lambda 表达式可以看出 Lambda 表达式的语法格式是:

(类型 参数1, 类型 参数2) -> { 方法体 }

参数的小括号可以省略不写;

方法体的花括号也可以省略不写(这是因为方法体有两种类型:单行代码 或者是 代码块,单行代码不需要花括号同时 return 关键字省略),例如:

(int x, int y) -> { return x + y; }
(int x, int y) -> x + y
x -> x + 1
() -> 3.14
(String s) -> { System.out.println(s); }

第一个表达式接收两个参数 x 和 y,返回 x + y;

第二个表达式和第一个表达式一样,省略了花括号和 return 关键字;

第三个表达式接收一个参数,返回 x+1;

第四个表达式不接收任何参数,返回数字 3.14;

第五个表达式接收一个参数,没有返回值。

看到第三个表达式,很多新接触 Lambda 表达式的朋友不免会有两个疑问:x 的类型是什么?这个整个函数的类型是什么呢,怎么表示它?

接下来就要引入另一个关键的内容,即 java.util.function 包,官方对它的定义是:“Functional interfaces provide target types for lambda expressions and method references. ” 即为定义函数对象提供的类,也就是如何存储一个函数对象。也就是它专门用来解决上面提出的这个问题:

回到刚刚的这个 Lambda 表达式:x -> x + 1,这个表达式接收一个参数,返回一个参数,我们可以将这个 Lambda 表达式表示为:

Function<Integer, Integer> add = x -> x + 1;
Function<String, String> concat = x -> x + 1;

第一行代码里的 x 的类型是 Integer,它表示将 Integer 类型的参数 x 加 1 然后并返回,即如果参数是 2,那么这行代码返回 3。

第二行代码里的 x 的类型是 String,它表示将 String 类型的参数 x 和 "1" 连接并返回,即如果参数是 "hi", 那么这行代码返回 hi1。

 

两个参数:

如果我们需要定义两个参数的函数要怎么写呢?比如将两个 Integer 类型的 x,y 进行求和并返回结果?

这里我们需要另一个类 BiFunction<T, U, R>,T表示第一个输入参数的类型,U表示第二个输入参数的类型,R表示返回值的类型,所以我们的答案是:

BiFunction<Integer, Integer, Integer> sum = (x, y) -> x + y;

 

只输入参数不返回结果:

如果我们需要定义一个只输入参数却不返回结果的函数要怎么写?这里我们需要另一个类 Consumer<T>:

Consumer<String> sayHi = name -> System.out.println("hi, " + name);

如果想要定义输入两个参数却不返回结果的函数需要用 BiConsumer<T, U> 类,用法和前面提到的 BiFunction 类型一样,不再赘述。

只返回结果不输入参数:

如果我们需要定义一个只返回结果但不需要输入参数的函数要怎么写?这里我们需要另一个类 Supplier<T>:

Supplier<String> getName = () -> "Carl";

如何执行函数:

既然有了函数,那么这些函数要怎么执行呢?

答案是用 apply() 方法:

nteger result = add.apply(2);  // 返回 3
String answer = concat.apply("hi"); // 返回 "hi1"
Integer total = sum.apply(1, 2); // 返回 3
String name = getName.apply(); // 返回 "Carl"

结束语:

之前一直用 C#、Objective-C、Swift 和 Python,因为今年研究生阶段的课程以 Java 语言为主,所以开始了解一下 Java。

感兴趣的朋友欢迎关注本博客,也欢迎大家留言讨论。

参考资料:

Beginning Java objects by Jacquie Barker

Java SE 8: Lambda Quick Start

The big change in Java SE 8

Functional Programming with Java 8 Functions

分享一张半个月前美国小镇的雪景:

Java 中的函数式编程(Functional Programming):Lambda 初识的更多相关文章

  1. 关于函数式编程(Functional Programming)

    初学函数式编程,相信很多程序员兄弟们对于这个名字熟悉又陌生.函数,对于程序员来说并不陌生,编程对于程序员来说也并不陌生,但是函数式编程语言(Functional Programming languag ...

  2. Java中的函数式编程(二)函数式接口Functional Interface

    写在前面 前面说过,判断一门语言是否支持函数式编程,一个重要的判断标准就是:它是否将函数看做是"第一等公民(first-class citizens)".函数是"第一等公 ...

  3. Java中的函数式编程(三)lambda表达式

    写在前面 lambda表达式是一个匿名函数.在Java 8中,它和函数式接口一起,共同构建了函数式编程的框架.   lambda表达式乍看像是匿名内部类的一种语法糖,但实际上,它们是两种本质不同的事物 ...

  4. Java中的函数式编程(五)Java集合框架中的高阶函数

    写在前面 随着Java 8引入了函数式接口和lambda表达式,Java 8中的集合框架(Java Collections Framework, JCF)也增加相应的接口以适应函数式编程.   本文的 ...

  5. Java中的函数式编程(六)流Stream基础

    写在前面 如果说函数式接口和lambda表达式是Java中函数式编程的基石,那么stream就是在基石上的最富丽堂皇的大厦. 只有熟悉了stream,你才能说熟悉了Java 的函数式编程. 本文主要介 ...

  6. Java中的函数式编程(八)流Stream并行编程

    写在前面 在本系列文章的第一篇,我们提到了函数式编程的优点之一是"易于并发编程". Java作为一个多线程的语言,它通过 Stream 来提供了并发编程的便利性. 题外话: 严格来 ...

  7. 函数式编程 - Functional Programming

    什么是函数式编程 函数式编程是一种编程范式. 编程范式又是什么? 编程范式是一种解决问题的思路. 命令式编程 把程序看作 一系列改变状态的指令: 函数式编程 把程序看作 一系列数学函数映射的组合. i ...

  8. Java中的函数式编程(七)流Stream的Map-Reduce操作

    写在前面 Stream 的 Map-Reduce 操作是Java 函数式编程的精华所在,同时也是最为复杂的部分.但一旦你啃下了这块硬骨头,那你就真正熟悉Java的函数式编程了. 如果你有大数据的编程经 ...

  9. Java中的函数式编程(四)方法引用method reference

    写在前面 我们已经知道,lambda表达式是一个匿名函数,可以用lambda表达式来实现一个函数式接口.   很自然的,我们会想到类的方法也是函数,本质上和lambda表达式是一样的,那是否也可以用类 ...

随机推荐

  1. 九度OJ 1205 N阶楼梯上楼问题 (DP)

    题目1205:N阶楼梯上楼问题 时间限制:1 秒 内存限制:128 兆 特殊判题:否 提交:2817 解决:1073 题目描写叙述: N阶楼梯上楼问题:一次能够走两阶或一阶.问有多少种上楼方式. (要 ...

  2. Order笔记-数据库创建

    过程: 1,为这个项目新建一个用户名(实例),专门用于这个项目 2,建表 问题: 列在此处不允许: 笔记: 建表设置默认值: alter table 表名 modify 字段名 default 默认值 ...

  3. Learning to Rank算法介绍:GBRank

    之前的博客:http://www.cnblogs.com/bentuwuying/p/6681943.html中简单介绍了Learning to Rank的基本原理,也讲到了Learning to R ...

  4. nginx配置(windows配置)

    以下是我的项目用到的一份配置文件#user  nobody;worker_processes  4; #进程数,一般cpu是几核就写多少#error_log  logs/error.log;#erro ...

  5. select的限制以及poll的使用

    1.先说select在多路IO中的限制:1)linux中每个程序能够打开的最多文件描述符是有限制的.默认是1024.可以通过ulimit -n进行查看和修改: xcy@xcy-virtual-mach ...

  6. 详解功能版本管理之使用eoLinker

    先看一个对话: "这里,你改一下,这里返回一个object." "好...好......" "还有这里,返回个String." ...... ...

  7. IX-Protected Dataplane Operating System解读

    一.概述 商业操作系统在应用程序每秒钟需要数百万次操作时才能保持高吞吐量和低(尾)延迟,对于最慢的请求只需几百微秒.通常认为对于高性能网络(小信息的高包率.低延迟)的构建,最好都是在内核之外构建用户态 ...

  8. PHP中静态方法和实例化方法的区别

    在PHP中类为什么要使用静态方法,有什么好处 不需要实例化?? 可以提高运行效率?? 这是一个经常被时时提出来的问题,很多时候我们以为理解了.懂了,但深究一下,我们却发现并不懂. 方法是我们每天都在写 ...

  9. SpringMVC底层数据传输校验的方案

    团队的项目正常运行了很久,但近期偶尔会出现BUG.目前观察到的有两种场景:一是大批量提交业务请求,二是生成批量导出文件.出错后,再执行一次就又正常了. 经过跟踪日志,发现是在Server之间进行jso ...

  10. 模板引擎(smarty)知识点总结

    首先我们必须知道使用smarty的流程   1.引入   2.实例化   3.配置 模板目录  编译目录  3.0版本 支持不存在的目录则新建   4.赋值   5.编译显示 某个模板文件(暗示需要哪 ...