Java 函数编程详细介绍
作者:sofia 发布时间:2022-07-18 14:31:15
前言:
函数式编程是一种编程范式,其中程序是通过应用和组合函数来构造的。它是一种声明式编程范式,其中函数定义是表达式树,每个表达式树返回一个值,而不是一系列改变程序状态的命令语句
Java8
引入了Lambda
形式的函数式编程。术语Lambda
来自Lambda
演算,用于描述计算。
一、函数编程Lambda
我们可以将lambda
表达式视为一个匿名函数,可以将其分配给变量并传递给方法,该方法接受函数接口作为参数。Lambda
表达式没有名称,但它有一个参数列表、一个主体和一个返回类型。
(parameters) -> expression
lambda
表达式可以在函数接口的上下文中使用。
1、接口
函数接口是只指定一个抽象方法的接口。
public interface Comparator<T> {
int compare(T o1, T o2);
}
public interface Runnable {
void run();
}
Lambda
表达式允许我们直接内联提供函数接口的抽象方法的实现,并将整个表达式视为函数接口的实例。
函数描述符:
我们将函数接口的抽象方法的签名称为函数描述符。函数描述符描述lambda表达式的签名。例如,我们可以将Runnable的函数描述符看作()->void,因为它有一个抽象方法,不接受任何内容,也不返回任何内容(void)。
二、Java函数接口
1、Predicate
Predicate
<T>接口定义了一个名为test
的抽象方法,该方法接受一个泛型类型为T的对象并返回一个布尔值。此接口可用于表示使用T类型对象的布尔表达式。
函数描述符: T->boolean
@FunctionalInterface
public interface Predicate<T> {
boolean test(T t);
}
2、Consumer
java.util.function.Consumer<T>
接口定义了一个名为accept
的抽象方法,该方法接受一个泛型类型为T的对象,并且不返回任何结果(void)。当我们需要访问T类型的对象并对其执行某些操作时,可以使用此接口。
函数描述符: T->void
3、Function
java.util.function.function<T,R>
接口定义了一个名为apply
的抽象方法,该方法将一个泛型类型为T的对象作为输入,并返回一个泛型类型为R的对象。当我们需要定义一个lambda
将信息从输入对象映射到输出时,可以使用该接口。
函数描述符: T->R
4、Supplier
接口java.util.function.Supplier<T>
定义了一个名为get
的抽象方法,该方法不接受任何内容并返回类型为T的对象。
函数描述符: ()->R
Primitive Specializations
原语接口是专用接口,用于在输入或输出为原语时避免自动装箱操作。
public interface IntPredicate {
boolean test(int t);
}
三、类型检查
lambda
的类型是从使用lambda
的上下文中推导出来的。上下文中lambda
表达式所需的类型(例如,传递给它的方法参数或分配给它的局部变量)称为目标类型。Lambda
表达式可以从赋值上下文、方法调用上下文(参数和返回)和强制转换上下文中获取其目标类型。
Object o = (Runnable) () -> System.out.println("Hello");
1、Capturing Lambda
lambda
可以不受限制地捕获(在其主体中引用)实例变量和静态变量。但是当捕获局部变量时,它们必须显式地声明为final或实际上是final
。
我们为何有这个限制?
实例变量存储在堆上,而局部变量位于堆栈上。如果lambda可以直接访问局部变量,并且lambda在线程中使用,那么使用lambda的线程可以在分配变量的线程解除分配变量后尝试访问该变量。因此,Java将对自由局部变量的访问实现为对其副本的访问,而不是对原始变量的访问。如果局部变量只分配给一次,则这没有什么区别,因此存在限制。
四、方法引用
有三种主要的方法参考:
对静态方法的方法引用。例如
,–Integer::parseInt
对任意类型的实例方法的方法引用。示例
–String::length
对现有对象或表达式的实例方法的方法引用。示例
–student::getRank
,其中student
是具有方法getRank
的student
类型的局部变量
List<String> list = Arrays.asList("a","b","A","B");
list.sort((s1, s2) -> s1.compareToIgnoreCase(s2));
可以写成
List<String> list = Arrays.asList("a","b","A","B");
list.sort(String::compareToIgnoreCase);
1、构造函数引用
可以使用ClassName::new引用现有构造函数
Supplier<List<String>> supplier = ArrayList::new;
与Supplier<List<String>> supplier = () -> new ArrayList<>()
相同;
2、组合Lambda
许多函数接口包含可用于组合lambda
表达式的默认方法。组合示例-
将两个谓词组合成一个较大的谓词,在两个谓词之间执行or操作
反向或链式比较器
3、Comparators
按逆序排列学生
Comparator<Student> c = Comparator.comparing(Student::getRank);
students.sort(comparing(Student::getRank).reversed());
根据姓名(反向)对学生进行排序,然后按反向顺序排列
students.sort(comparing(Student::getName).reversed()
.thenComparing(Student::getRank));
Predicates
Predicates接口包括三个方法:negate
, and
, 和 or
,可用于创建更复杂的谓词。
Predicate<Integer> naturalNumber = i -> i > 0;
Predicate<Integer> naturalNumberLessThanHundred = naturalNumber.and( i -> i < 100);
4、Functions
函数接口带有两个默认方法,andThen
和compose
。
Consider f(x) = x2 and g(x) = x3 + 1 then
g(f(x)) ->
Function<Integer,Integer> square = n -> n*n;
Function<Integer,Integer> squareAndCube = square.andThen(n -> n*n*n+1);
System.out.println(squareAndCube.apply(2));
65
f(g(x)) ->
Function<Integer,Integer> square = n -> n*n;
Function<Integer,Integer> squareAndCube = square.compose(n -> n*n*n + 1);
System.out.println(squareAndCube.apply(2));
应用Lambda
让我们看看如何编写一个通用方法来根据veratain
属性过滤一组书籍(将其视为sql
的where
子句)。
public static List<Book> filter(Predicate<Book> where) {
List<Book> books = Catalogue.books();
return books.stream().filter(where).collect(Collectors.toList());
}
Lambda
表达式通过不同的过滤器过滤不同的书籍
List<Book> javaBook = filter(book -> book.getCategory().equals(JAVA));
List<Book> joshuaBlochBook = filter(book -> book.getAuthor().equals("Joshua Bloch"));
五、总结
lambda
表达式可以被认为是一个匿名函数,可以在函数接口的上下文中使用。函数接口是只指定一个抽象方法的接口。
来源:https://javakk.com/2449.html


