JavaSE 方法引用

1. 方法引用

1.1 体验方法引用

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

package bug.MethodReference01;

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

package bug.MethodReference01;
/*
    需求:
        1.定义一个接口(Printable),里面定义一个抽象方法
            void printString(String s);
        2.定义一个测试类(PrintableDemo)
            usePrintable(Printable p)
            主方法中调用usePrintable方法
 */
public class PrintableDemo {
    public static void main(String[] args) {
        usePrintable( s -> System.out.println(s));
        System.out.println("半仙儿努力考研");

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

    }
    private static void usePrintable(Printable p){
        p.printString("半仙儿努力考研");
    }
}

1.2 方法引用符

方法引用符

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

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

  • Lambda表达式:usePrintable( s -> System.out.println(s));
    分析:拿到参数 s 之后通过Lambda表达式传递给System.out.println方法去处理

  • 方法引用:usePrintable(System.out::println);
    分析:直接使用System.out 对象中的 println 方法来取代 Lambda,代码更加简洁

推导与省略

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

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

package bug.MethodReference02;
/*
    需求:
        1.定义一个接口(Printable),里面定义一个抽象方法
            void printString(int i);
        2.定义一个测试类(PrintableDemo)
            usePrintable(Printable p)
            主方法中调用usePrintable方法
 */
public class PrintableDemo {
    public static void main(String[] args) {
        usePrintable( i -> System.out.println(i));

        //方法引用
        usePrintable(System.out::println);
    }
    private static void usePrintable(Printable p){
        p.printInt(376);
    }
}

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

常见的引用方式:

  1. 引用类方法
  2. 引用对象的实例方法
  3. 引用类的实例方法
  4. 引用构造器

1.4 引用类方法

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

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

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

package bug.MethodReference03;
/*
    练习:
        1.定义一个接口(Converter),里面定义一个抽象方法
            int convert(String s);
        2.定义一个测试类
            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("376");
        System.out.println(number);
    }
}

1.5 引用对象的实例方法

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

  • 格式:对象::成员方法
  • 范例:“bug”::toUpperCase
    String 类中的方法:public String toUpperCase():将此String所有字符转换为大写
package bug.MethodReference04;
/*
    练习:
        1.定义一个类(PrintString)
            public void printUpper(String s):把字符串参数变成大写的数据,然后在控制台输出
        2.定义一个接口(Printer),里面定义一个抽象方法
            void printUpperCase(String s);
        3.定义一个测试类,提供两个方法
            usePrinter(Printer p)
            主方法中调用usePrinter方法
 */
public class PrinterDemo {
    public static void main(String[] args) {
        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("bug");
    }
}

package bug.MethodReference04;

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

package bug.MethodReference04;

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

1.6 引用类的实例方法

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

  • 格式:类名::成员方法
  • 范例:String::substring
    String 类中的方法:public String substring(int beginIndex,int endIndex):从beginIndex开始到endIndex结束,截取字符串。返回一个子串,子串的长度为endIndex-beginIndex
package bug.MethodReference05;

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

package bug.MethodReference05;
/*
    练习:
        1.定义一个接口(MyString),里面定义一个抽象方法
            String mySubString(String s,int x,int y);
        2.定义一个测试类(MyStringDemo)
            useMyString(MyString my)
            主方法中调用useMyString方法
 */
public class MyStringDemo {
    public static void main(String[] args) {
//        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("cappuccino",2,7);
        System.out.println(s);
    }
}

1.7 引用构造器

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

  • 格式:类名::new
  • 范例:Student::new
package bug.MethodReference06;

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 bug.MethodReference06;

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

package bug.MethodReference06;
/*
    练习:
        1.定义一个类(Student),里面有两个成员变量(name,age)
        2.定义一个接口(StudentBuilder),里面定义一个抽象方法
            Student build(String name,int age);
        3.定义一个测试类(StudentDemo)
            useStudentBuilder(StudentBuilder sb)
            主方法调用useStudentBuilder方法
 */
public class StudentDemo {
    public static void main(String[] args) {
        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("半仙儿", 22);
        System.out.println(s.getName() + "-----" + s.getAge());
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值