为什么要使用函数式编程

函数式编程更多时候是一种编程的思维方式,是种方法论。函数式与命令式编程的区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉代码要怎么做。说白了,函数式编程是基于某种语法或调用API去进行编程。例如,我们现在需要从一组数字中,找出最小的那个数字,若使用用命令式编程实现这个需求的话,那么所编写的代码如下:

public static void main(String[] args) {
int[] nums = new int[]{1, 2, 3, 4, 5, 6, 7, 8}; int min = Integer.MAX_VALUE;
for (int num : nums) {
if (num < min) {
min = num;
}
}
System.out.println(min);
}

而使用函数式编程进行实现的话,所编写的代码如下:

public static void main(String[] args) {
int[] nums = new int[]{1, 2, 3, 4, 5, 6, 7, 8}; int min = IntStream.of(nums).min().getAsInt();
System.out.println(min);
}

从以上的两个例子中,可以看出,命令式编程需要自己去实现具体的逻辑细节。而函数式编程则是调用API完成需求的实现,将原本命令式的代码写成一系列嵌套的函数调用,在函数式编程下显得代码更简洁、易懂,这就是为什么要使用函数式编程的原因之一。所以才说函数式编程是告诉代码你要做什么,而命令式编程则是告诉代码要怎么做,是一种思维的转变。

说到函数式编程就不得不提一下lambda表达式,它是函数式编程的基础。在Java还不支持lambda表达式时,我们需要创建一个线程的话,需要编写如下代码:

public static void main(String[] args) {
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("running");
}
}).start();
}

而使用lambda表达式一句代码就能完成线程的创建,lambda强调了函数的输入输出,隐藏了过程的细节,并且可以接受函数当作输入(参数)和输出(返回值):

public static void main(String[] args) {
new Thread(() -> System.out.println("running")).start();
}

注:箭头的左边是输入,右边则是输出。

该lambda表达式的作用其实就是返回了Runnable接口的实现对象,这与我们调用某个方法获取实例对象类似,只不过是将实现代码直接写在了lambda表达式里。我们可以做个简单的对比:

public static void main(String[] args) {
Runnable runnable1 = () -> System.out.println("running");
Runnable runnable2 = RunnableFactory.getInstance();
}

JDK8接口新特性

1.函数接口,接口只能有一个需要实现的方法,可以使用@FunctionalInterface 注解进行声明。如下:

@FunctionalInterface
interface Interface1 {
int doubleNum(int i);
}

使用lambda表达式获取该接口的实现实例的几种写法:

public static void main(String[] args) {
// 最常见的写法
Interface1 i1 = (i) -> i * 2;
Interface1 i2 = i -> i * 2; // 可以指定参数类型
Interface1 i3 = (int i) -> i * 2; // 若有多行代码可以这么写
Interface1 i4 = (int i) -> {
System.out.println(i);
return i * 2;
};
}

2.比较重要的一个接口特性是接口的默认方法,用于提供默认实现。默认方法和普通实现类的方法一样,可以使用this等关键字:

@FunctionalInterface
interface Interface1 {
int doubleNum(int i); default int add(int x, int y) {
return x + y;
}
}

之所以说默认方法这个特性比较重要,是因为我们借助这个特性可以在以前所编写的一些接口上提供默认实现,并且不会影响任何的实现类以及既有的代码。例如我们最熟悉的List接口,在JDK1.2以来List接口就没有改动过任何代码,到了1.8之后才使用这个新特性增加了一些默认实现。这是因为如果没有默认方法的特性的话,修改接口代码带来的影响是巨大的,而有了默认方法后,增加默认实现可以不影响任何的代码。

3.当接口多重继承时,可能会发生默认方法覆盖的问题,这时可以去指定使用哪一个接口的默认方法实现,如下示例:

@FunctionalInterface
interface Interface1 {
int doubleNum(int i); default int add(int x, int y) {
return x + y;
}
} @FunctionalInterface
interface Interface2 {
int doubleNum(int i); default int add(int x, int y) {
return x + y;
}
} @FunctionalInterface
interface Interface3 extends Interface1, Interface2 { @Override
default int add(int x, int y) {
// 指定使用哪一个接口的默认方法实现
return Interface1.super.add(x, y);
}
}

