目录
初识方法引用
方法引用的分类
引用静态方法
引用成员方法
引用构造方法
其它调用方式
类名引用成员方法
引用数组的构造方法
总结
初识方法引用
方法引用就是拿现有的方法来当做函数式接口中抽象方法的方法体。
方法引用注意事项
1. 引用处必须是函数式接口;
2. 被引用的方法必须已经存在;
3. 被引用方法的形参和返回值需要跟抽象方法保持一致;
4. 被引用方法的功能要满足当前需求。
方法引用代码示例如下
public class Demo {
public static void main(String[] args) {
Integer[] arr = {2, 1, 3, 6, 4, 5};
//Lambda表达式对数组进行升序排序
Arrays.sort(arr, (o1, o2) -> o1 - o2);
System.out.println(Arrays.toString(arr)); //[1, 2, 3, 4, 5, 6]
//方法引用对数组进行降序排序
Arrays.sort(arr, Demo::subtraction);
System.out.println(Arrays.toString(arr)); //[6, 5, 4, 3, 2, 1]
}
public static int subtraction(int num1, int num2) {
return num2 - num1;
}
}
方法引用的分类
引用静态方法
格式:类名::静态方法
范例:Integer::parseInt
代码示例如下
//方法引用:引用静态方法
//需求:将集合中的字符串类型数字转换成int类型
ArrayList<String> list1 = new ArrayList<>();
Collections.addAll(list1, "1", "2", "3", "4", "5", "6");
//常规方法
ArrayList<Integer> list2 = new ArrayList<>();
for (String s : list1) {
list2.add(Integer.parseInt(s));
}
//Lambda表达式
list1.stream().map(s -> Integer.parseInt(s));
//方法引用
list1.stream().map(Integer::parseInt);
引用成员方法
格式:对象::成员方法
成员方法的引用又可分为以下三类
1. 引用其它类的成员方法:其他类对象::方法名
2. 引用本类的成员方法:this::方法名
3. 引用父类的成员方法:super::方法名
注意:在静态方法中无法使用this和super关键字。因此,若想在静态方法中引用本类或父类的成员方法,需要通过创建对象来调用这些方法。
代码示例如下
public class Demo {
public static void main(String[] args) {
//需求:保留集合中以”张“开头,且长度为3的字符串
ArrayList<String> list = new ArrayList<>();
Collections.addAll(list, "张无忌", "周芷若", "张强", "赵敏", "张三丰");
//Lambda表达式
list.stream().filter(s -> s.startsWith("张") && s.length() == 3);
//引用本类成员方法
//list.stream().filter(this::stringJudge); 报错:静态方法中无法使用this关键字
list.stream().filter(new Demo()::stringJudge);
}
public boolean stringJudge(String s) {
return s.startsWith("张") && s.length() == 3;
}
}
引用构造方法
格式:类名::new
范例:Student::new
注意:被引用构造方法的形参需要跟抽象方法保持一致
代码示例如下
//需求:将字符串集合转换为Student集合
ArrayList<String> list = new ArrayList<>();
Collections.addAll(list, "张无忌-16", "周芷若-17", "张强-12", "赵敏-20", "张三丰-22");
//匿名内部类写法
list.stream().map(new Function<String, Student>() {
@Override
public Student apply(String s) {
return new Student(s.split("-")[0], Integer.parseInt(s.split("-")[1]));
}
});
/*
* 注意:被引用方法的形参和返回值需要跟抽象方法保持一致
* 由匿名内部类可知:抽象方法为 public Student apply(String s)
* 因此我们需要编写一个Student的构造方法,令它的形参和抽象方法的形参保持一致,
* 由于是构造方法,因此不需要返回任何值,只需要保证构造方法创建的对象与抽象方法返回的类型一致即可
* 于是编写构造方法如下
* public Student(String str) {
* this.name = str.split("-")[0];
* this.age = Integer.parseInt(str.split("-")[1]);
* }
*/
//构造方法引用
List<Student> newList = list.stream()
.map(Student::new)
.collect(Collectors.toList());
System.out.println(newList);
其它调用方式
类名引用成员方法
格式:类名::成员方法
范例:String::substring
注意1:类名引用成员方法同样需要满足以下条件
1. 引用处必须是函数式接口;
2. 被引用的方法必须已经存在;
3. 被引用方法的功能要满足当前需求。
注意2:除此之外,还有一些其它规则,具体见下图。
局限性
不能引用所有类的成员方法,只能引用抽象方法中第一个参数所属类中的成员方法。
引用数组的构造方法
作用:创建一个指定类型的数组
格式:数据类型[]::new
范例:int[]::new
代码示例如下
public static void main(String[] args) {
//需求:将集合中的元素保存到数组中
ArrayList<Integer> list = new ArrayList<>();
Collections.addAll(list, 1, 2, 3, 4);
//匿名内部类
Integer[] arr1 = list.stream().toArray(new IntFunction<Integer[]>() {
@Override
public Integer[] apply(int value) {
return new Integer[value];
}
});
//数组构造方法引用
Integer[] arr2 = list.stream().toArray(Integer[]::new);
}