Maison > Java > javaDidacticiel > le corps du texte

Une analyse approfondie de la programmation fonctionnelle Java

WBOY
Libérer: 2022-11-10 16:17:22
avant
1307 Les gens l'ont consulté

Cet article vous apporte des connaissances pertinentes sur java, qui présente principalement du contenu pertinent sur la programmation fonctionnelle. Java ne prenait pas en charge la programmation fonctionnelle au début, mais dans la grande version de java8, afin de prendre en charge la programmation fonctionnelle et Java en ont introduit de nombreuses. fonctionnalités importantes. Examinons-les ensemble, j'espère que cela sera utile à tout le monde.

Une analyse approfondie de la programmation fonctionnelle Java

Étude recommandée : "Tutoriel vidéo Java"

Java ne prenait pas en charge la programmation fonctionnelle au début C'est facile à comprendre quand on y pense, car en Java, les classes sont des citoyens de premier ordre, ce qui. L'implémentation de la programmation en Java n'est pas une tâche facile, mais bien que cela soit difficile, nous connaissons déjà les résultats. Afin de prendre en charge la programmation fonctionnelle dans la version majeure de Java 8, Java a introduit de nombreuses fonctionnalités importantes, dont nous avons parlé dans. Dans les chapitres précédents, nous avons découvert les expressions Lambda et diverses opérations de flux dans l'API Stream. Dans l'article d'aujourd'hui, nous allons trier les interfaces fonctionnelles intégrées à Java.

Le résumé de cet article est le suivant :

Java résume plusieurs interfaces fonctionnelles intégrées que les développeurs peuvent utiliser en fonction de cas d'utilisation de scénarios de demande courants, tels que Fonction, Fournisseur<.>, etc. Etc., les paramètres ou les types de valeurs de retour de diverses méthodes de fonctionnement dans Stream sont souvent ces interfaces fonctionnelles intégrées. <code>FunctionSupplier 等等,Stream 中各种操作方法的参数或者是返回值类型往往就是这些内置的函数式接口。

比如 Stream 中 map 操作方法的参数类型就是 Function

<R> Stream<R> map(Function<? super T, ? extends R> mapper);
Copier après la connexion
Copier après la connexion

那为什么我们在平时使用 Stream 操作的 map 方法时,从来没有见过声明这个类型的参数呢?大家可以回顾一下我们 Stream API 操作那一篇文章里使用 map 方法的例子,比如下面这个通过 map 方法把流中的每个元素转换成大写的例子。

List<String> list = new ArrayList<String>();
Stream<String> stream = list.stream();

Stream<String> streamMapped = stream.map((value) -> value.toUpperCase());
Copier après la connexion

map 方法的参数直接是一个 Lambada 表达式:

(value) -> value.toUpperCase()
Copier après la connexion

这个Lambda 表达式就是Function接口的实现。

函数式接口的载体通常是 Lambda 表达式,通过 Lambda 表达式,编译器会根据 Lambda 表达式的参数和返回值推断出其实现的是哪个函数式接口。使用 Lambda 表达式实现接口,我们不必像匿名内部类那样--指明类要实现的接口,所以像 Stream 操作中虽然参数或者返回值类型很多都是 Java 的内置函数式接口,但是我们并没有显示的使用匿名类实现它们。

虽然Lambda 表达式使用起来很方便,不过这也从侧面造成了咋一看到那些 Java 内置的函数式接口类型时,我们会有点迷惑“这货是啥?这货又是啥?”的感觉。

下面我们先说一下函数式编程、Java 的函数式接口、Lambda 为什么只能实现函数式接口这几个问题,把这些东西搞清楚了再梳理 Java 内置提供了哪些函数式接口。

函数式编程

函数式编程中包含以下两个关键的概念:

  • 函数是第一等公民
  • 函数要满足一下约束
    • 函数的返回值仅取决于传递给函数的输入参数。
    • 函数的执行没有副作用。

即使我们在写程序的时候没有一直遵循所有这些规则,但仍然可以从使用函数式编程思想编写程序中获益良多。

