java — 函数式编程

函数式编程

面向对象过分强调“必须通过对象的形式来做事情”,而函数式思想则尽量忽略面向对象的复杂语法——强调做什么,而不是怎么做
有时只是为了做某事情而不得不创建一个对象,而传递一段代码才是我们真正的目的。

Lambda

Lambda是一个匿名函数,可以理解为一段可以传递的代码。
当需要启动一个线程去完成任务时, 通常会通过java.lang.Runnable接口来定义任务内容,并使用java.lang.Thread类来启动该线程
传统写法,代码如下:

public class Demo {
    public static void main(String[] args) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("多线程任务执行!");
            }
        }).start();
    }
}

借助Java 8的全新语法,上述Runnable接口的匿名内部类写法可以通过更简单的Lambda表达式达到同样的效果:

public class Demo04LambdaRunnable {
    public static void main(String[] args) {
        new Thread(() -> System.out.println("多线程任务执行!")).start(); // 启动线程
    }
}

Lambda的优点 简化匿名内部类的使用,语法更加简单。

前提条件

必须是接口, 接口中有且只有一个抽象方法

有且仅有一个抽象方法的接口,称为函数式接口

格式

Lambda表达式的标准格式为:

() -> {}
() 参数列表,无参数留空
-> 固定写法, 代表指向动作
{} 方法体

省略规则
在Lambda标准格式的基础上,使用省略写法的规则为:

  1. 参数类型可省略
  2. 如果只有一个参数 ()可以省略
  3. 如果方法体只有一句话 return 大括号 分号都可省略, 但必须同时省略
new Thread(() -> System.out.println("省略格式开启线程")).start();

原理

  1. 在匿名方法所在类中新增一个方法,方法体为Lambda表达式中的代码
  2. 运行时形成一个新的类,并实现对应接口
  3. 重写方法的方法体中调用匿名方法所在类中新生成的方法.

函数式接口

函数式接口在Java中是指:有且仅有一个抽象方法的接口
函数式接口,即适用于函数式编程场景的接口。而Java中的函数式编程体现就是Lambda,所以函数式接口就是可以适用于Lambda使用的接口。

从应用层面来讲,Java中的Lambda可以看做是匿名内部类的简化格式,但是二者在原理上不同。

格式

只要确保接口中有且仅有一个抽象方法即可:

修饰符 interface 接口名称 {
    public abstract 返回值类型 方法名称(可选参数信息);
}

由于接口当中抽象方法的public abstract是可以省略的,所以定义一个函数式接口很简单:

public interface MyFunctionalInterface {
    void myMethod();
}

@FunctionalInterface

@FunctionalInterface 该注解可用于一个接口的定义上:

@FunctionalInterface
public interface MyFunctionalInterface {
    void myMethod();
}

一旦使用该注解来定义接口,编译器将会强制检查该接口是否确实有且仅有一个抽象方法,否则将会报错。不过,即使不使用该注解,只要满足函数式接口的定义,这仍然是一个函数式接口

常用函数式接口

Supplier接口

java.util.function.Supplier<T>接口,它意味着"供给" , 对应的Lambda表达式需要“对外提供”一个符合泛型类型的对象数据。
抽象方法:
T get() 用来获取一个泛型参数指定类型的对象数据
求数组元素最大值
使用Supplier接口作为方法参数类型,通过Lambda表达式求出int数组中的最大值

public class supplierInterface {
    public static void main(String[] args) {
        int[] arr = {3,24,346,4,13};
        method(() -> {
            Arrays.sort(arr);
            return arr[arr.length - 1];
        });
    }

    public static void method(Supplier<Integer> s) {
        Integer max = s.get();
        System.out.println(max);
    }
}
Consumer接口

java.util.function.Consumer<T> 接口不生产数据,而是消费一个数据,其数据类型由泛型参数决定
抽象方法
void accept(T t),意为消费一个指定泛型的数据
默认方法
default Consumer<T> andThen(Consumer<? super T> after)

public class _4_consumerInterface {
    public static void main(String[] args) {
        method("Hello World", (String s) -> {
            System.out.println(s.toUpperCase());
        });

        method("HEllO WorlD", s -> System.out.println(s.toLowerCase()));

        System.out.println("==========================");
        method("HEllO WorlD", (String s) -> {
            System.out.println(s.toUpperCase());
        }, (String s) -> {
            System.out.println(s.toLowerCase());
        });
        method("HEllO WorlD",
                s -> System.out.println(s.toUpperCase()),
                s -> System.out.println(s.toLowerCase())
        );
    }

    public static void method(String s, Consumer<String> c) {
        c.accept(s);
    }
    public static void method(String s, Consumer<String> c1, Consumer<String> c2) {
//        c1.accept(s);
//        c2.accept(s);
        // andThen c1.accept(s)后执行c2.accept(s) 等同于上面的写法
        c1.andThen(c2).accept(s);
    }
}
Function接口

