::
双冒号在工作中经常能碰到,但不一定能叫上名字,它在Java中有个专业名词叫方法引用(Method References)。那方法引用具体是什么呢?
方法引用是什么
方法引用是Java 8随着Lambda表达式引入的新特性,它是仅包含一个方法调用的Lambda表达式的简写语法。它的作用是「通过名称引用一个现有方法」。与Lambda表达式相比,它紧凑、易于阅读。
如果使用方法引用,方法体只能有一个简单语句
代码语言:javascript复制List<String> messages = Arrays.asList("hello", "suncodernote", "methodrefernce");
messages.forEach(s -> {
System.out.println(s);
});
就像上面这段代码一样,方法体中只有一个System.out.println(s)
语句。
用方法引用简化上面的代码:
代码语言:javascript复制messages.forEach(System.out::println);
方法引用的种类
方法引用有以下4种类型:
- 静态方法引用
- 特定对象的方法引用
- 特定类的任意对象的方法引用
- 构造方法引用
下面用举个例子来说明一下这4种方法引用。
静态方法引用
静态方法引用就是通过方法引用调用静态方法。
如果使用Lambda表达式引用静态方法,格式大概是这样的(args) -> Class.staticMethod(args)。
如果使用静态方法引用,格式是这样的Class::staticMethod。
举个例子:
先创建一个User实体类
代码语言:javascript复制@NoArgsConstructor
@AllArgsConstructor
@Data
public class User {
private Integer age;
/**
* 比较年龄
*/
public static int compareByAge(User a , User b){
return a.getAge().compareTo(b.getAge());
}
/**
* 创建对象
*/
public static User createUser(Supplier<User> userSupplier){
return userSupplier.get();
}
}
测试类
代码语言:javascript复制public class MethodReferenceTest {
public static void main(String[] args) {
List<User> userList = new ArrayList<>();
userList.add(new User(10));
userList.add(new User(5));
userList.add(new User(8));
// 引用User静态方法compareByAge
userList.sort(User::compareByAge);
System.out.println(userList);
}
}
输出结果:
代码语言:javascript复制[User(age=5), User(age=8), User(age=10)]
特定对象的方法引用
引用特定对象的实例方法。
如果使用Lambda表达式引用特定对象的方法,格式大概是这样的(args)->obj.instanceMethod(args)。
如果使用方法引用,格式是这样的obj::instanceMethod。
举个例子:
新建一个User类的比较器类UserComparator比较用户的年龄age
代码语言:javascript复制public class UserComparator implements Comparator<User> {
@Override
public int compare(User o1, User o2) {
return o1.getAge().compareTo(o2.getAge());
}
}
测试类
代码语言:javascript复制public class MethodReferenceTest {
public static void main(String[] args) {
List<User> userList = new ArrayList<>();
userList.add(new User(10));
userList.add(new User(5));
userList.add(new User(8));
UserComparator userComparator = new UserComparator();
// 引用userComparator对象的compare方法
Collections.sort(userList, userComparator::compare);
}
}
特定类的任意对象的方法引用
对特定类型任意对象的实例方法的引用。
如果使用Lambda表达式引用特定类型任意对象的实例方法,格式大概是这样的(obj, args) -> obj.instanceMethod(args)。
如果使用方法引用,格式是这样的ObjectType::instanceMethod。
举个例子:
代码语言:javascript复制public class MethodReferenceTest {
public static void main(String[] args) {
List<String> messages = Arrays.asList("hello", "suncodernote", "methodrefernce");
//引用String 类的实例方法compareToIgnoreCase
Collections.sort(messages, String::compareToIgnoreCase);
messages.forEach(System.out::println);
}
}
构造方法引用
构造方法引用返回一个Supplier<T>
对象,通过supplier.get()
方法就可以获取到真正的对象。构造方法引用跟上面3个相比属于特例了,因为它没有直接调用已经存在的方法,而是跟了new
关键字。
如果使用Lambda表达式引用构造方法,格式大概是这样的(args) -> new ClassName(args)。
如果使用方法引用,格式是这样的ClassName::new。
举个例子:
代码语言:javascript复制// 创建一个User类对象
User user = User.createUser(User::new);
列表总结一下4种方法引用的格式:
种类 | 语法 | 示例 |
---|---|---|
静态方法引用 | ContainingClass::staticMethodName | User::compareByAge |
构造方法引用 | ClassName::new | User::new |
特定对象的方法引用 | containingObject::instanceMethodName | userComparator::compare |
特定类的任意对象的方法引用 | ContainingType::methodName | String::compareToIgnoreCase |
总结
方法引用是Java 8随着Lambda表达式引入的新特性,可能有些同学还不习惯这种编码方式,不过还好IDE可以提示我们简化或者通过IDE的功能直接转成Lambda和方法引用以简化代码。