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()); } }
本文作者:编程初学者求大佬指点
本文链接:https://www.cnblogs.com/fragmentary/p/16990570.html
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步