函数式接口和方法引用

奋斗吧
奋斗吧
擅长邻域:未填写

标签: 函数式接口和方法引用 Java博客 51CTO博客

2023-06-25 18:24:24 137浏览

函数式接口和方法引用,1. 概述   45只有一个抽象方法的接口我们称之为函数接口。JDK的函数式接口都加上了**@FunctionalInterface** 注解进行标识。但是无论是否加上该注解只要接口中只有一个抽象方法,都是函数式接口。2. 常见函数式接口2.1 Consumer 消费接口   &nbs

1. 概述   45

只有一个抽象方法的接口我们称之为函数接口。

JDK的函数式接口都加上了**@FunctionalInterface** 注解进行标识。但是无论是否加上该注解只要接口中只有一个抽象方法,都是函数式接口。

2. 常见函数式接口

2.1 Consumer 消费接口     46

根据其中抽象方法的参数列表和返回值类型知道,我们可以在方法中对传入的参数进行消费。

函数式接口和方法引用_抽象方法

2.2 Function 计算转换接口    46

根据其中抽象方法的参数列表和返回值类型知道,我们可以在方法中对传入的参数计算或转换,把结果返回

函数式接口和方法引用_ide_02

2.3 Predicate 判断接口   46

根据其中抽象方法的参数列表和返回值类型知道,我们可以在方法中对传入的参数条件判断,返回判断结果

函数式接口和方法引用_ide_03

2.4 Supplier 生产型接口    46

根据其中抽象方法的参数列表和返回值类型知道,我们可以在方法中创建对象,把创建好的对象返回

函数式接口和方法引用_抽象方法_04

3. 常用的默认方法   47

3.1 and    47

我们在使用Predicate接口时候可能需要进行判断条件的拼接。而and方法相当于是使用&&来拼接两个判断条件

例如:

打印作家中年龄大于17并且姓名的长度大于1的作家。

StreamDemo
// and  打印作家中年龄大于17并且姓名的长度大于1的作家。  47
    private static void testAnd() {
        List<Author> authors = getAuthors();
        authors.stream()
                .filter(new Predicate<Author>() {
                    @Override
                    public boolean test(Author author) {
                        return author.getAge()>17;
                    }
                }.and(new Predicate<Author>() {
                    @Override
                    public boolean test(Author author) {
                        return author.getName().length()>1;
                    }
                })).forEach(author -> System.out.println(author.getAge()+":::"+author.getName()));
    }

函数式接口和方法引用_抽象方法_05

使用lambda表达式写法

authors.stream()
                .filter(author -> author.getAge()>17&&author.getName().length()>1)
                .forEach(author -> System.out.println(author.getAge()+":::"+author.getName()));;

函数式接口和方法引用_静态方法_06

3.2 or 

我们在使用Predicate接口时候可能需要进行判断条件的拼接。而or方法相当于是使用||来拼接两个判断条件。

例如:

打印作家中年龄大于17或者姓名的长度小于2的作家。

StreamDemo
//or   打印作家中年龄大于17或者姓名的长度小于2的作家。  47
    private static void testOr() {
        List<Author> authors = getAuthors();
        authors.stream()
                .filter(new Predicate<Author>() {
                    @Override
                    public boolean test(Author author) {
                        return author.getAge()>17;
                    }
                }.or(new Predicate<Author>() {
                    @Override
                    public boolean test(Author author) {
                        return author.getName().length()<2;
                    }
                })).forEach(author -> System.out.println(author.getName()));
    }

函数式接口和方法引用_ide_07

3.3 negate  47

Predicate接口中的方法。negate方法相当于是在判断添加前面加了个! 表示取反

例如:

打印作家中年龄不大于17的作家。

//negate 打印作家中年龄不大于17的作家。  47
    private static void tesNegate() {
        List<Author> authors = getAuthors();
        authors.stream()
                .filter(new Predicate<Author>() {
                    @Override
                    public boolean test(Author author) {
                        return author.getAge()>17;
                    }
                }.negate()).forEach(author -> System.out.println(author.getAge()));
    }

函数式接口和方法引用_静态方法_08

4. 方法引用   48

我们在使用lambda时,如果方法体中只有一个方法的调用的话(包括构造方法),我们可以用方法引用进一步简化代码。

4.1 推荐用法   48

我们在使用lambda时不需要考虑什么时候用方法引用,用哪种方法引用,方法引用的格式是什么。我们只需要在写完lambda方法发现方法体只有一行代码,并且是方法的调用时使用快捷键尝试是否能够转换成方法引用即可。

当我们方法引用使用的多了慢慢的也可以直接写出方法引用。

4.2 基本格式   48

类名或者对象名::方法名

4.3 语法详解(了解)  49

4.3.1 引用类的静态方法   49

其实就是引用类的静态方法

格式

类名::方法名

