作为比较老牌的面向对象的编程语言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. Coloring Flame Graphs: Code Hues

    转自:http://www.brendangregg.com/blog/2017-07-30/coloring-flamegraphs-code-type.html I recently improv ...

  2. [luogu4158 SCOI2009] 粉刷匠(dp)

    传送门 Solution 把状态都记上暴力转移即可 Code //By Menteur_Hxy #include <queue> #include <cmath> #inclu ...

  3. OpenCV实现USM锐化与测试

    OpenCV实现USM锐化 [转]http://www.programdevelop.com/4964391/ USM (Unsharp masking) is a common operation ...

  4. 利用Socket 客户端---->服务端 传送文件到指定路径,并返回一个友好的回馈

    首先盲写的一个传输文件的方法,但测试发现了一个非常不容易发现的问题,这里先说明一下. 错误的代码如下: package com.TCP.java; import java.io.File; impor ...

  5. Problem G 宝石合成 (内蒙古14年省赛)

    1117: Problem G 宝石合成 时间限制: 1 Sec  内存限制: 128 MB 提交: 18  解决: 4 [提交][状态][讨论版] 题目描写叙述 故事人物:豆豆强 走上致富之路的豆豆 ...

  6. 《编程导论(Java)&#183;1.4.1 范式》

    这个楼主,是我的学生么?2013年写的! 嗯."编程范式或许是学习不论什么一门编程语言时要理解的最重要的术语".这句话早在2005年出版<Java程序设计>(宋中山,严 ...

  7. Makefile中怎样调用python和perl文件为自己提供须要的数据

    Makefile中怎样调用python和perl文件为自己提供须要的数据,利用print函数对外输出数据 实例代码例如以下 perl.pl #!/usr/bin/perl print("he ...

  8. graphviz.js划线操作

    digraph A{ graph[color=red bgcolor="cadetblue" label="海阔天空",fontname="FangS ...

  9. VS2013+ffmpeg开发环境搭建-->【转】

    本文转载自:http://blog.csdn.net/qq_28425595/article/details/51488869 版权声明:本文为博主原创文章,未经博主允许不得转载. 今天整理资料时,发 ...

  10. php传值调用和传值调用和变量函数

    php传值调用和传值调用和变量函数 代码 <?php //传值调用,$m的值不改变 function text($i){ $i = 'Clive'; echo $i; } text(123); ...