作为比较老牌的面向对象的编程语言java,在对函数式编程的支持上一直不温不火。

认为面向对象式编程就应该纯粹的面向对象,于是经常看到这样的写法:如果你想写一个方法,那么就必须把它放到一个类里面,然后new出来对象,对象调用这个方法。

这种方式在函数式编程语言看来太死板,没有必要在对待多种编程范式上采取非此即彼的做法。

如今比较现代的编程语言也都是多编程范式的支持,不再去对一种编程范式固守一隅,一种语言可能会同时具有面向对象、函数式、元编程等多种特性,这方面java的后来者C#都走在她的前面。

终于在jdk8里发现了lambda表达式的影子,java也开始加入这种函数式编程特性,java码农们终于在之前老土的方法之外有了一种更为简便的选择。

首先来看,lambda之前java的做法: 使用匿名内部类:

public void testAnonymousClass() {
Integer[] nums = {2, 5, 1, 6};
Arrays.sort(nums, new Comparator<Integer>() {
@Override
public int compare(Integer o1, Integer o2) {
if(o1 < o2)
return -1;
return 0;
}
});
for (Integer n : nums) {
System.out.println(n);
}
}

函数式编程语言的做法,这里拿go的代码为例:

package main
import (
"fmt"
)
// 插入排序
func sort(nums []int, compare func (a, b int) int) {
length := len(nums)
for i := length - 1; i >= 0; i-- {
for j := i; j + 1 < length; j++ {
cur := nums[j]
next := nums[j + 1]
if compare(cur, next) > 0 {
nums[j], nums[j + 1] = next, cur
}
}
}
}
func main() {
nums := []int{2, 5, 1, 6}
sort(nums, func(a, b int) int {
if a > b {
return 1
}
return 0
})
fmt.Println(nums)
}

go的代码看上去比较长,由于没有像java一样使用类库提供的排序算法,所以go自己实现的插入排序。 这里go语言具有函数里面传函数的能力(也叫高阶函数),所以代码看起来简洁了很多。一般这种场景,函数式编程语言使用匿名函数的方式,在java的看来就必须通过匿名内部类来实现。首先实现一个接口,接口里面定义好方法,匿名内部类实现接口,然后在传入的函数中,通过传递的对象,实现对匿名内部类里的方法的回调。这也就是lambda表达式之前的基本做法。

lambda表达式是对java实现函数式编程一个取巧方式的补充,下面来看lambda方式的做法:

public void testAnonymousClass() {
Integer[] nums = {2, 5, 1, 6};
Arrays.sort(nums, (o1, o2) -> {
if(o1 < o2)
return -1;
return 0;
});
for (Integer n : nums) {
System.out.println(n);
}
}
函数式接口:这是java在解决函数式编程,引入lambda表达式的同时引入的一个概念,具体的意思就是,定义的一个接口,接口里面必须有且只有一个方法,这样的接口就成为函数式接口。 在可以使用lambda表达式的地方,方法声明时必须包含一个函数式的接口。任何函数式接口都可以使用lambda表达式替换。 下面来看lambda的基本逻辑:
button.onAction(new EventHandler<ActionEvent>() {
@Override
public void handle(ActionEvent e) {
doSomethingWith(e);
}
});

使用lambda表达式替换:

button.onAction((ActionEvent e) -> {
doSomethingWith(e);
});

此lambda表达式的类型可由编译器推断为EventHandler,因为onAction()方法采用的对象类型为 EventHandler。 由于EventHandler只有一个方法即handle(),此lambda表达式必然是handle()方法的实现。 可以继续简化lambda表达式:

button.onAction((e) -> {
doSomethingWith(e);
});

此lambda表达式的参数必须是ActionEvent,因为其类型是由EventHandler接口的 handle()方法指定的。 因此,我们可以简化此lambda表达式,因为其参数类型可以推断。 还可以继续简化:

button.onAction(e -> doSomethingWith(e));

当lambda表达式只有一个参数且参数类型可以推断时,则不需要括号。 lambda表达式中的代码块只包含一个语句,因此可以去掉大括号和分号。

可以猜测lambda表达式的实现可能是由java编译器在编译java字节码时,会翻译这样的语法糖,最终还是转化为匿名内部类来实现,至少从语义上看来是这样的。那么它究竟怎样做到的,这里的文章可以给出答案: 和Lambdas的第一次亲密接触

采用的办法是在使用lambda表达式的类中生成一个实例方法,那么当然能够访问到这个类中定义的实例变量、静态变量和公开、私有方法。 那和函数式编程相随相生的闭包问题是否支持了呢? 通过上面的介绍可以看出java对函数式编程的实现,主要还是在编译时对lambda表达式的一些转化。 让人看起来像是支持了匿名函数等函数式编程的特性,其实还是使用java自己的一套实现。所以在使用lambda表达式的时候最好头脑清醒,不要纠结是否闭包了。 以上谈的是jdk8的预览版本,也可能正式版会做很多的改进,那就不得而知了。