使用前提

如果我们在重写方法的时候,方法体中只有一行代码,并且这行代码是调用了某个类的静态方法并且我们把要重写的抽象方法中所有的参数都按照顺序传入了这个静态方法中,这个时候我们就可以引用类的静态方法。

例如:

如下代码就可以用方法引用进行简化

StreamDemo

//方法引用  引用类的静态方法   49
    private static void test27() {
        List<Author> authors = getAuthors();
        Stream<Author> authorStream = authors.stream();

        //匿名内部类写法
        /*authorStream.map(new Function<Author, Integer>() {
            @Override
            public Integer apply(Author author) {
                return author.getAge();
            }
        })
                    .map(new Function<Integer, String>() {
                        @Override
                        public String apply(Integer age) {
                            return String.valueOf(age);
                        }
                    });*/

        //lambda表达式
        /*authorStream.map(author -> author.getAge())
                .map(age -> String.valueOf(age));*/

     }

注意,如果我们所重写的方法是没有参数的,调用的方法也是没有参数的也相当于符合以上规则。

优化后如下:

StreamDemo

//方法引用  引用类的静态方法   49
    private static void test27() {
        List<Author> authors = getAuthors();
        Stream<Author> authorStream = authors.stream();
        authorStream.map(author -> author.getAge())
        .map(String::valueOf);
    }

4.3.2 引用对象的实例方法  50

格式

对象名::方法名

使用前提

如果我们在重写方法的时候,方法体中只有一行代码,并且这行代码是调用了某个对象的成员方法,并且我们把要重写的抽象方法中所有的参数都按照顺序传入了这个成员方法中,这个时候我们就可以引用对象的实例方法

例如:

//方法引用  引用对象的实例方法   50
    private static void test28() {
        List<Author> authors = getAuthors();

        Stream<Author> authorStream = authors.stream();
        StringBuilder sb = new StringBuilder();

        //匿名内部类
       /* authorStream.map(new Function<Author, String>() {
            @Override
            public String apply(Author author) {
                return author.getName();
            }
        })
                .forEach(new Consumer<String>() {
                    @Override
                    public void accept(String name) {
                        sb.append(name);
                    }
                });*/

        //lambda表达式
        /*authorStream.map(author -> author.getName())
                .forEach(name -> sb.append(name));*/

       
    }

优化后:

//方法引用
        authorStream.map(author -> author.getName())
                .forEach(sb::append);

4.3.3 引用类的实例方法  51

格式

类名::方法名

使用前提

如果我们在重写方法的时候,方法体中只有一行代码,并且这行代码是调用了第一个参数的成员方法,并且我们把要重写的抽象方法中剩余的所有的参数都按照顺序传入了这个成员方法中,这个时候我们就可以引用类的实例方法。

例如:

MethodDemo

package com.sangeng;

//方法引用  引用类的实例方法   51
public class MethodDemo {
    interface UseString{
        String use(String str,int start,int length);
    }

    public static String subAuthorName(String str, UseString useString){
        int start = 0;
        int length = 1;
        return useString.use(str,start,length);
    }
    public static void main(String[] args) {

        //匿名内部类
        /*subAuthorName("三更草堂", new UseString() {
            @Override
            public String use(String str, int start, int length) {
                return str.substring(start,length);
            }
        });*/
    }
}

优化后

package com.sangeng;

//方法引用  引用类的实例方法   51
public class MethodDemo {
    interface UseString{
        String use(String str,int start,int length);
    }

    public static String subAuthorName(String str, UseString useString){
        int start = 0;
        int length = 1;
        return useString.use(str,start,length);
    }
    public static void main(String[] args) {

        //方法引用
        subAuthorName("三更草堂", String::substring);

    }
}

4.3.4 构造器引用    52

如果方法体中的一行代码是构造器的话就可以使用构造器引用。

格式

类名::new

使用前提

如果我们在重写方法的时候,方法体中只有一行代码,并且这行代码是调用了某个类的构造方法,并且我们把要重写的抽象方法中的所有的参数都按照顺序传入了这个构造方法中,这个时候我们就可以引用构造器。

例如:

StreamDemo

//方法引用 构造器引用   52
    private static void test29() {
        List<Author> authors = getAuthors();
        /*authors.stream()
                .map(author -> author.getName())
                .map(name->new StringBuilder(name))
                .map(sb->sb.append("-三更").toString())
                .forEach(str-> System.out.println(str));*/

       


    }

优化后

 //方法引用
        authors.stream()
                .map(author -> author.getName())
                .map(StringBuilder::new)
                .map(sb->sb.append("-三更").())
                .forEach(str-> System.out.println(str));

好博客就要一起分享哦!分享海报

此处可发布评论

评论(0展开评论

暂无评论,快来写一下吧

展开评论

您可能感兴趣的博客

客服QQ 1913284695