方法引用(Java)

本文详细介绍了Java中的方法引用,包括类方法引用、对象实例方法引用、类的实例方法引用以及构造器引用,并通过实例展示了如何在Lambda表达式中使用方法引用,使代码更简洁高效。同时提供了多个练习案例,帮助读者加深理解。
摘要由CSDN通过智能技术生成

方法引用

1.1 体验方法引用

  • 在使用Lambda表达式的时候,我们实际上传递进去的代码就是一种解决方案:拿参数做操作
  • 那么考虑一种情况:如果我们在Lambda中所指定的操作方案,已经有地方存在相同方案,那是否还有必要再写重复逻辑呢?
  • 答案肯定是没有必要
  • 那我们又是如何使用已经存在的方案的呢?
  • 这就是我们要讲解的方法引用,我们是通过方法引用来使用已经存在的方案

需求:

  1. 定义一个接口(Printable):里面定义一个抽象方法:void printString(String s);
  2. 定义一个测试类(PrintableDemo),在测试类中提供两个方法
    一个方法是:usePrintable(Printable p)
    一个方法是主方法,在主方法中调用usePrintable方法

案例代码:

PrintableDemo.java

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

//        usePrintable((String s)->{
//            System.out.println(s);
//        });

//        usePrintable(s-> System.out.println(s));
//
//        System.out.println("爱生活爱Java");

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

        //可推导的就是可省略的
    }

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

Printable.java

public interface Printable {
    void printString(String s);
}

1.2 方法引用符

方法引用符

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

回顾一下我们在体验方法引用中的代码

  • Lambda表达式:usePrintable(s->System.out.println(s));
    • 分析:拿到参数s之后通过Lambda表达式,传递给System.out.println方法去处理
  • 方法引用:usePrintable(System.out.println))
    • 分析:直接使用System.out中的println方法来取代Lambda,代码更加的简洁

推导与省略

  • 如果使用Lambda,那么根据“可推导就是可省略”的原则,无需指定参数类型,也无需指定的重载形式,它们都将被自动推导
  • 如果使用方法引用,也是同样可以根据上下文进行推导
  • 方法引用是Lambda的李生兄弟

需求:

  1. 定义一个接口(Printable):里面定义一个抽象方法:void printInt(int i);
  2. 定义一个测试类(PrintableDemo),在测试类中提供两个方法
    一个方法是:usePrintable(Printable p)
    一个方法是主方法,在主方法中调用usePrintable方法

案例代码:

PrintableDemo.java

/**
 * 需求:
 * 1. 定义一个接口(Printable):里面定义一个抽象方法:void printInt(int i);
 * 2. 定义一个测试类(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);
    }
}

Printable.java

public interface Printable {
    void printInt(int i);
}

1.3 Lambda表达式支持的方法引用

常见的引用方法:

  • 引用类方法
  • 引用对象的实例方法
  • 引用类的实例方法
  • 引用构造器

1.4 引用类方法

引用类方法,其实就是引用方法

  • 格式:类名::静态方法
  • 范例:Integer::parseInt
    • Integer类的方法:public static int parselnt(String s)将此String转换为int类型数据

练习:

  • 定义一个接口(Converter),里面定义一个抽象方法
    • int convert(String s);
  • 定义一个测试类(ConverterDemo),在测试类中提供两个方法
    • 一个方法是:useConverter(Converter c)
    • 一个方法是主方法,在主方法中调用useConverter方法

案例代码:

ConverterDemo.java

/**
 * 练习:
 *
 * - 定义一个接口(Converter),里面定义一个抽象方法
 *   - int convert(String s);
 * - 定义一个测试类(ConverterDemo),在测试类中提供两个方法
 *   - 一个方法是:useConverter(Converter c)
 *   - 一个方法是主方法,在主方法中调用useConverter方法
 */

public class ConverterDemo {
    public static void main(String[] args) {
//        useConverter((String s)->{
//            return Integer.parseInt(s);
//        });

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

        //引用类方法
        useConverter(Integer::parseInt);

        //Lambda表达式被类方法替代的时候,它的形式参数全部传递给静态方法作为参数
        


    }

    private static void useConverter(Converter c){
        int number = c.convert("666");
        System.out.println(number);
    }
}

Converter.java

public interface Converter {
    int convert(String s);
}

1.5 引用对象的实例方法

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

  • 格式:对象::成员方法
  • 范例:“HelloWorld”::toUpperCase
    • String类中的方法:public String toUpperCase()将此String所有字符转换为大写

