方法引用
1.1方法引用使用
在使用L ambda表达式的时候,我们实际上传递进去的代码就是-种解决方案:拿参数做操作 那么考虑一种情况:如果我们在l ambda中所指定的操作方案,已经有地方存在相同方案,那是否还有必要再写重复逻辑呢? 答案肯定是没有必要 那我们又是如何使用已经存在的方案的呢? 这就是我们要讲解的方法引用,我们是通过方法引用来使用已经存在的方案
需求,
1:定义一个接口(Printable)·里面定义一个抽象方法. void printstring(String s);
2:定义一个测试类(PrintableDemo).在测试类中提供兩个方法
一个方法是. usePrintable(Printable p)
一个方法是主方法, 在主方法中調用usePrintable方法
public interface Printable {
void printstring(String s);
}
public class PrintableDemo {
public static void main(String[] args) {
//一个方法是主方法, 在主方法中調用usePrintable方法
// usePrintable((String s) ->{
// System.out.println(s);
// });
//参数类型可以省略
//只有一个参数 小括号可以省略
//代码块中只有一条语句 {} 可以省略
//优化
usePrintable(s -> System.out.println(s));
//这里的s的类型是根据接口的 printstring中的参数类型推导出来的
System.out.println("system.out的输出结果 Java");
//用方法引用 来 改进 方法引用符 ::
usePrintable(System.out::println);
//可推导的就是可省略的
}
private static void usePrintable(Printable p) {
p.printstring("Java");
}
}
输出结果:
Java
system.out的输出结果 Java
Java
1.2方法引用符
方法引用符
- :: 该符号为引用运算符,而它所在的表达式被称为方法引用
- Lamdba表达式: usePrintable(s -> System.out.println(s)); 分析:拿到参数s之后通过Lambda表达式,传递给System.out.println方法去处理
- 方法引用:usePrintable(System.out::println); 分析:直接使用System.out中的println方法来取代Lambda,代码更加简洁
推导与省略
- 如果使用Lambda,那么根据“可推导就是可省略”的原则,无需指定参数类型,也无需指定的重载形式,它们都将被自动推导
- 如果使用方法引用,也是同样可以根据上下文进行推导
- 方法引用是Lambda的孪生兄弟
1.3Lambda表达式支持的方法引用
常见引用方式:
- 引用类方法
- 引用对象的实例方法
- 引用类的实例方法
- 引用构造器
1.4 引用类方法
就是引用类的静态方法
- 格式: 类名::静态方法
- 范例: Integer::parselnt Integer类的方法: public static int parseInt(String s)将此String转换为int类型数据
练习:
1.定义一个接口(Converter), 里面定义-个抽象方法
int convert(String s);
2.定义一个测试类(ConverterDemo), 在测试类中提供两个方法
一个方法是: useConverter(Converter c)
一个方法是主方法,在主方法中调用useConverter方法
public class ConverterDemo {
public static void main(String[] args) {
Converter c = new Converter() {
@Override
public int convert(String s) {
return 0;
}
};
useConvertre((String s) ->{
return Integer.parseInt(s);
});
//改进
useConvertre(s -> Integer.parseInt(s));
//引用类方法
useConvertre(Integer::parseInt);
//Lambda表达式被类方法替代的时候,它的形式参数全部传递给静态方法作为参数
//多个参数都是直接传递
}
private static void useConvertre(Converter c) {
int number = c.convert("6662");
System.out.println(number);
}
}
public interface Converter {
int convert(String s);
}
1.5引用方法的实例方法
引用对象的实例方法,其实就引用类中的成员方法
- 格式:对象::成员方法
- 范例: “helloWorld” :: toUpperCase Strin类中的方法:public String toUpperCase() 将此String所哟字符转化为大写
练习
1:定义一个类(PrintString),里面定义一个方法
public void printupper(String, s):把字符串参数变成大写的数据,然后在控制台输出
2:定义一个接口(Printer),里面定义一个抽象方法
void printUpperCase(String s)
3:定义一个测试类(PrinterDemo),在测试类中提供两个方法
一个方法是: usePrinter(Printer p)
一个方法是主方法,在主方法中实例PrintString,并引用对象的实例方法
public class PrinterDemo {
public static void main(String[] args) {
//引用对象的实例方法
PrintString ps = new PrintString();
usePrinter(ps::printupper);
//Lambda表达式被对象的实例方法替代的时候,它的形式参数全部传递给该方法作为参数
}
private static void usePrinter(Printer p) {
p.printUpperCase("HelloWorld");
}
}
public class PrintString {
//把字符串参数变成大写的数据,然后在控制台输出
public void printupper(String s) {
String result = s.toUpperCase();
System.out.println(result);
}
}
public interface Printer {
void printUpperCase(String s);
}
1.6引用类的实例方法
引用类的实例方法,其实就是引用类中的成员方法
- 格式:类名::成员方法
- 范例: String::substring String类中的方法:public String substring(int beginIndex, int endIndex) 从beginindex开始到endIndex结束,截取字符串。返回一个子串,子串长度为endIndex-beginIndex
练习
1:定义一个接口(MyString),里面定义一个抽象方法:
string. mySubstring(string s,int. x,int );。
2:定义-一个测试类(MyStringDemo),在测试类中提供两个方法
一个方法是, uselMyString(Mystring my)
一个方法是主方法,在主方法中调用seMyString方法
public class MyStringDemo {
public static void main(String[] args) {
//在主方法中调用seMyString方法
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);
}
}
public interface MyString {
String mySubString(String s, int x, int y);
}
1.7引用构造器
其实就是引用构造方法
- 格式:类名::new
- 范例:Student::new
练习
1:定义-个类(Student), 里面有两个成员变量(name, age)
并提供无参构造方法和带参构造方法, 以及成员变量对应的get和set方法
2:定义一个接口(StudentBuilder),里面定义一个抽象方法
Student build(String name, int age);
3:定乂一个测试类(StudentDemo),在测试类中提供両个方法
一个方法是: useStudentBuilder(StudentBuilder s)
一个方法是主方法,在主方法中调用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("Collin", 22);
System.out.println(s.getName() + " - " + s.getAge());
}
}
public interface StudentBuilder {
Student build(String name, int age);
}
public class Student {
private String name;
private int age;
public Student(String name, int age) {
super();
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;
}
}