函数接口

我们本小节来看看JDK8里自带了哪些重要的函数接口:



可以看到上表中有好几个接口,而其中最常用的是Function接口,它能为我们省去定义一些不必要的函数接口,减少接口的数量。我们使用一个简单的例子演示一下 Function 接口的使用:

import java.text.DecimalFormat;
import java.util.function.Function; class MyMoney {
private final int money; public MyMoney(int money) {
this.money = money;
} public void printMoney(Function<Integer, String> moneyFormat) {
System.out.println("我的存款: " + moneyFormat.apply(this.money));
}
} public class MoneyDemo {
public static void main(String[] args) {
MyMoney me = new MyMoney(99999999); Function<Integer, String> moneyFormat = i -> new DecimalFormat("#,###").format(i);
// 函数接口支持链式操作,例如增加一个字符串
me.printMoney(moneyFormat.andThen(s -> "人民币 " + s));
}
}

运行以上例子,控制台输出如下:

我的存款: 人民币 99,999,999

若在这个例子中不使用Function接口的话,则需要自行定义一个函数接口,并且不支持链式操作,如下示例:

import java.text.DecimalFormat;

// 自定义一个函数接口
@FunctionalInterface
interface IMoneyFormat {
String format(int i);
} class MyMoney {
private final int money; public MyMoney(int money) {
this.money = money;
} public void printMoney(IMoneyFormat moneyFormat) {
System.out.println("我的存款: " + moneyFormat.format(this.money));
}
} public class MoneyDemo {
public static void main(String[] args) {
MyMoney me = new MyMoney(99999999); IMoneyFormat moneyFormat = i -> new DecimalFormat("#,###").format(i);
me.printMoney(moneyFormat);
}
}

然后我们再来看看Predicate接口和Consumer接口的使用,如下示例:

public static void main(String[] args) {
// 断言函数接口
Predicate<Integer> predicate = i -> i > 0;
System.out.println(predicate.test(-9)); // 消费函数接口
Consumer<String> consumer = System.out::println;
consumer.accept("这是输入的数据");
}

运行以上例子,控制台输出如下:

false
这是输入的数据

这些接口一般有对基本类型的封装,使用特定类型的接口就不需要去指定泛型了,如下示例:

public static void main(String[] args) {
// 断言函数接口
IntPredicate intPredicate = i -> i > 0;
System.out.println(intPredicate.test(-9)); // 消费函数接口
IntConsumer intConsumer = (value) -> System.out.println("输入的数据是:" + value);
intConsumer.accept(123);
}

运行以上代码,控制台输出如下:

false
输入的数据是:123

有了以上接口示例的铺垫,我们应该对函数接口的使用有了一个初步的了解,接下来我们演示剩下的函数接口使用方式:

public static void main(String[] args) {
// 提供数据接口
Supplier<Integer> supplier = () -> 10 + 1;
System.out.println("提供的数据是:" + supplier.get()); // 一元函数接口
UnaryOperator<Integer> unaryOperator = i -> i * 2;
System.out.println("计算结果为:" + unaryOperator.apply(10)); // 二元函数接口
BinaryOperator<Integer> binaryOperator = (a, b) -> a * b;
System.out.println("计算结果为:" + binaryOperator.apply(10, 10));
}

运行以上代码,控制台输出如下:

提供的数据是:11
计算结果为:20
计算结果为:100

而BiFunction接口就是比Function接口多了一个输入而已,如下示例:

class MyMoney {
private final int money;
private final String name; public MyMoney(int money, String name) {
this.money = money;
this.name = name;
} public void printMoney(BiFunction<Integer, String, String> moneyFormat) {
System.out.println(moneyFormat.apply(this.money, this.name));
}
} public class MoneyDemo {
public static void main(String[] args) {
MyMoney me = new MyMoney(99999999, "小明"); BiFunction<Integer, String, String> moneyFormat = (i, name) -> name + "的存款: " + new DecimalFormat("#,###").format(i);
me.printMoney(moneyFormat);
}
}

运行以上代码,控制台输出如下:

