方法引用基本介绍与方法引用_通过对象名引用成员方法
方法引用基本介绍
在使用Lambda表达式的时候,我们实际上传递进去的代码就是一种解决方案∶拿什么参数做什么操作。那么考虑一种情况∶如果我们在Lambda中所指定的操作方案,已经有地方存在相同方案,那是否还有必要再写重复逻辑?
package day01.Demo01_Day016.Demo01_JieKou; public class Demo01_Printbale { //定义一个方法,参数传递Printbale接口,对字符串进行打印 public static void PrinString(Printbale p){ p.print("HellWorld"); } public static void main(String[] args) { //调用PrinString方法,方法的参数Printbale是一个函数式接口,所以可以传递Lambda PrinString(s -> { System.out.println(s); }); /* Lambda表达式的目的,打印参数传递的字符串 把参数s ,传递给了System.out对象,调用out对象中的方法printLn对字符串进行了输出注意: 1.System.out对象是已经存在的 2.printLn方法也是已经存在的 所以我们可以使用方法引用来优化Lambda表达式 可以使用System.out方法直接引用(调用)println方法 */ PrinString(System.out::println); } }
方法引用符
双冒号:为引用运算符,而它所在的表达式被称为方法引用。如果Lambda要表达的函数方案已经存在于某个方法的实现中,那么则可以通过双冒号来引用该方法作为Lambda的替代者。+
语义分析
例如上例中,System.out对象中有一个重载的println(String)方法恰好就是我们所需要的。那么对于printstring方法的函数式接口参数,对比下面两种写法,完全等效︰
Lambda表达式写法: s -> System.out.println( s ) ;
·方法引用写法:System.out: : println
第一种语义是指︰拿到参数之后经Lambda之手,继而传递给system.ouit.print1n方法去处理。
第二种等效写法的语义是指:直接让sysfem .out 中的println方法来取代Lambda。两种写法的执行效果完全一样,而第二种方法引用的写法复用了已有方案,更加简洁。
注:Lambda中传递的参数一定是方法引用中的那个方法可以接收的类型,否则会抛出异常
推导与省略
如果使用Lambda,那么根据“可推导就是可省略”的原则,无需指定参数类型,也无需指定的重载形式——它们都
将被自动推导。而如果使用方法引用,也是同样可以根据上下文进行推导。
函数式接口是Lambda的基础,而方法引用是Lambda的孪生兄弟。
通过对象名引用成员方法
package day01.Demo01_Day016.Demo01_JieKou; /* 通过对象名引用成员方法 使用前提是对象名是已经存在的,成员方法也是已经存在 就可以使用对象名来引用成员方法 */ public class Demo01objctMethodReference { //定义一个方法,方法的参数传递Printbale接口 public static void printString(Printbale p){ p.print("Hello"); } public static void main(String[] args) { //调用printString方法,方法的参数Printable是一个函数式接口,所以可以传递Lambda表达式 printString((s)->{ //创建MethodRerObject对象 MethodRerObject object = new MethodRerObject(); //调用MethodRerObject对象中的成员方法printUpperCaseString,把字符串按照大写输出 object.printUpperCaseString(s); }); /* 使用方法引用优化Lambda 对象是已经存在的MethodRerObject 成员方法也是已经存在的printUpperCaseString 所以我们可以使用对象名引用成员方法 */ //创建MethodRerObject对象; MethodRerObject obj = new MethodRerObject(); printString(obj::printUpperCaseString); } }
package day01.Demo01_Day016.Demo01_JieKou;
/*
通过对象名引用成员方法
使用前提是对象名是已经存在的,成员方法也是已经存在
就可以使用对象名来引用成员方法
*/
public class Demo01objctMethodReference {
//定义一个方法,方法的参数传递Printbale接口
public static void printString(Printbale p){
p.print("Hello");
}
public static void main(String[] args) {
//调用printString方法,方法的参数Printable是一个函数式接口,所以可以传递Lambda表达式
printString((s)->{
//创建MethodRerObject对象
MethodRerObject object = new MethodRerObject();
//调用MethodRerObject对象中的成员方法printUpperCaseString,把字符串按照大写输出
object.printUpperCaseString(s);
});
/*
使用方法引用优化Lambda
对象是已经存在的MethodRerObject
成员方法也是已经存在的printUpperCaseString
所以我们可以使用对象名引用成员方法
*/
//创建MethodRerObject对象;
MethodRerObject obj = new MethodRerObject();
printString(obj::printUpperCaseString);
}
}