相关文章:

Java 8为什么需要Lambda表达式

Java 8的Lambda表达式

说说Java 8中的Lambda表达式

还可以访问我树莓派上搭的博客地址:

http://www.codeforfun.info/

java函数式编程之lambda表达式的更多相关文章

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

    一:匿名函数的定义 lambda parameter_list: expression 二:三元表达式 条件为真时返回的结果 if 条件判断 else 条件为假的时候返回的结果 三:map map(f ...

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

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

  3. Java 函数式编程(Lambda表达式)与Stream API

    1 函数式编程 函数式编程(Functional Programming)是编程范式的一种.最常见的编程范式是命令式编程(Impera Programming),比如面向过程.面向对象编程都属于命令式 ...

  4. Java 函数式编程和Lambda表达式

    1.Java 8最重要的新特性 Lambda表达式.接口改进(默认方法)和批数据处理. 2.函数式编程 本质上来说,编程关注两个维度:数据和数据上的操作. 面向对象的编程泛型强调让操作围绕数据,这样可 ...

  5. Java函数式接口与Lambda表达式

    什么是函数式接口? 函数式接口是一种特殊的接口,接口中只有一个抽象方法. 函数式接口与Lambda表达式有什么关系? 当需要一个函数式接口的对象时,可以提供一个lambda表达式. package l ...

  6. java函数式编程之Supplier

    原创 2016年12月25日 10:02:52 标签: 函数式编程 / java 3250 描述:Supplier< T>接口没有入参,返回一个T类型的对象,类似工厂方法. 源码: pub ...

  7. python函数式编程之yield表达式形式

    先来看一个例子 def foo(): print("starting...") while True: res = yield print("res:",res ...

  8. java函数式编程之Consumer

    参考https://blog.csdn.net/z345434645/article/details/53794724 https://blog.csdn.net/chuji2012/article/ ...

  9. python 函数式编程之lambda( ), map( ), reduce( ), filter( )

    lambda( ), map( ), reduce( ), filter( ) 1. lambda( )主要用于“行内函数”: f = lambda x : x + 2 #定义函数f(x)=x+2 g ...

随机推荐

  1. 15.6.1 【Task使用】基于任务的异步模式

    C# 5异步函数特性的一大好处是,它为异步提供了一致的方案.但如果在命名异步方法以及 触发异常等方面做法存在着差异,则很容易破坏这种一致性.微软因此发布了基于任务的异步模 式(Task-based A ...

  2. 洛谷P1996 约瑟夫问题【队列】

    题目背景 约瑟夫是一个无聊的人!!! 题目描述 n个人(n<=100)围成一圈,从第一个人开始报数,数到m的人出列,再由下一个人重新从1开始报数,数到m的人再出圈,--依次类推,直到所有的人都出 ...

  3. _markupbase.py if not match: UnboundLocalError: local variable 'match' referenced before assignment,分析Python 库 html.parser 中存在的一个解析BUG

    BUG触发时的完整报错内容(本地无关路径用已经用 **** 隐去): **************\lib\site-packages\bs4\builder\_htmlparser.py:78: U ...

  4. AtCoder ARC 076E - Connected?

    传送门:http://arc076.contest.atcoder.jp/tasks/arc076_c 平面上有一个R×C的网格,格点上可能写有数字1~N,每个数字出现两次.现在用一条曲线将一对相同的 ...

  5. 组合数性质求K个数选取i*j个数分成j组的方案数

    分析:设方案数为ANS,C代表组合数: ANS=(C[K,I]*C[K-I,I][K-2*I,I]*...*C[K-(J-1)*I,I])/(J!); 也即: ANS=C[K,I*J]*(C[I*J, ...

  6. C#使用PowerShell 操作Exchange

    先介绍一篇文章来参考一下 点开文章 该文章的最后使用了SSL来保证账户在连接服务器的时候不发生账户认证错误,但是我经过测试发现这个是不可行的,有一种更为简单的方法 首先要对服务器进行winrm设置 就 ...

  7. Project Euler18题 从上往下邻接和

    题目:By starting at the top of the triangle below and moving to adjacent numbers on the row below, the ...

  8. springmvc 监听器getWriter() has already been called for this response问题

    springmvc 监听器getWriter() has already been called for this response问题 在监听器中,如果return true,就不要使用 respo ...

  9. [Javascript] Highlights from IO18 Javascript new features

    Latest Javascript features, not supported by all broswers, but can use with babel. # try-catch-final ...

  10. &#39;hibernate.dialect&#39; must be set when no Connection available

    今天碰到的这个问题,非常无厘头.网上搜索了非常多.都不靠谱,还是靠自己 解决方法是在hibernate.cfg.xml中加入 <property name="dialect" ...