小明的存款: 99,999,999

方法引用

在学习了lambda表达式之后,我们通常会使用lambda表达式来创建匿名方法。但有的时候我们仅仅是需要调用一个已存在的方法。如下示例:

Arrays.sort(stringsArray, (s1, s2) -> s1.compareToIgnoreCase(s2));

在jdk8中,我们可以通过一个新特性来简写这段lambda表达式。如下示例:

Arrays.sort(stringsArray, String::compareToIgnoreCase);

这种特性就叫做方法引用(Method Reference)。方法引用的标准形式是:类名::方法名。(注意:只需要写方法名,不需要写括号)。

目前方法引用共有以下四种形式:

类型 示例 代码示例 对应的Lambda表达式
引用静态方法 ContainingClass::staticMethodName String::valueOf (s) -> String.valueOf(s)
引用某个对象的实例方法 containingObject::instanceMethodName x::toString() () -> this.toString()
引用某个类型的任意对象的实例方法 ContainingType::methodName String::toString (s) -> s.toString
引用构造方法 ClassName::new String::new () -> new String()

下面我们用一个简单的例子来演示一下方法引用的几种写法。首先定义一个实体类:

public class Dog {
private String name = "二哈";
private int food = 10; public Dog() {
} public Dog(String name) {
this.name = name;
} public static void bark(Dog dog) {
System.out.println(dog + "叫了");
} public int eat(int num) {
System.out.println("吃了" + num + "斤");
this.food -= num;
return this.food;
} @Override
public String toString() {
return this.name;
}
}

通过方法引用来调用该实体类中的方法,代码如下:

package org.zero01.example.demo;

import java.util.function.*;

/**
* @ProjectName demo
* @Author: zeroJun
* @Date: 2018/9/21 13:09
* @Description: 方法引用demo
*/
public class MethodRefrenceDemo { public static void main(String[] args) {
// 方法引用,调用打印方法
Consumer<String> consumer = System.out::println;
consumer.accept("接收的数据"); // 静态方法引用,通过类名即可调用
Consumer<Dog> consumer2 = Dog::bark;
consumer2.accept(new Dog()); // 实例方法引用,通过对象实例进行引用
Dog dog = new Dog();
IntUnaryOperator function = dog::eat;
System.out.println("还剩下" + function.applyAsInt(2) + "斤"); // 另一种通过实例方法引用的方式,之所以可以这么干是因为JDK默认会把当前实例传入到非静态方法,参数名为this,参数位置为第一个,所以我们在非静态方法中才能访问this,那么就可以通过BiFunction传入实例对象进行实例方法的引用
Dog dog2 = new Dog();
BiFunction<Dog, Integer, Integer> biFunction = Dog::eat;
System.out.println("还剩下" + biFunction.apply(dog2, 2) + "斤"); // 无参构造函数的方法引用,类似于静态方法引用,只需要分析好输入输出即可
Supplier<Dog> supplier = Dog::new;
System.out.println("创建了新对象:" + supplier.get()); // 有参构造函数的方法引用
Function<String, Dog> function2 = Dog::new;
System.out.println("创建了新对象:" + function2.apply("旺财"));
}
}

最后需要说一句的就是能够使用方法引用的地方就尽量不要使用lambda表达式,这样就不会多生成一个类似 lambda$0这样的函数,能够减少一些资源的开销。

类型推断

通过以上的例子,我们知道之所以能够使用Lambda表达式的依据是必须有相应的函数接口。这一点跟Java是强类型语言吻合,也就是说你并不能在代码的任何地方任性的写Lambda表达式。实际上Lambda的类型就是对应函数接口的类型。Lambda表达式另一个依据是类型推断机制,在上下文信息足够的情况下,编译器可以推断出参数表的类型,而不需要显式指名。

所以说 Lambda 表达式的类型是从 Lambda 的上下文推断出来的,上下文中 Lambda 表达式需要的类型称为目标类型,如下图所示:



接下来我们使用一个简单的例子,演示一下 Lambda 表达式的几种类型推断,首先定义一个简单的函数接口:

@FunctionalInterface
interface IMath {
int add(int x, int y);
}