接下来,我们来看一下这两个关键概念再 Java 函数编程中的落地。

函数是一等公民

在函数式编程范式中,函数是语言中的第一等公民。这意味着可以创建函数的“实例”,对函数实例的变量引用,就像对字符串、Map 或任何其他对象的引用一样。函数也可以作为参数传递给其他函数。

在 Java 中,函数显然不是第一等公民,类才是。所以 Java 才引入 Lambda 表达式,这个语法糖从表现层上让 Java 拥有了函数,让函数可以作为变量的引用、方法的参数等等。为啥说是从表现层呢?因为实际上在编译的时候 Java 编译器还是会把 Lambda 表达式编译成类。

纯函数

函数编程中,有个纯函数(Pure Function)的概念,如果一个函数满足以下条件,才是纯函数:

  • 该函数的执行没有副作用。
  • 函数的返回值仅取决于传递给函数的输入参数。

下面是一个 Java 中的纯函数(方法)示例

public class ObjectWithPureFunction{    public int sum(int a, int b) {        return a + b;
    }
}
Copier après la connexion

上面这个sum()方法的返回值仅取决于其输入参数,而且sum()

Par exemple, le type de paramètre de la méthode d'opération map dans Stream est Function

public class ObjectWithNonPureFunction{    private int value = 0;    public int add(int nextValue) {        this.value += nextValue;        return this.value;
    }
}
Copier après la connexion
Copier après la connexion
Alors pourquoi ne voyons-nous jamais de paramètres de ce type déclarés alors que nous utilisons habituellement la méthode map d'opération Stream ? Vous pouvez consulter l'exemple d'utilisation de la méthode map dans notre article sur le fonctionnement de l'API Stream, comme l'exemple suivant de conversion de chaque élément du flux en majuscules via la méthode map. Le paramètre de la méthode 🎜
public interface MyInterface {    public void run();
}
Copier après la connexion
Copier après la connexion
🎜map est directement une expression Lambada : 🎜
public interface MyInterface2 {
    public void run();

    public default void doIt() {
        System.out.println("doing it");
    }

    public static void doItStatically() {
        System.out.println("doing it statically");
    }
}
Copier après la connexion
Copier après la connexion
🎜Cette expression Lambda est l'implémentation de l'interface Function. 🎜🎜Le support d'une interface fonctionnelle est généralement une expression Lambda. Grâce à l'expression Lambda, le compilateur déduira quelle interface fonctionnelle il implémente en fonction des paramètres et de la valeur de retour de l'expression Lambda. En utilisant des expressions Lambda pour implémenter des interfaces, nous n'avons pas besoin de spécifier l'interface à implémenter par la classe comme les classes internes anonymes. Par conséquent, bien que de nombreux paramètres ou types de valeurs de retour dans les opérations Stream soient des interfaces fonctionnelles intégrées à Java, nous ne le faisons pas. affichez-les. Implémentez-les à l’aide de classes anonymes. 🎜🎜Bien que les expressions Lambda soient très pratiques à utiliser, cela nous rend également un peu confus lorsque nous voyons ces types d'interfaces fonctionnelles intégrées en Java, "Qu'est-ce que c'est ? Qu'est-ce que c'est que cette chose ?" 🎜🎜Parlons d'abord de la programmation fonctionnelle, des interfaces fonctionnelles de Java et de la raison pour laquelle Lambda ne peut implémenter que des interfaces fonctionnelles. Après avoir clarifié ces éléments, nous déterminerons les interfaces fonctionnelles intégrées à Java. 🎜

Programmation fonctionnelle

🎜La programmation fonctionnelle contient les deux concepts clés suivants : 🎜
  • Les fonctions sont des citoyens de premier ordre
  • La fonction doit satisfaire les contraintes suivantes
    • La valeur de retour de la fonction dépend uniquement des paramètres d'entrée passés à la fonction.
    • L'exécution de la fonction n'a aucun effet secondaire.
