:: 双冒号在工作中经常能碰到,但不一定能叫上名字,它在Java中有个专业名词叫方法引用(Method References)。那方法引用具体是什么呢?
方法引用是什么
方法引用是Java 8随着Lambda表达式引入的新特性,它是仅包含一个方法调用的Lambda表达式的简写语法。它的作用是通过名称引用一个现有方法。与Lambda表达式相比,它紧凑、易于阅读。
如果使用方法引用,方法体只能有一个简单语句
List<String> messages = Arrays.asList("hello", "suncodernote", "methodrefernce");messages.forEach(s -> { System.out.println(s);});就像上面这段代码一样,方法体中只有一个System.out.println(s)语句
用方法引用简化上面的代码:
messages.forEach(System.out::println);方法引用的种类
方法引用有以下4种类型:
- 静态方法引用
- 特定对象的方法引用
- 特定类的任意对象的方法引用
- 构造方法引用
下面用举个例子来说明一下这4种方法引用。
静态方法引用
静态方法引用就是通过方法引用调用静态方法。
如果使用Lambda表达式引用静态方法,格式大概是这样的(args) -> Class.staticMethod(args)
如果使用静态方法引用,格式是这样的Class::staticMethod
举个例子:
先创建一个User实体类
@NoArgsConstructor@AllArgsConstructor@Datapublic 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(); }}测试类
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); }}输出结果:
[User(age=5), User(age=8), User(age=10)]特定对象的方法引用
引用特定对象的实例方法。
如果使用Lambda表达式引用特定对象的方法,格式大概是这样的(args) -> obj.instanceMethod(args)
如果使用方法引用,格式是这样的obj::instanceMethod
举个例子:
新建一个User类的比较器类UserComparator比较用户的年龄age
public class UserComparator implements Comparator<User> { @Override public int compare(User o1, User o2) { return o1.getAge().compareTo(o2.getAge()); }}测试类
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
举个例子:
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
举个例子:
// 创建一个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和方法引用以简化代码。