示例代码如下:

public class TypeDemo {

    public static void main(String[] args) {
// 1.通过变量类型定义
IMath iMath = (x, y) -> x + y; // 2.数组构建的方式
IMath[] iMaths = {(x, y) -> x + y}; // 3.强转类型的方式
Object object = (IMath) (x, y) -> x + y; // 4.通过方法返回值确定类型
IMath result = createIMathObj(); // 5.通过方法参数确定类型
test((x, y) -> x + y); } public static IMath createIMathObj() {
return (x, y) -> x + y;
} public static void test(IMath iMath){
return;
}
}

变量引用

Lambda表达式类似于实现了指定接口的内部类或者说匿名类,所以在Lambda表达式中引用变量和我们在匿名类中引用变量的规则是一样的。如下示例:

public static void main(String[] args) {
String str = "当前的系统时间戳是: ";
Consumer<Long> consumer = s -> System.out.println(str + s);
consumer.accept(System.currentTimeMillis());
}

值得一提的是,在JDK1.8之前我们一般会将匿名类里访问的外部变量设置为final,而在JDK1.8里默认会将这个匿名类里访问的外部变量给设置为final。例如我现在改变str变量的值,ide就会提示错误:



至于为什么要将变量设置final,这是因为在Java里没有引用传递,变量都是值传递的。不将变量设置为final的话,如果外部变量的引用被改变了,那么最终得出来的结果就会是错误的。

下面用一组图片简单演示一下值传递与引用传递的区别。以列表为例,当只是值传递时,匿名类里对外部变量的引用是一个值对象:



若此时list变量指向了另一个对象,那么匿名类里引用的还是之前那个值对象,所以我们才需要将其设置为final防止外部变量引用改变:



而如果是引用传递的话,匿名类里对外部变量的引用就不是值对象了,而是指针指向这个外部变量:



所以就算list变量指向了另一个对象,匿名类里的引用也会随着外部变量的引用改变而改变:

级联表达式和柯里化

在函数式编程中,函数既可以接收也可以返回其他函数。函数不再像传统的面向对象编程中一样,只是一个对象的工厂或生成器,它也能够创建和返回另一个函数。返回函数的函数可以变成级联 lambda 表达式,特别值得注意的是代码非常简短。尽管此语法初看起来可能非常陌生,但它有自己的用途。

级联表达式就是多个lambda表达式的组合,这里涉及到一个高阶函数的概念,所谓高阶函数就是一个可以返回函数的函数,如下示例:

// 实现了 x + y 的级联表达式
Function<Integer, Function<Integer, Integer>> function1 = x -> y -> x + y;
System.out.println("计算结果为: " + function1.apply(2).apply(3)); // 计算结果为: 5

这里的 y -> x + y 是作为一个函数返回给上一级表达式,所以第一级表达式的输出是 y -> x + y这个函数,如果使用括号括起来可能会好理解一些:

x -> (y -> x + y)

级联表达式可以实现函数柯里化,简单来说柯里化就是把本来多个参数的函数转换为只有一个参数的函数,如下示例:

Function<Integer, Function<Integer, Function<Integer, Integer>>> function2 = x -> y -> z -> x + y + z;
System.out.println("计算结果为: " + function2.apply(1).apply(2).apply(3)); // 计算结果为: 6

如果大家想学习以上路线内容,在此我向大家推荐一个架构学习交流群。交流学习群号874811168 里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化、分布式架构等这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多

函数柯里化的目的是将函数标准化,函数可灵活组合,方便统一处理等,例如我可以在循环里只需要调用同一个方法,而不需要调用另外的方法就能实现一个数组内元素的求和计算,代码如下:

public static void main(String[] args) {
Function<Integer, Function<Integer, Function<Integer, Integer>>> f3 = x -> y -> z -> x + y + z;
int[] nums = {1, 2, 3};
for (int num : nums) {
if (f3 instanceof Function) {
Object obj = f3.apply(num);
if (obj instanceof Function) {
f3 = (Function) obj;
} else {
System.out.println("调用结束, 结果为: " + obj); // 调用结束, 结果为: 6
}
}
}
}

