Java 函数编程详细介绍

 更新时间:2021年11月17日 10:09:07   作者:sofia  
这篇文章主要介绍了Java函数式编程,lambda表达式可以被认为是一个匿名函数,可以在函数接口的上下文中使用。函数接口是只指定一个抽象方法的接口,下面来看文章的详细内容,需要的朋友可以参考下

前言:

函数式编程是一种编程范式,其中程序是通过应用和组合函数来构造的。它是一种声明式编程范式,其中函数定义是表达式树,每个表达式树返回一个值,而不是一系列改变程序状态的命令语句

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是具有方法getRankstudent类型的局部变量
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

函数接口带有两个默认方法,andThencompose

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属性过滤一组书籍(将其视为sqlwhere子句)。

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表达式可以被认为是一个匿名函数,可以在函数接口的上下文中使用。函数接口是只指定一个抽象方法的接口。

到此这篇关于Java函数式编程详情的文章就介绍到这了,更多相关Java函数式编程内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Maven的安装配置详解

    Maven的安装配置详解

    这篇文章主要介绍了Maven的安装配置详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-08-08
  • 快速理解Java设计模式中的组合模式

    快速理解Java设计模式中的组合模式

    这篇文章主要介绍了快速理解Java设计模式中的组合模式,具有一定参考价值,需要的朋友可以了解下。
    2017-11-11
  • springmvc处理模型数据ModelAndView过程详解

    springmvc处理模型数据ModelAndView过程详解

    这篇文章主要介绍了springmvc处理模型数据ModelAndView过程详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-01-01
  • SpringBoot整合screw实现数据库文档自动生成的示例代码

    SpringBoot整合screw实现数据库文档自动生成的示例代码

    这篇文章主要介绍了SpringBoot整合screw实现数据库文档自动生成的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-09-09
  • Java 集合系列(二)ArrayList详解

    Java 集合系列(二)ArrayList详解

    这篇文章主要介绍了Java集合系列ArrayList,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-03-03
  • JAVA设计模式之建造者模式原理与用法详解

    JAVA设计模式之建造者模式原理与用法详解

    这篇文章主要介绍了JAVA设计模式之建造者模式,简单说明了建造者模式的原理、组成,并结合实例形式分析了java建造者模式的定义与用法,需要的朋友可以参考下
    2017-08-08
  • 详解Spring Data JPA系列之投影(Projection)的用法

    详解Spring Data JPA系列之投影(Projection)的用法

    本篇文章主要介绍了详解Spring Data JPA系列之投影(Projection)的用法,具有一定的参考价值,有兴趣的可以了解一下
    2017-07-07
  • 浅谈Java并发编程基础知识

    浅谈Java并发编程基础知识

    这篇文章主要介绍了浅谈Java并发编程基础知识,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-11-11
  • Java基础之三大控制流程结构

    Java基础之三大控制流程结构

    这篇文章主要介绍了Java基础之三大控制流程结构,文中有非常详细的代码示例,对正在学习java基础的小伙伴们有非常好的帮助,需要的朋友可以参考下
    2021-04-04
  • 详解Java内部类与对象的打印概念和流程

    详解Java内部类与对象的打印概念和流程

    在 Java 中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类。广泛意义上的内部类一般来说包括这四种:成员内部类、局部内部类、匿名内部类和静态内部类
    2021-10-10

最新评论