java8 方法引用详解_Java8方法引用及构造方法引用原理实例解析

如果不熟悉Java8新特性的小伙伴,初次看到函数式接口写出的代码可能会是一种懵逼的状态,我是谁,我在哪,我可能学了假的Java,(・∀・(・∀・(・∀・*),但是语言都是在进步的,就好比面向对象的语言Java也可以写出优雅的函数式调用,学习的过程并不复杂,当你学会了Java8中函数式编程的新特性,你一定会对他爱不释手的。下面介绍一下基于Lambda表达式简写的两种引用。避免再次看到这种代码时的尴尬😅。

方法引用

方法引用,一般包含下面三种写法,传统的写法我们可能都是通过对象.去调用实例方法或使用类.调用静态方法,但是学完方法引用后,就可以可以使用这三种方式去调用方法,但是要符合一定的规则。

对象::实例方法

/**

* 对象调用实例方法

*/

public static void objMethod(){

List list = new ArrayList<> ();

list.add(1);

list.add(2);

list.add(3);

list.forEach((i)->{

PrintStream out = System.out;

Consumer consumer = out::println;

consumer.accept(i);

});

list.forEach(System.out::println);

}

最常用的System.out.println

类::实例方法

/**

* 判断两个字符串是否相同

*

* @param str1

* @param str2

* @return

*/

public static boolean isEqual(String str1, String str2) {

BiPredicate b = (s1,s2)->s1.equals(str2); ①

BiPredicate bp = String::equals;

return bp.test(str1, str2);

}

类::静态方法

/**

* 比较大小

* @param x

* @param y

* @return

*/

public static boolean compareValue(int x, int y){

Comparator compare = Integer::compare; ②

return compare.compare(x, y) > 0;

}

其实不管是哪一种调用方式都是有规律可循的,这里总结一下在使用Lambda表达式的过程中符合什么样的规则才可以使用方法引用的模式去写。

Lambda体中调用方法的参数列表与返回值类型,要与函数式接口中抽象方法的函数列表和返回值类型保持一致 Integer::compare ②

Lambda参数列表中的第一参数是实例方法的调用者,而第二个参数是实例方法的参数时 可以使用ClassName::method ①

构造方法引用#

简称花式new对象,一个简单的new对象也要写的高端、大气、上档次😄,既可以掌握新知识,又可以ZB,赶紧学习吧。

ClassName::new

资源类:

public class Apple {

private String color;

private double weight;

public Apple(){

}

public Apple(String color) {

this.color = color;

}

public Apple(double weight) {

this.weight = weight;

}

public Apple(String color, double weight) {

this.color = color;

this.weight = weight;

}

public String getColor() {

return color;

}

public void setColor(String color) {

this.color = color;

}

public double getWeight() {

return weight;

}

public void setWeight(double weight) {

this.weight = weight;

}

@Override

public String toString() {

return "Apple{" +

"color='" + color + '\'' +

", weight=" + weight +

'}';

}

}

测试代码:

public static void main(String[] args) {

//无参构造

//Supplier supplier = () -> new Apple(); Lambda表达式写法

Supplier supplier = Apple::new;

Apple apple = supplier.get();

System.out.println("NoArgsConstructor: "+apple);

//有参构造

//Function function = (x) -> new Apple(x); Lambda表达式写法

// 构造引用

Function function = Apple::new;

Apple apply = function.apply(1.0);

System.out.println("OneArgsConstructor: "+apply);

BiFunction bf = Apple::new;

Apple bi = bf.apply("Red", 2.0);

System.out.println("TwoArgsConstructor: "+bi);

}

输出结果:

NoArgsConstructor: Apple{color='null', weight=0.0}

OneArgsConstructor: Apple{color='null', weight=1.0}

TwoArgsConstructor: Apple{color='Red', weight=2.0}

当构造方法无参时使用Supplier,有一个参数时使用Function,两个参数时使用BiFunction。这里很容易得出一个规律,当使用构造方法引用时,函数式接口的参数列表需要和构造方法的参数列表保持一致。

我们也可以用这些函数式接口改写传统的创建数组的方式,初始化一个指定长度的数组,比如

Function fun = String[]::new;

String[] strArr = fun.apply(10);

也可以这样写:

public static T[] initArray(int num, Function function){

return function.apply(num);

}

调用:

Copy

Apple[] strings = initArray(10, x -> new Apple[x]);

System.out.println(strings.length);

疑惑

根据传入的参数返回指定的对象数组引用,不过这样还不如直接创建。不知道读者有没有考虑这里为什么不可以用一个泛型来new,那样就可以创建一个通用数组引用,但是Java中的泛型是伪泛型,在编译器就会进行泛型擦除,所以不能通过new关键字来创建一个泛型对象,具体内容可以在查阅其他资料了解泛型以及泛型擦除的原理,这里不做深究。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值