方法引用
Lambda表达式中,提供了特殊的语法,能让我们直接引用已经存在的方法,作为当前要表示的函数,例如
public class Test {
public static void main(String[] args) {
//通过lambda表达式表示一个函数,这个函数作为run方法的具体实现
Action a1 = () -> System.out.println("hello!");
//方法引用:被引用的Student类中的sayHello方法作为run方法的具体实现
Action a2 = Student::sayHello;
//调用接口中的run()方法
a1.run();
a2.run();
}
}
interface Action{
void run();
}
class Student{
public static void sayHello(){
System.out.println("hello!");
}
}
方法引用和lambda表达式一样,需要保证接口时函数式接口才可以使用。
静态方法引用
使用Lambda表达式可以引用类中的静态方法,语法要求为:
类名::静态方法名
注意:方法名字后面一定没有小括号,因为这不是在调用方法,而是在引用这个方法的定义
例如:
public class Test {
public static void main(String[] args) {
//Lambda表达式中只要参数是String类型,返回值是int类型就可以作为Action接口的具体实现
Action a1 = str -> str.length();
//使用静态方法引用(类名::静态方法名)来引用当前类中的静态方法len
Action a2 = Test::len;
System.out.println(a1.run("hello")); //输出5
System.out.println(a2.run("hello")); //输出5
}
public static int len(String str){
return str.length();
}
}
interface Action{
int run(String str);
}
实例方法引用
使用Lambda表达式可以引用类中的非静态方法,语法如下:
类名::非静态方法名
例如:
public class Test {
public static void main(String[] args) {
//引用的非静态方法中没有参数
Action a1 = String::length;
/* 等价于:
Action a1 = new Action() {
@Override
public int run(String str) {
return str.length();
}
};
*/
//引用的非静态方法中有一个参数
Index i1 = String::indexOf;
/* 相当于:
Index i1 = new Index(){
@Override
public int run(String str, String s) {
return str.indexOf(s);
}
};
*/
System.out.println(a1.run("hello")); //输出5
System.out.println(i1.run("hello","e")); //输出1
}
}
interface Action{
//该方法中只有一个参数
int run(String str);
}
interface Index{
//该方法中有两个参数
int run(String str, String s);
}
补充:
(1)对于int run(String str)方法来讲,可以直接引用String类中的length()方法来表示对run()方法的实现,需要注意的是run方法中有一个参数,而length方法是没有参数的,run方法中的参数是到时要调用方法的对象,相当于public int run(String str){ return str.xxx(); }
(2)对于int run(String str,String s)来讲,可以直接引用String类中的inidexOf(int ch)方法来表示对run方法的实现,需要注意该run方法中有两个参数,且indexOf方法中也有一个参数,可以理解为run方法中的第一个参数是到时要调用方法的对象,后一个参数是传入到indexOf方法中的参数。相当于public int run(String str, String i) { return str.xxx(i); }
-----当被调用的方法中参数变多,那么同样的,接口中方法的参数也要变多,且第一个参数为想要调用该方法的那个对象。
对象引用方法
语法格式为:
对象::非静态方法
例如:
public class Test {
public static void main(String[] args) {
/* 匿名内部类实现
Action a1 = new Action() {
@Override
public String run(String str) {
return "ByeBye! " + str;
}
};
*/
//Lambda表达式简化
Action a1 = str -> "ByeBye! " + str;
//对象引用方法
MyHandler myHandler = new MyHandler();
Action a2 = myHandler::test;
System.out.println(a1.run("Tom"));;
System.out.println(a2.run("John"));
}
}
interface Action{
String run(String str);
}
class MyHandler{
public String test(String name){
return "hello! " + name;
}
}
其实这时候,任何一个对象的中的方法,只要是参数列表和返回类型和run方法保持一致,都可以使用这个对象中的这个方法,来对Action接口进行实现。
构造方法引用
语法格式
类名::new
无参构造器
public class Test {
public static void main(String[] args) {
/* 匿名内部类实现
Action a1 = new Action() {
@Override
public Student run() {
return new Student();
}
};
*/
//Lambda表达式简化
//Action a1 = ()->new Student();
//使用Student类的无参构造函数对Action接口进行实现
Action a1 = Student::new;
System.out.println(a1.run());
}
}
interface Action{
//返回值是Student类型
Student run();
}
class Student{
}
含参构造器
public class Test {
public static void main(String[] args) {
/* 匿名内部类实现
Action a1 = new Action() {
@Override
public Student run(String name) {
return new Student(name);
}
};
*/
//Lambda表达式简化
//Action a1 = name->new Student(name);
//使用Student类的含参构造器对Action接口进行实现
Action a1 = Student::new;
System.out.println(a1.run("tom"));
//输出的是student对象中的name属性值tom所在的位置
}
}
interface Action{
//返回值是Student类型
Student run(String name);
}
class Student{
private String name;
//含参构造器
public Student(String name){
this.name = name;
}
}
数组构造
使用lambda表达式可以引用数组类型的构造函数,语法要求为
数组类型::new
例如:
public class Test {
public static void main(String[] args) {
/* 匿名内部类实现
Action a1 = new Action() {
@Override
public int[] run(int len) {
return new int[len];
}
};
*/
//Lambda表达式简化
//Action a1 = len->new int(len);
//使用int数组的构造函数,来对Action接口进行实现
//int数组的构造函数恰好是和run方法的参数列表和返回类型保持一致,所以可引用
Action a1 = int[]::new;
System.out.println(a1.run(5)); //输出的是数组所在的内存地址
System.out.println(Arrays.toString(a1.run(5)); //输出的是数组值,因为没有传值,默认值都是0,即[0, 0, 0, 0,0]
}
}
interface Action{
//返回值是数组类型
int[] run(int len);
}