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());
}
}
标签:27,Java,String,int,void,引用,方法,public
From: https://www.cnblogs.com/fragmentary/p/16990570.html