🎜Même si nous ne suivons pas toujours toutes ces règles lors de l'écriture de programmes, nous pouvons quand même tirer de nombreux avantages de l'écriture de programmes utilisant des idées de programmation fonctionnelle. 🎜🎜Ensuite, jetons un coup d'œil à l'implémentation de ces deux concepts clés dans la programmation fonctionnelle Java. 🎜

Les fonctions sont des citoyens de première classe

🎜Dans le paradigme de programmation fonctionnelle, les fonctions sont des citoyens de première classe dans le langage. Cela signifie que vous pouvez créer des « instances » de fonctions et des références de variables à des instances de fonction, tout comme des références à des chaînes, des cartes ou tout autre objet. Les fonctions peuvent également être transmises comme arguments à d'autres fonctions. 🎜🎜En Java, les fonctions ne sont évidemment pas des citoyens de première classe, les classes le sont. C'est pourquoi Java a introduit les expressions Lambda. Ce sucre syntaxique permet à Java d'avoir des fonctions de la couche de présentation, permettant d'utiliser des fonctions comme références à des variables, des paramètres de méthodes, etc. Pourquoi dites-vous que cela vient de la couche de présentation ? Parce qu'en réalité, le compilateur Java compilera toujours les expressions Lambda en classes lors de la compilation. 🎜

Fonction pure

🎜En programmation fonctionnelle, il existe la notion de fonction pure. Si une fonction remplit les conditions suivantes, elle peut l'être. une fonction pure : 🎜
  • L'exécution de cette fonction n'a aucun effet secondaire.
  • La valeur de retour d'une fonction dépend uniquement des paramètres d'entrée passés à la fonction.
🎜Ce qui suit est un exemple de fonction (méthode) pure en Java🎜
@FunctionalInterface 
// 标明接口为函数式接口
public interface MyInterface {    public void run(); 
//抽象方法}
Copier après la connexion
Copier après la connexion
🎜La valeur de retour de la méthode sum() ci-dessus ne dépend que de ses paramètres d'entrée, et sum() n'a aucun effet secondaire, il ne modifie aucun état (variables) en dehors de la fonction nulle part. 🎜🎜Voici plutôt un exemple de fonction impure : 🎜
public class ObjectWithNonPureFunction{    private int value = 0;    public int add(int nextValue) {        this.value += nextValue;        return this.value;
    }
}
Copier après la connexion
Copier après la connexion

add()方法使用成员变量value来计算其返回值,并且它还修改了value成员变量的状态,这代表它有副作用,这两个条件都导致add方法不是一个纯函数

正如我们看到的,函数式编程并不是解决所有问题的银弹。尤其是“函数是没有副作用的”这个原则就使得在一些场景下很难使用函数式编程,比如要写入数据库的场景,写入数据库就算是一个副作用。所以,我们需要做的是了解函数式编程擅长解决哪些问题,把它用在正确的地方。

函数式接口

Java中的函数式接口在 Lambda 表达式那篇文章里提到过,这里再详细说说。函数式接口是只有一个抽象方法的接口(抽象方法即未实现方法体的方法)。一个 Interface 接口中可以有多个方法,其中默认方法和静态方法都自带实现,但是只要接口中有且仅有一个方法没有被实现,那么这个接口就可以被看做是一个函数式接口

下面这个接口只定义了一个抽象方法,显然它是一个函数式接口:

public interface MyInterface {    public void run();
}
Copier après la connexion
Copier après la connexion

下面这个接口中,定义了多个方法,不过它也是一个函数式接口:

public interface MyInterface2 {
    public void run();

    public default void doIt() {
        System.out.println("doing it");
    }