猜你喜欢
- 使用mybatis生成逆向工程的详细步骤,我个人感觉这个是最简单的一个了,虽然网上有很多种的方法来生成逆向工程,可是这个方法最简单。在这里我
- Android程序有很多Activity,比如说主窗口A,调用了子窗口B,如果在B中直接finish(), 接下里显示的是A。在B中如何关闭
- 一. 封装封装是面向对象的三大特性之一;面向对象程序三大特性:封装、继承、多态 。封装:将数据和操作数据的方法进行有机结合,隐藏对象的属性和
- 目录IntroSampleWhat insideMoreReferenceIntroC# 9 中引入了 record,record 是一个特
- 前言之前我们在浅谈6个成员函数中有提到深浅拷贝的问题,现在再回首掏一把。一、深浅拷贝哪家强?先给出代码理一理#define _CRT_SEC
- 参考链接IDEA 2020.2.3版本IntelliJ IDEA 2020.2.3永久激活码(亲测有效)IDEA 2019.3版本Intel
- 开发中遇到Eclipse报错:java.lang.OutOfMemoryError: PermGen space。PermGen space
- 看到这个标题,相信不少人会感到疑惑,回忆你们自己的场景会发现,在Spring的项目中很少有使用多线程处理任务的,没错,大多数时候我们都是使用
- 从Microsoft .Net 2.0版本以后,就默认提供了System.IO.Ports.SerialPort类,用户可以非常简单地编写少
- 这篇文章主要介绍了Spring整合Dubbo框架过程及原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,
- 最近需要用到微信的标签功能(如下图所示)。该功能可以添加已有标签,也可以自定义标签。也可以删除已编辑菜单。研究了一番。发现还是挺
- 首先对图片进行UUID 防止图片被覆盖以及爬图UUID的生成规则:日期时间,MAC地址,HashCode,随机数(多种之一)开发上传接口,两
- Webview打开本地图片选择器十分之麻烦,其在安卓系统3x 4x 5x上的行为都不同,处理也不同,所以之前差点崩溃。经过测试和完善,最终其
- 最近项目中用到了service进行计时,在连接USB的情况下一切正常,但是拔掉USB后发现,手机进入休眠后service停止了工作。最后通过
- 本文研究的主要是Hibernate hql查询的相关内容,具体如下。HQL介绍Hibernate语言查询(Hibernate Query L
- 发一个库存程序,好像是几个礼拜之前写的吧,是一个用安卓实现的简易的计算器,写这个小程序之前,看了很多人写的计算器,觉得使用一个 EditTe
- 前言单例模式最初的定义出现于《设计模式》(艾迪生维斯理, 1994):“保证一个类仅有一个实例,并提供一个访问它的全局访问点。”而我对单例的
- 基于IDEA生成可执行jar包1.编写class的代码,注意一定要有main()方法才可以生成jar包,main()方法可以没有内容。例如:
- 话不多说,上代码: /** * 获取字符串的长度,如果有中文,则每个中文字符计为2位 * @param value
- 本文介绍了如何使用缓存来提高UI的载入输入和滑动的流畅性。使用内存缓存、使用磁盘缓存、处理配置改变事件等方法将会有效的解决这个问题。在您的U