java.util.function.Function<T,R> 接口用来根据一个类型的数据得到另一个类型的数据,前者称为前置条件,后者称为后置条件
抽象方法:
R apply(T t) 根据类型T的参数获取类型R的结果

public class Test {
    public static void main(String[] args) {
        Function<String,Integer> f = new Function<String, Integer>() {
            @Override
            public Integer apply(String s) {
                return Integer.parseInt(s);
            }
        };

        Integer apply = f.apply("100");
        System.out.println(apply);

        Function<String,Integer> f2 = s -> Integer.parseInt(s);
        System.out.println(f2.apply("200"));
    }
}

默认方法:andThen

Function接口中有一个默认的andThen方法,用来进行组合操作,与Consumer接口相同

public class Test {
    public static void main(String[] args) {
        method5("10", (String s) -> {
            return Integer.parseInt(s);
        }, (Integer i) -> {
            return i * 10;
        });
        method5("100", s -> Integer.parseInt(s), i -> i * 10);
        method5("1000", Integer::parseInt, i -> i * 10);
    }
    private static void method5(String s, Function<String, Integer> f1, Function<Integer, Integer> f2) {
//        Integer i = f1.apply(s);
//        Integer n = f2.apply(i);
        Integer n = f1.andThen(f2).apply(s);
        System.out.println(n);
    }
}

Function的前置条件泛型和后置条件泛型可以相同

Predicate接口

java.util.function.Predicate 判断型接口
抽象方法: boolean test(T t) 返回boolean

public class predicateInterface {
    public static void main(String[] args) {
        method("HelloWorld.java", (String s) -> {
            return s.toLowerCase().endsWith(".java");
        });

        method("Hello.java", s -> s.toLowerCase().endsWith(".java"));

    }
    public static void method(String filename, Predicate<String> p) {
        boolean b = p.test(filename);
        System.out.println(b);
    }
}

默认方法
Predicate<T> and(Predicate<? super T> other) 并且, 底层使用 &&
Predicate<T> or(Predicate<? super T> other) 或者, 底层使用 ||
Predicate<T> negate() 取反, 底层使用 !

public class Test {
    public static void main(String[] args) {
        method("Helloworld" ,s -> s.contains("H"), s -> s.contains("W"));
    }
    private static void method(String str ,Predicate<String> one, Predicate<String> two) {
        boolean b1 = one.test(str);
        boolean b2 = two.test(str);
        System.out.println("字符串符合要求吗:" + (b1 && b2));

        boolean isValid = one.and(two).test(str);
        System.out.println("字符串符合要求吗:" + isValid);
    }
}

public class Test {
    public static void main(String[] args) {
        method("Helloworld" ,s -> s.contains("H"), s -> s.contains("W"));
    }
    private static void method(String str ,Predicate<String> one, Predicate<String> two) {
        boolean b1 = one.test(str);
        boolean b2 = two.test(str);
        System.out.println("字符串符合要求吗:" + (b1 || b2));

        boolean isValid = one.or(two).test(str);
        System.out.println("字符串符合要求吗:" + isValid);
    }
}


public class Test {
    public static void main(String[] args) {
       isLong("aaa", new Predicate<String>() {
           @Override
           public boolean test(String s) {
               return  s.length()<5;
           }
       });
       isLong("bbbaa",s -> s.length()>=5);
    }
    public static void isLong(String s , Predicate<String> p){
        boolean test = p.test(s);
        System.out.println(!test);
        boolean b2 =  p.negate().test(s);
        System.out.println(b2);
    }
}

方法引用

前提

Lambda表达式中只有一句话时 可能使用

格式

符号表示 : ::
符号说明 : 双冒号为方法引用运算符,而它所在的表达式被称为方法引用
推导与省略 : ** 如果使用Lambda,那么根据“可推导就是可省略**”的原则,无需指定参数类型,也无需指定的重载形式——它们都将被自动推导

应用Lambda表达式 , 在accept方法中接收字符串 , 目的就是为了打印显示字符串 , 那么通过Lambda来使用它的代码很简单:

public class DemoPrintSimple {
    private static void printString(Consumer<String> data, String str) {
        data.accept(str);
    }
    public static void main(String[] args) {
      	printString(s -> System.out.println(s), "Hello World");
    }
}

使用方法引用进行简化

public class DemoPrintRef {
    private static void printString(Consumer<String> data, String str) {
        data.accept(str);
    }
    public static void main(String[] args) {
      	printString(System.out::println, "HelloWorld");
    }
}

其他引用

public class _5_functionInterface {
    public static void main(String[] args) {
        method("100", (String s) -> {
            return Integer.parseInt(s);
        });
        method("10", s -> Integer.parseInt(s));
        /*
            类名引用静态方法
                类名::方法名
         */
        method("1000", Integer::parseInt);
        /*
            类名引用构造方法
                类名::new
         */
        method2("张三", Person::new);
        method2("李四", Person::new);
        method2("王五", s -> new Person(s));
        method3(Person::new);
         /*
            数组引用构造方法
                数据类型[]::new
         */
        method4(5,int[]::new);
        method4(3, (Integer i) -> {
            return new int[i];
        });
        method4(1, i -> new int[i]);

    }