    public static void doItStatically() {
        System.out.println("doing it statically");
    }
}
Copier après la connexion
Copier après la connexion

因为doIt方法在接口中定义了默认实现,静态方法也有实现,接口中只有一个抽象方法run没有提供实现,所以它满足函数式接口的要求。

这里要注意,如果接口中有多个方法没有被实现,那么接口将不再是函数式接口,因此也就没办法用 Java 的 Lambda 表达式实现接口了

编译器会根据 Lambda 表达式的参数和返回值类型推断出其实现的抽象方法,进而推断出其实现的接口,如果一个接口有多个抽象方法,显然是没办法用 Lambda 表达式实现该接口的。

@FunctionalInterface 注解

这里扩充一个标注接口是函数式接口的注解@FunctionalInterface

@FunctionalInterface 
// 标明接口为函数式接口
public interface MyInterface {    public void run(); 
//抽象方法}
Copier après la connexion
Copier après la connexion

一旦使用了该注解标注接口,Java 的编译器将会强制检查该接口是否满足函数式接口的要求:“确实有且仅有一个抽象方法”,否则将会报错。

需要注意的是,即使不使用该注解,只要一个接口满足函数式接口的要求,那它仍然是一个函数式接口,使用起来都一样。该注解只起到--标记接口指示编译器对其进行检查的作用。

Java 内置的函数式接口

Java 语言内置了一组为常见场景的用例设计的函数式接口,这样我们就不必每次用到Lambda 表达式、Stream 操作时先创建函数式接口了,Java 的接口本身也支持泛型类型,所以基本上 Java 内置的函数式接口就能满足我们平时编程的需求,我自己在开发项目时,印象里很少见过有人自定义函数式接口。

在接下来的部分中,我们详细介绍下 Java 内置为我们提供了的函数式接口。

Function

Function接口(全限定名:java.util.function.Function)是Java中最核心的函数式接口。 Function 接口表示一个接受单个参数并返回单个值的函数(方法)。以下是 Function 接口定义的:

@FunctionalInterface
public interface Function<T, R> {
    R apply(T t);

    default <V> Function<V, R> compose(Function<? super V, ? extends T> before) {
        Objects.requireNonNull(before);
        return (V v) -> apply(before.apply(v));
    }

    default <V> Function<T, V> andThen(Function<? super R, ? extends V> after) {
        Objects.requireNonNull(after);
        return (T t) -> after.apply(apply(t));
    }

    static <T> Function<T, T> identity() {
            return t -> t;
    }
Copier après la connexion

Function接口本身只包含一个需要实现的抽象方法apply,其他几个方法都已在接口中提供了实现,这正好符合上面我们讲的函数式接口的定义:“有且仅有一个抽象方法的接口”。

Function 接口中的其他三个方法中compseandThen 这两个方法用于函数式编程的组合调用,identity用于返回调用实体对象本身,我们之前在把对象 List 转换为 Map 的内容中提到过,可以回看前面讲 List 的文章复习。

Function接口用Java 的类这么实现

public class AddThree implements Function<Long, Long> {

    @Override
    public Long apply(Long aLong) {
        return aLong + 3;
    }

