大家好,我是乐字节的小乐,上一次我们说到了 Java8 核心特性之函数式接口,接下来我们继续了解 Java8 又一核心特性——方法引用。
Java8 中引入方法引用新特性, 用于简化应用对象方法的调用,方法引用是用来直接访问类或者实例的已经存在的方法或者构造方法。方法引用提供了一种引用而不执行方法的方式,它需要由兼容的函数式接口构成的目标类型上下文。计算时,方法引用会创建函数式接口的一个实例。当 Lambda 表达式中只是执行一个方法调用时,不用 Lambda 表达式,直接通过方法引用的形式可读性更高一些。方法引用是一种更简洁易懂的 Lambda 表达式。
1、语法
目标引用:: 方法名称
目标引用: 类名、实例对象
方法名称: 实例方法名、静态方法名
等效 Lambda 的方法引用示例如下:
2、方法引用分类
Java8 中对于方法引用主要分为四大类:
- 构造器引用
Class::new
- 静态方法引用
Class::static_method
- 实例对象方法引用
Instance::method
- 特定类型任意对象的实例方法引用
Class:: method
3、构造器引用
语法是 Class::new
,或者更一般的Class< T >::new
实例如下
/**
* Lambda 表达式 实例化 User 对象
*/
Supplier<User> s01 =()->new User();
Function<Integer,User> f01 = (id)->{return new User(id);
};
f01.apply(2019);
BiFunction<Integer,String,User> bf01 = (id,uname)->{return new User(id,uname);
};
bf01.apply(2019,"admin");
// 方法引用 等价写法
Supplier<User> s02 = User::new;
Function<Integer,User> f02 = User::new;
f02.apply(2019);
BiFunction<Integer,String,User> bf02 = User::new;
bf02.apply(2019,"admin");
4、静态方法引用
语法是Class::static_method
,实例如下:
// 判断字符串是否为空串
Function<String,Boolean> f03 = (str)-> StringUtils.isBlank(str);
System.out.println(f03.apply(""));
// Base64 对输入字符串执行编码操作
Supplier<Base64.Encoder> s01 = ()->Base64.getEncoder();
s01.get().encodeToString("java8 is so easy!!!".getBytes());
Function<String,Boolean> f04 = StringUtils::isBlank;
System.out.println(f04.apply(""));
Supplier<Base64.Encoder> s02 = Base64::getEncoder;
s02.get().encodeToString("java8 is so easy!!!".getBytes());
5、实例对象方法引用
语法是Instance::method
,此时引用方法时必须存在实例,示例如下
// 方法引用构造 User 对象
BiFunction<Integer,String,User> bf02 = User::new;
User u02 = bf02.apply(2019,"admin");
// 调用实例对象方法
Supplier<String> s01 = ()->u02.getUserName();
System.out.println(s01.get());
// 方法引用 等价 Lambda 写法
Supplier<String> s02 = u02::getUserName;
System.out.println(s02.get());
6、特定类型任意对象的实例方法引用
语法是Class:: method
,第四种类型比较特殊,这里特定类型指多个对象类型说的,对象类型属于同一类类型 通常是一个集合,元素类型一致,此时可以对类方法实现引用。
/**
准备测试数据
**/
List<Integer> list= Arrays.asList(10,2,30,5,8,10,60,99,101);
List<String> emails = Arrays.asList("126@126.com","","","java8@163.com");
Goods g01=new Goods(1,"小米 9",1789,200, BigDecimal.valueOf(2500));
Goods g02=new Goods(2,"华为 Mate20",5000,3000, BigDecimal.valueOf(7000));
Goods g03=new Goods(3,"OPPO R17",2000,2827, BigDecimal.valueOf(1500));
Goods g04=new Goods(4,"魅族 Note9",2000,1600, BigDecimal.valueOf(1600));
Goods g05=new Goods(5,"一加 6T",8000,5000, BigDecimal.valueOf(3500));
List<Goods> goods= Arrays.asList(g01,g02,g03,g04,g05);
// Lambda 对集合 list 元素排序 list 存放整数元素
list.sort((a1,a2)->a1-a2);
// Lambda 对集合 goods 按销量排序
goods.sort((g1,g2)->g1.getSale()-g2.getSale());
// 过滤邮箱集合空串
emails.stream().filter((e) -> StringUtils.isNoneBlank(e))
.collect(Collectors.toList());
// 任意对象的实例方法引用 对集合 list 元素排序
list.sort(Comparator.comparing(Integer::intValue));
// 任意对象的实例方法引用 对集合 goods 按销量排序
goods.sort(Comparator.comparing(Goods::getSale));
// 过滤邮箱集合空串
emails.stream().filter(StringUtils::isNoneBlank).collect(Collectors.toList());
7、方法引用应用
方法引用简化了 Lambda 表达式书写方式,对于函数式接口实现可以使用方法引用来替换 Lambda , 当对 Lambda 掌握后使用方法引用时代码会变得更加简洁、自然。
感谢光临阅读小乐的 Java8,敬请关注 乐字节 后续将继续讲述 Java8、Java12 等前沿知识技术。