练习

  • 定义一个类(PrintString),里面定义一个方法
    • public void printUpper(String s):把字符串参数变成大写的数据,然后在控制台输出
  • 定义一个接口(Printer),里面定义一个抽象方法
    • void printUpperCase(String s)
  • 定义一个测试类(PrinterDemo),在测试类中提供两个方法
    • 一个方法是:usePrinter(Printer p)
    • 一个方法是主方法,在主方法中调用usePrinter方法

案例代码:

PrinterDemo.java

/**
 * 练习
 *
 * - 定义一个类(PrintString),里面定义一个方法
 *   - public void printUpper(String s):把字符串参数变成大写的数据,然后在控制台输出
 * - 定义一个接口(Printer),里面定义一个抽象方法
 *   - void printUpperCase(String s)
 * - 定义一个测试类(PrinterDemo),在测试类中提供两个方法
 *   - 一个方法是:usePrinter(Printer p)
 *   - 一个方法是主方法,在主方法中调用usePrinter方法
 */

public class PrinterDemo {
    public static void main(String[] args) {
        //在主方法中调用usePrinter方法

//        usePrinter((String s)->{
            String result = s.toUpperCase();
            System.out.println(result);
//            System.out.println(s.toUpperCase());
//        });

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

        //引用对象的实例方法
        PrintString ps = new PrintString();
        usePrinter(ps::printUpper);

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

    private static void usePrinter(Printer p){
        p.printUpperCase("HelloWorld");
    }
}

PrintString.java

public class PrintString {
    /**
     * 把字符串参数变成大写的数据,然后在控制台输出
     *
     * @param s
     */
    public void printUpper(String s) {
        String result = s.toUpperCase();
        System.out.println(result);
    }
}

Printer.java

public interface Printer {
    void printUpperCase(String s);
}

1.6 引用类的实例方法

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

  • 格式:类名::成员方法
  • 范例:String::substring
    • String类中的方法:publicString substring(int beginindexintendindex)
    • 从beginIndex开始到endindex结束,截取字符串。返回一个子串,子串的长度为endindex-beginindex

练习

  • 定义一个接口(MyString),里面定义一个抽象方法:
    String mySubString(String s,int x,int y);
  • 定义一个测试类(MyStringDemo),在测试类中提供两个方法
    • 一个方法是:useMyString(MyString my)
    • 一个方法是主方法,在主方法中调用useMyString方法

案例代码:

MyStringDemo.java

/**
 * 练习
 *
 * - 定义一个接口(MyString),里面定义一个抽象方法:
 *   String mySubString(String s,int x,int y);
 * - 定义一个测试类(MyStringDemo),在测试类中提供两个方法
 *   - 一个方法是:useMyString(MyString my)
 *   - 一个方法是主方法,在主方法中调用useMyString方法
 */
public class MyStringDemo {
    public static void main(String[] args) {
        //主方法中调用useMyString方法

//        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 my) {
        String s = my.mySubString("HelloWorld", 2, 5);
        System.out.println(s);
    }
}

MyString.java

public interface MyString {
    String mySubString(String s,int x,int y);
}

1.7 引用构造器

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

  • 格式:类名::new
  • 范例:Student::new

练习

  • 定义一个类(Student),里面有两个成员变量(name,age)
    • 并提供无参构造方法和带参构造方法,以及成员变量对应的get和set方法
  • 定义一个接口(StudentBuilder),里面定义一个抽象方法
    • Student build(String name,int age);
  • 定义一个测试类(StudentDemo),在测试类中提供两个方法
    • 一个方法是:useStudentBuilder(StudentBuilder s)
    • 一个方法是主方法,在主方法中调用useStudentBuilder方法

案例代码:

StudentDemo.java

/**
 * 练习
 *
 * - 定义一个类(Student),里面有两个成员变量(name,age)
 *   - 并提供无参构造方法和带参构造方法,以及成员变量对应的get和set方法
 * - 定义一个接口(StudentBuilder),里面定义一个抽象方法
 *   - Student build(String name, int age);
 * - 定义一个测试类(StudentDemo),在测试类中提供两个方法
 *   - 一个方法是:useStudentBuilder(StudentBuilders)
 *   - 一个方法是主方法,在主方法中调用useStudentBuilder方法
 */
public class StudentDemo {
    public static void main(String[] args) {
        //在主方法中调用useStudentBuilder方法

//        useStudentBuilder((String name,int age) ->{
            Student s = new Student(name,age);
            return s;
//            return new Student(name, age);
//        });

        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());
    }
}

StudentBuilder.java

public interface StudentBuilder {
    Student build(String name, int age);
}

Student.java

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;
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Java小幽

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值