    public static void method(String s, Function<String, Integer> f) {
        Integer n = f.apply(s);
        System.out.println(n);
    }
    private static void method2(String s, Function<String, Person> f) {
        Person p = f.apply(s);
        System.out.println(p);
    }
    private static void method3(Supplier<Person> su) {
        Person p = su.get();
        System.out.println(p);
    }
    private static void method4(Integer i, Function<Integer, int[]> f) {
        int[] arr = f.apply(i);
        System.out.println(Arrays.toString(arr));
    }
}
class Person {
    private String name;

    public Person() {}
    public Person(String name) {
        this.name = name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                '}';
    }
}

原文链接:https://www.cnblogs.com/paopaoT/p/17337998.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:java — 函数式编程 - Python技术站

(0)
上一篇 2023年4月22日
下一篇 2023年4月22日

相关文章

  • Java的作业调度类库Quartz基本使用指南

    Java的作业调度类库Quartz基本使用指南 Quartz是一个开源的作业调度框架,可以用来进行周期性任务、延时任务和基于时间的任务等的调度。本文将介绍Quartz的基本使用指南。 安装与配置 Quartz的安装与配置非常简单,在项目的依赖中添加以下Maven坐标: <dependency> <groupId>org.quartz-…

    Java 2023年6月1日
    00
  • 带你详细了解Java值传递和引用传递

    带你详细了解Java值传递和引用传递 Java中传递参数有两种方式:值传递和引用传递。理解这两种方式在Java编程中的应用非常重要。 值传递 在Java中,所有的基本数据类型都是采用值传递。当调用方法时,基本数据类型的参数会被复制一份,传入方法内部,方法内对参数的任何改变都不会影响方法外部的数据。这是因为复制后的参数是一个新的对象,与外界的原始对象没有关系。…

    Java 2023年5月26日
    00
  • 使用MybatisPlus自定义模版中能获取到的信息

    MybatisPlus(简称MP)是一个基于Mybatis的增强工具库,可以大大简化Mybatis开发,提高开发效率。MP支持使用自定义模版来生成代码,通过自定义模版可以快速生成符合公司业务规范的代码,而且MP在模版中提供了很多变量,方便我们在模版中使用。 下面详细讲解在MP自定义模板中能够获取到的信息及使用方法: 1. 可以获取表的元数据信息 在自定义模版…

    Java 2023年6月15日
    00
  • java中struts2实现文件上传下载功能

    下面是java中struts2实现文件上传下载功能的完整攻略: 一、文件上传功能的实现 1. 安装文件上传插件 在struts2中实现文件上传功能需要依赖文件上传插件,可以通过以下方式进行安装: 在pom.xml中加入以下依赖: <dependency> <groupId>org.apache.struts</groupId&g…

    Java 2023年5月20日
    00
  • extjs 分页使用jsp传递数据示例

    下面是关于”extjs 分页使用jsp传递数据示例”的完整攻略。 什么是extjs? ExtJS是目前最为流行的JavaScript应用程序图形界面库之一,它很好地支持了Web应用中的MVC模式,使Web应用变得强壮且易于维护。 extjs分页使用jsp传递数据的示例 在使用Extjs进行分页时,通常都会在后台使用jsp开发,所以这里就以jsp作为后台语言来…

    Java 2023年6月15日
    00
  • Ewebeditor 的一些漏洞总结 推荐

    Ewebeditor 的一些漏洞总结 Ewebeditor 是一款常见的 Web 富文本编辑器,但是它也存在着一些安全漏洞。本文将对这些漏洞进行总结,并提供一些示例说明。 漏洞一:目录遍历漏洞 漏洞描述:Ewebeditor 在处理文件上传请求时,未对上传文件的路径进行充分的校验,导致攻击者可以通过构造特殊的上传请求,上传任意文件到服务器中。 漏洞利用:攻击…

    Java 2023年6月16日
    00
  • Java Structs框架原理案例详解

    Java Structs框架原理案例详解 什么是Struts框架 Struts框架是一种流行的Java Web应用程序框架,Struts采用MVC(Model-View-Controller)架构模式,将应用程序的数据(Model),用户界面(View)和控制逻辑(Controller)清晰地分离开,以便维护和扩展应用程序。 Struts框架包含许多重要的组…

    Java 2023年5月20日
    00
  • Spring Boot的几种统一处理方式梳理小结

    对于Spring Boot的几种统一处理方式,我们可以从以下几个方面来进行梳理: 1. 统一异常处理 在Spring Boot中,我们通常会使用@ControllerAdvice注解来统一处理异常,具体的步骤如下: 首先,我们需要新建一个处理器类,并在类上使用@ControllerAdvice注解,表示该类是一个统一的异常处理器。 然后,我们可以在该类中定义…

    Java 2023年5月31日
    00
合作推广
合作推广
分享本页
返回顶部