如果不使用Lambda表达式进行程序编写的话,那么大可不必关注方法引用和构造器引用,但是如果使用Lambda表达式,再配合方法引用和构造器引用之后,那么就可以使Lambda编写匿名内部类的代码变得更加简洁。在不影响性能的前提下简洁的代码可以增强代码的可读性,当然是在阅读者知晓对方语法的前提下。
方法引用 什么是方法引用?当要传递给Lambda体的 *** 作,已经有实现的方法了,那么这时便可以使用方法引用了。或者,你也可以理解为方法引用是Lambda表达式的另外一种表现形式。
不过有一点需要我们注意,就是Lambda体中调用方法的参数列表与返回值类型,要与函数式接口中抽象方法的参数列表与返回值类型保持一致!
方法引用的语法格式方法引用使用 *** 作符::将方法名和对象或类的名字分隔开来。而且,方法引用主要有如下三种固定语法格式。
- 第一种语法格式:对象::实例方法名。
- 第二种语法格式:类::静态方法名。
- 第三种语法格式:类::实例方法名。
接下来,我会为大家分别一一介绍这三种语法格式。
第一种语法格式:对象::实例方法名观察如下Java程序,你会发现在Lambda体中有一个println方法已经完成了我们要 *** 作的功能,即完成了在Lambda体中所写的功能。
package com.meimeixia.java8; import org.junit.Test; import java.io.PrintStream; import java.util.function.Consumer; public class TestMethodRef { // 对象::实例方法名 @Test public void test1() { // 未使用方法引用时 // Consumercon = (x) -> System.out.println(x); PrintStream ps1 = System.out; Consumer con = (x) -> ps1.println(x); } }
那么在这种情况下我们就可以使用方法引用了!
package com.meimeixia.java8; import org.junit.Test; import java.io.PrintStream; import java.util.function.Consumer; public class TestMethodRef { // 对象::实例方法名 @Test public void test1() { // 未使用方法引用时 // Consumercon = (x) -> System.out.println(x); PrintStream ps1 = System.out; Consumer con = (x) -> ps1.println(x); // 使用方法引用时 PrintStream ps = System.out; Consumer con1 = ps::println; // 以上还可以简写为下面这样,相信大家应该都看得懂 Consumer con2 = System.out::println; con2.accept("abcdef"); } }
在使用方法引用时,有一个前提,大家得注意一下,就是需要实现的接口(例如Consumer
再来看另外一个案例,使用Lambda表达式来实现供给型接口,如下所示。
@Test public void test2() { // 未使用方法引用时 Employee emp = new Employee(); Suppliersup = () -> emp.getName(); String str = sup.get(); System.out.println(str); }
其中,Employee实体类的代码如下所示。
package com.meimeixia.java8; public class Employee { private int id; private String name; private int age; private double salary; public Employee() { } public Employee(int id, String name, int age, double salary) { this.id = id; this.name = name; this.age = age; this.salary = salary; } public int getId() { return id; } public void setId(int id) { this.id = id; } 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; } public double getSalary() { return salary; } public void setSalary(double salary) { this.salary = salary; } @Override public String toString() { return "Employee{" + "id=" + id + ", name='" + name + ''' + ", age=" + age + ", salary=" + salary + '}'; } }
大家发现没有在以上Lambda体中是不是已经有实现的方法了啊,所以这时我们就可以使用方法引用了。
@Test public void test2() { // 未使用方法引用时 Employee emp = new Employee(); Supplier第二种语法格式:类::静态方法名sup = () -> emp.getName(); String str = sup.get(); System.out.println(str); // 使用方法引用时 Supplier sup2 = emp::getAge; Integer num = sup2.get(); System.out.println(num); }
使用方法引用的这种格式,可以引用类的静态方法,就像下面这样。
// 类::静态方法名 @Test public void test3() { // 未使用方法引用时 Comparatorcom = (x, y) -> Integer.compare(x, y); // 使用方法引用时 Comparator com1 = Integer::compare; }
如果你怀疑上述使用方法引用时的写法,那么不妨去查看一下Integer类中compare方法的参数列表与返回值类型是否和Comparator接口中compare方法的参数列表与返回值类型保持一致。不用想,肯定是保持一致的,保持一致就可以像上面这样使用方法引用来书写代码了。
第三种语法格式:类::实例方法名使用方法引用的这种格式得有一个前提,也可以说是规则,即若Lambda表达式参数列表中的第一个参数是实例方法的调用者,而第二个参数是实例方法的参数时,则可使用ClassName::method这种格式来书写代码。不知道,大家能记住嘛?
// 类::实例方法名 @Test public void test4() { // 未使用方法引用时,来比较两个字符串是不是一样的。注意,BiPredicate接口是Predicate接口的子接口 BiPredicate构造器引用bp = (x, y) -> x.equals(y); // 使用方法引用时 BiPredicate bp2 = String::equals; }
构造器引用的语法格式是ClassName::new。它与函数式接口相结合,自动与函数式接口中方法兼容,可以把构造器引用赋值给定义的方法,只不过构造器的参数列表要与接口中抽象方法的参数列表保持一致!
为了演示构造器引用的使用,我们在Employee实体类中多写几个构造方法,如下所示。
package com.meimeixia.java8; public class Employee { private int id; private String name; private int age; private double salary; public Employee() { } public Employee(int id) { this.id = id; } public Employee(int id, int age) { this.id = id; this.age = age; } public Employee(int id, String name, int age, double salary) { this.id = id; this.name = name; this.age = age; this.salary = salary; } public int getId() { return id; } public void setId(int id) { this.id = id; } 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; } public double getSalary() { return salary; } public void setSalary(double salary) { this.salary = salary; } @Override public String toString() { return "Employee{" + "id=" + id + ", name='" + name + ''' + ", age=" + age + ", salary=" + salary + '}'; } }
然后,我们就能使用构造器引用的方式来书写代码了。
// 构造器引用 @Test public void test5() { // 未使用构造器引用时 Suppliersup = () -> new Employee(); // 使用构造器引用时 Supplier sup2 = Employee::new; Employee emp = sup2.get(); System.out.println(emp); }
不知你有没有想过,Employee实体类中有好几个构造器,那么现在调用的是哪个构造器呢?道理其实跟方法引用一样,由于构造器的参数列表要与接口中抽象方法的参数列表保持一致,所以此时调用的必然就是无参的构造器,也就是说Java 8会根据函数式接口中抽象方法的参数列表自动匹配对应的构造器。
如果此时我们想要调用Employee实体类中一个参数的构造器,那么该怎么办呢?是不是该像下面这样做啊!
@Test public void test6() { Functionfun = (x) -> new Employee(x); Function fun2 = Employee::new; // 注意,此时调用的是Employee实体类中一个参数的构造器 Employee emp = fun2.apply(101); System.out.println(emp); }
举一反三,大家试着思考一下,如果此时想要调用Employee实体类中两个参数的构造器,那么又该怎么办呢?很简单嘛,代码像下面这样写就行了。
@Test public void test6() { Function数组引用fun = (x) -> new Employee(x); Function fun2 = Employee::new; // 注意,此时调用的是Employee实体类中一个参数的构造器 Employee emp = fun2.apply(101); System.out.println(emp); // BiFunction接口是Function接口的子接口 BiFunction bf = Employee::new; // 注意,此时调用的是Employee实体类中两个参数的构造器 }
数组引用的语法格式是Type[]::new。
下面我会通过一个案例来简单使用一下数组引用。
// 数组引用 @Test public void test7() { // 未使用数组引用时 Functionfun = (x) -> new String[x]; String[] strs = fun.apply(10); System.out.println(strs.length); // 使用数组引用时 Function fun2 = String[]::new; String[] strs2 = fun2.apply(20); System.out.println(strs2.length); }
至此,Java 8中有关Lambda表达式的所有内容我就算是总结完了!接下来咱们就要开始学习Java 8中另外一个最核心的新特性(即Stream API)了。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)