    public static void main(String[] args) {
        Function<Long, Long> adder = new AddThree();
		Long result = adder.apply(4L);
		System.out.println("result = " + result);
    }
}
Copier après la connexion

不过现实中没有这么用的,前面说过 Lambda 表达式是搭配函数式接口使用的,用Lambda表达式实现上Function 接口只需要一行,上面那个例子用 Lambda 实现的形式是:

Function<Long, Long> adder = (value) -> value + 3;Long resultLambda = adder.apply(8L);
System.out.println("resultLambda = " + resultLambda);
Copier après la connexion

是不是简洁了很多。后面的接口示例统一用 Lambda 表达式举例,不再用类实现占用太多篇幅。

Function接口的常见应用是 Stream API 中的 map 操作方法,该方法的参数类型是Function接口,表示参数是一个“接收一个参数,并返回一个值的函数”。

<R> Stream<R> map(Function<? super T, ? extends R> mapper);
Copier après la connexion
Copier après la connexion

所以我们在代码里常会见到这样使用 map 操作:

stream.map((value) -> value.toUpperCase())
Copier après la connexion

Predicate

Predicate 接口 (全限定名:java.util.function.Predicate)表示一个接收单个参数,并返回布尔值 true 或 false 的函数。以下是 Predicate 功能接口定义:

public interface Predicate<T> {    boolean test(T t);
}
Copier après la connexion

Predicate 接口里还有几个提供了默认实现的方法,用于支持函数组合等功能,这里不再赘述。 用 Lambda 表达式实现 Predicate 接口的形式如下:

Predicate predicate = (value) -> value != null;
Copier après la connexion

Stream API 中的 filter 过滤操作,接收的就是一个实现了 Predicate 接口的参数。

Stream<T> filter(Predicate<? super T> predicate);
Copier après la connexion

写代码时,会经常见到这样编写的 filter 操作:

Stream<String> longStringsStream = stream.filter((value) -> {    
// 元素长度大于等于3,返回true,会被保留在 filter 产生的新流中。
    return value.length() >= 3;
});
Copier après la connexion

Supplier

Supplier 接口(java.util.function.Supplier),表示提供某种值的函数。其定义如下:

@FunctionalInterfacepublic interface Supplier<T> {
    T get();
}
Copier après la connexion

Supplier接口也可以被认为是工厂接口,它产生一个泛型结果。与 Function 不同的是,Supplier 不接受参数。

Supplier<Integer> supplier = () -> new Integer((int) (Math.random() * 1000D));
Copier après la connexion

上面这个 Lambda 表达式的 Supplier 实现,用于返回一个新的 Integer 实例,其随机值介于 0 到 1000 之间。

Consume

Consumer 接口(java.util.function.Consume)表示一个函数,该函数接收一个参数,但是不返回任何值。

@FunctionalInterfacepublic interface Consumer<T> {    void accept(T t);
}
Copier après la connexion

Consumer 接口常用于表示:要在一个输入参数上执行的操作,比如下面这个用Lambda 表达式实现的 Consumer,它将作为参数传递给它的value变量的值打印到System.out标准输出中。

Consumer<Integer> consumer = (value) -> System.out.println(value);
Copier après la connexion

Stream API 中的 forEach、peek 操作方法的参数就是 Consumer 接口类型的。

Stream<T> peek(Consumer<? super T> action);
void forEach(Consumer<? super T> action);
Copier après la connexion

比如,Stream API 中的 forEach 操作,会像下面这样使用 Consume 接口的实现

Stream<String> stream = stringList.stream();
// 下面是Lambda 的简写形式
// 完整形式为:value -> System.out.println(value);
stream.forEach(System.out::println);
Copier après la connexion

Optional

最后再介绍一下 Optional 接口,Optional 接口并不是一个函数式接口,这里介绍它主要是因为它经常在一些 Stream 操作中出现,作为操作的返回值类型,所以趁着学习函数式编程的契机也学习一下它。

Optional 接口是预防NullPointerException的好工具,它是一个简单的容器,其值可以是 null 或非 null。比如一个可能返回一个非空结果的方法,方法在有些情况下返回值,有些情况不满足返回条件返回空值,这种情况下使用 Optional 接口作为返回类型,比直接无值时返回 Null 要更安全。 接下来我们看看 Optional 怎么使用:

// of 方法用于构建一个 Optional 容器
Optional<String> optional = Optional.of("bam");
// 判断值是否为空
optional.isPresent();           // true
// 取出值,如果不存在直接取会抛出异常
optional.get();                 // "bam"
// 取值,值为空时返回 orElse 提供的默认值
optional.orElse("fallback");    // "bam"
// 如果只存在,执行ifPresent参数中指定的方法
optional.ifPresent((s) -> System.out.println(s.charAt(0)));// "b"
Copier après la connexion

Stream 操作中像 findAny、 findFirst这样的操作方法都会返回一个 Optional 容器,意味着结果 Stream 可能为空,因此没有返回任何元素。我们可以通过 Optional 的 isPresent() 方法检查是否找到了元素。Java 编程那些绕不开的接口 这个子系列的文章已经更新完毕,感兴趣的请持续关注,后面还有更多实用、精彩的内容。


推荐学习:《java视频教程

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Étiquettes associées:
source:juejin.im
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal