27_Java中的方法引用

Java中的方法引用

​ 方法引用是在Lambda的基础进行的,本质是将Lambda中的参数列表传递给我们使用的方法引用中的方法。(这样在使用一些本就存在的方法来完成函数式编程时更为方便简洁)

​ 注意:在引用类中的实例方法时接口中的第一个参数表示调用者(即调用方法的对象),后面的参数传递到方法中

一、体验方法引用

在使用Lambda表达式的时候,我们实际上传递进去的代码就是一种解决方案:拿参数做操作

那么考虑一种情况:如果我们在Lambda中所指的操作方案,已经有地方存在相同方案,那是否还有必要再写重复的逻辑呢?

答案肯定是没有必要

那我们有时如何使用已经存在的方案的呢?

这就是我们要讲的方法引用,我们是通过方法引用来使用已经存在的方案

参考代码:

package com.itheima_01;

public interface Printable {
    void printString(String s);
}
package com.itheima_01;
/*
    需求:
        定义一个接口(Printable),里面定义一个抽象方法:void printString(String s);
        定义一个测试类(PrintableDemo),在测试类中提供两个方法
            一个方法是:usePrintable(Printable p)
            一个方法是主方法,在方法中调用usePrintable方法
 */
public class PrintableDemo {
    public static void main(String[] args) {
        //调用方法
//        usePrintable((String s) -> {
//            System.out.println(s);
//        });

        usePrintable(s -> System.out.println(s));

//        System.out.println("爱生活,爱健康");

        //使用方法引用改进语句    方法引用符"::"
        usePrintable(System.out::println);  //解释一下,System.out获得的是一个PrintStream打印输出流对象

        //可推导的就是可省略的,根据形参进行判断
    }

    private static void usePrintable(Printable p){
        p.printString("爱生活,爱健康");
    }
}

二、方法引用符

1、方法引用符

​ “::” 该符号被称为方法引用运算符,而它所在的表达式被称为方法引用

2、回顾方法引用的体验:

​ Lambda表达式:usePrintable(s -> System.out.print(s));

​ 分析:拿到参数s之后通过Lambda表达式,传递给System.out.println方法去处理,而其中s的数据类型通过接口中的唯一方法进行确定

​ 方法引用:usePrintable(System.out::println);

​ 分析:直接使用System.out中的println方法来取代Lambda,代码更加简洁

3、推导与省略:

​ 如果使用Lambda,那么根据"可推到可省略"的原则,无需指定参数类型,也无需指定重载形式,它们都将被自动推导

​ 如果使用方法引用,也是同样可以根据上下文进行推断

​ 方法引用时Lambda的孪生兄弟

参考代码:

package com.itheima_02;
/*
    接口
 */
public interface Printable {
    void printInt(int i);
}
package com.itheima_02;
/*
    需求:
        定义一个接口(Printable),里面定义一个抽象方法,void printInt(int i);
        定义一个测试类(PrintableDemo),在测试类中提供两个方法
            一个方法是:usePrintable(Printable p)
            一个方法是主方法,在主方法中调用usePrintable方法
 */
public class PrintableDemo {
    public static void main(String[] args) {
        //调用usePrintable方法
        usePrintable(i -> System.out.println(i));

        //方法引用
        usePrintable(System.out::println);  //会将整型参数传递给这个函数
    }

    private static void usePrintable(Printable p){
        p.printInt(666);
    }
}

三、Lambda表达式支持的方法引用

常见的引用方式:

​ 引用类方法

​ 引用对象的实例方法

​ 引用类的实例方法

​ 引用构造器

1、引用类方法:

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

格式:类名::静态方法

范例:Integer::parseInt

​ Integer类的方法:public static int parseInt(String s); 将此String转换为int类型数据

练习:

​ 定义一个接口(Converter),里面定义一个抽象方法

​ int convert(String s);

​ 定义一个测试类(ConverterDemo),在测试类中提供两个方法

​ 一个方法是:useConverter(Converter c)

​ 一个方法是主方法,在主方法中调用useConverter方法

参考代码:

package com.itheima_03;

public interface Converter {
    int convert(String s);
}
package com.itheima_03;
/*
    定义一个接口(Converter),里面定义一个抽象方法
			int convert(String s);
	定义一个测试类(ConverterDemo),在测试类中提供两个方法
			一个方法是:useConverter(Converter c)
			一个方法是主方法,在主方法中调用useConverter方法
 */
public class ConverterDemo {
    public static void main(String[] args) {
        //在主方法中调用useConverter方法
       /* useConverter((String s) -> {
            return Integer.parseInt(s);
        });*/

        useConverter(s -> (Integer.parseInt(s)));

        //引用类方法
        //Lambda表达式被引用类方法替代时,它的形式参数全部传递给静态方法作为参数
        useConverter(Integer::parseInt);

    }

    private static void useConverter(Converter c){
        int number = c.convert("666");
        System.out.println(number);
    }
}
2、引用对象的实例方法:

引用对象的实例方法,其实就是引用类中的成员方法

格式: 对象::成员方法

范例:"HelloWorld"::toUpperCase

​ String类中的方法:public String toUpperCase(); 将此String中的所有字符串转换未大写

练习:

​ 定义一个类(PrintString),里面定义一个方法

​ public void printUpper(String s); 把字符串参数变成大写的数据,然后在控制台输出

​ 定义一个接口(Printer),里面定义一个抽象方法

​ void printUpperCase(String s)

​ 定义一个测试类(PrinterDemo),在测试类中提供两个方法

​ 一个方法时:usePrinter(Printer p)

​ 一个方法是主方法,在主方法中调用usePrinter方法

参考代码:

package com.itheima_04;
//具体类
public class PrintString {
    public void printUpper(String s){
        //把字符串中所有的字母变为大写,然后输出
        String result = s.toUpperCase();
        System.out.println(result);
    }
}
package com.itheima_04;
//接口
public interface Printer {
    void printUpperCase(String s);
}
package com.itheima_04;
/*
    定义一个类(PrintString),里面定义一个方法
			public void printUpper(String s);	把字符串参数变成大写的数据,然后在控制台输出
	定义一个接口(Printer),里面定义一个抽象方法
			void printUpperCase(String s)
	定义一个测试类(PrinterDemo),在测试类中提供两个方法
			一个方法时:usePrinter(Printer p)
    		一个方法是主方法,在主方法中调用usePrinter方法
 */
public class PrintDemo {
    public static void main(String[] args){
        //在主方法中调用usePrinter方法
        //使用Lambda
      /*  usePrinter((String s) -> {
            String result = s.toUpperCase();
            System.out.println(result);
        });*/

        usePrinter(s -> System.out.println(s.toUpperCase()));

        //引用对象的实例方法
        usePrinter(new PrintString()::printUpper);  //使用了匿名对象

        //Lambda表达式被对象的实例方法替代的时候,它的形式参数全部传递给该方法作为参数
    }

    private static void usePrinter(Printer p){
        p.printUpperCase("HelloWorld");
    }
}
3、引用类的实例方法:

引用类的实例方法,其实就是引用类中的成员方法

格式:类名::成员方法

范例:String::substring

​ String类中的方法:public String substring(int beginIndex, int endIndex);

​ 从第一个参数表示的下标索引开始,到第一个参数表示的下标索引结束,截取一个子串,子串长度为endIndex - beginIndex

练习:

​ 定义一个接口(MyString),里面定义一个抽象方法:

​ String mySubString(String s, int x, int, y);

​ 定义一个测试类(MyStringDemo),在测试类中提供两个方法

​ 一个方法是:useMyString(MyString my)

​ 一个方法是主方法,在主方法中调用useMyString方法

参考代码:

package com.itheima_05;

public interface MyString {
    //Lambda表达式被类的实例方法替代的时候
    //第一个参数作为调用者    (用于指明对象)
    //后面的参数全部传递给该方法作为参数
    String mySubString(String s, int x, int y);
}
package com.itheima_05;
/*
    定义一个接口(MyString),里面定义一个抽象方法:
			String mySubString(String s, int x, int y);
	定义一个测试类(MyStringDemo),在测试类中提供两个方法
			一个方法是:useMyString(MyString my)
			一个方法是主方法,在主方法中调用useMyString方法
 */
public class MyStringDemo {
    public static void main(String[] args) {
        //在主方法中调用useMyString方法
        //使用Lambda表达式
        /*useMyString((String s, int x, int y) ->{
            return s.substring(x, y);
        });*/

        useMyString((s, x, y) -> s.substring(x, y));

        //引用类中的实例方法
        useMyString(String::substring);

        //Lambda表达式被类的实例方法替代的时候
        //第一个参数作为调用者    (用于指明对象)
        //后面的参数全部传递给该方法作为参数

    }

    private static void useMyString(MyString myString){
        String s = myString.mySubString("HelloWorld", 2, 5);
        System.out.println(s);
    }
}
4、引用构造器

引用构造器,其实就是引用构造方法

格式:类名::new

范例:Student::new

练习:

​ 定义一个类(Student),里面有两个成员变量(name,age)

​ 并提供无参构造方法和带参构造方法,以及成员变量对应的get方法和set方法

​ 定义一个接口(StudentBuilder),里面定义一个抽象方法

​ Student build(String name, int age);

​ 定义一个测试类(StudentDemo),在测试类中提供两个方法

​ 一个方法时:useStudentBuilder(StudentBuilder s)

​ 一个方法是主方法,在主方法中调用useStudentBuilder方法

参考代码:

package com.itheima_06;

public class Student {
    private String name;
    private int age;

    public Student(){}

    public Student(String name, int age){
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

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

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}
package com.itheima_06;

public interface StudentBuilder {
    Student build(String name, int age);
}
package com.itheima_06;
/*
    定义一个类(Student),里面有两个成员变量(name,age)
			并提供无参构造方法和带参构造方法,以及成员变量对应的get方法和set方法
	定义一个接口(StudentBuilder),里面定义一个抽象方法
			Student build(String name, int age);
	定义一个测试类(StudentDemo),在测试类中提供两个方法
			一个方法时:useStudentBuilder(StudentBuilder s)
			一个方法是主方法,在主方法中调用useStudentBuilder方法
 */
public class StudentDemo {
    public static void main(String[] args) {
        //在主方法中调用useStudentBuilder方法
        //使用Lambda表达式
        /*useStudentBuilder((String name, int age) -> {
            Student s = new Student(name, age);
            return s;
        });*/

        useStudentBuilder((name, age) -> new Student(name, age));


        //引用构造器
        useStudentBuilder(Student::new);

        //Lambda表达式被构造器替代时,它的形式参数全部传递给构造器作为参数
    }

    private static void useStudentBuilder(StudentBuilder sb){
        Student s = sb.build("张三", 30);
        System.out.println(s.getName() + "," + s.getAge());
    }
}
posted @ 2022-12-18 16:56  如此而已~~~  阅读(65)  评论(0编辑  收藏  举报