级联表达式和柯里化一般在实际开发中并不是很常见,所以对其概念稍有理解即可,这里只是简单带过,若对其感兴趣的可以查阅相关资料。

参考:developerworks

来源:zero菌

Java函数式编程和lambda表达式的更多相关文章

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

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

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

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

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

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

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

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

  5. Java8函数式编程和lambda表达式

    文章目录函数式编程JDK8接口新特性函数接口方法引用函数式编程函数式编程更多时候是一种编程的思维方式,是一种方法论.函数式与命令式编程区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉 ...

  6. Java8函数式编程以及Lambda表达式

    第一章 认识Java8以及函数式编程 尽管距离Java8发布已经过去7.8年的时间,但时至今日仍然有许多公司.项目停留在Java7甚至更早的版本.即使已经开始使用Java8的项目,大多数程序员也仍然采 ...

  7. Python函数式编程:Lambda表达式

    首先我们要明白在编程语言中,表达式和语句的区别. 表达式是一个由变量.常量.有返回值的函数加运算符组成的一个式子,该式子是有返回值的 ,如  a + 1 就是个表达式, 单独的一个常量.变量 或函数调 ...

  8. C# 函数式编程 —— 使用 Lambda 表达式编写递归函数

    最近看了赵姐夫的这篇博客http://blog.zhaojie.me/2009/08/recursive-lambda-expressions.html,主要讲的是如何使用 Lambda 编写递归函数 ...

  9. 函数式编程--使用lambda表达式

    前面一篇博客我们已经说到了,lambda表达式允许使用更简洁的代码来创建只有一个抽象方法的接口的实例.现在我们来写一段java的命令者模式来自己研究下lambda表达式的语法. 这里重复下命令者模式: ...

随机推荐

  1. Linux如何查看YUM的安装目录

    Linux下如何查看使用YUM安装过的包的安装路径呢? 在搞清楚这个问题前,我们先来了解一下YUM. YUM(全称为 Yellow dog Updater, Modified)是一个在Fedora和R ...

  2. spring4笔记----spring生命周期属性

    init-method : 指定bean的初始化方法-spring容器会在bean的依赖关系注入完成后调用该方法 destroy-method :指定bean销毁之前的方法-spring容器将会在销毁 ...

  3. Android 5.0 版本 USB 调试模式打开方法

    Android 4.2 版本 USB 调试模式打开方法 1. 进入“设置”页面,点击“关于平板电脑”.见下图红色方框.   2. 疯狂点击“版本号”,见下图红色方框,直到出现“您现在处于开发者模式!” ...

  4. iOS 键盘上方增加工具栏

    UIToolbar *keyboardDoneButtonView = [[UIToolbar alloc] init]; [keyboardDoneButtonView sizeToFit]; UI ...

  5. Linux学习历程——Centos 7重置root密码

    一.自述 最近刚刚接触linux,因为我设置密码比较随性,把自己做系统的时候设置的root密码给forget,每当系统崩溃,重新把虚拟机备份还原后,就面临无法登陆的尴尬情况,只得重置root密码,好了 ...

  6. php配置文件php.ini的详细解析

    ;;;;;;;;;;;;;;;;;;;; ; Language Options ; ;;;;;;;;;;;;;;;;;;;; ; Enable the PHP scripting language e ...

  7. Linux系统挂载Windows的共享文件夹

    解决方案 在window上文件设置共享权限 在Linux上在 使用mount 命令挂载 Windows 共享文件夹 实验环境 redhat linux 7操作系统 服务器IP 192.168.100. ...

  8. June 15. 2018 Week 24th Friday

    If at first you don't succeed, then dust yourself off and try again. 失败了没关系,重振旗鼓,从头再来. From Aaliyah, ...

  9. npm 安装卸载模块

    npm安装模块 npm install xxx利用 npm 安装xxx模块到当前命令行所在目录 npm install -g xxx利用npm安装全局模块xxx 1 2 本地安装时将模块写入packa ...

  10. FinalShell安装

    Mac版安装路径/Applications/finalshelldata Linux版安装路径/usr/lib/finalshelldata 注意:1.FinalShell运行需要java或者jdk支 ...