大家好,我是乐字节的小乐,上一次我们说到了 Java8核心特性之函数式接口 ,接下来我们继续了解Java8又一核心特性——方法引用。
Java8 中引入方法引用新特性,用于简化应用对象方法的调用, 方法引用是用来直接访问类或者实例的已经存在的方法或者构造方法。 方法引用提供了一种引用而不执行方法的方式,它需要由兼容的函数式接口构成的目标类型上下文。计算时,方法引用会创建函数式接口的一个实例。 当Lambda表达式中只是执行一个方法调用时,不用Lambda表达式,直接通过方法引用的形式可读性更高一些。方法引用是一种更简洁易懂的Lambda表达式。
目标引用::方法名称
目标引用:类名、实例对象
方法名称:实例方法名、静态方法名
等效Lambda的方法引用示例如下:
Java8 中对于方法引用主要分为四大类:
构造器引用
Class::new
静态方法引用
Class::static_method
实例对象方法引用
Instance::method
特定类型任意对象的实例方法引用
Class:: method
语法是 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");
语法是 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());
语法是 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());
语法是 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());
方法引用简化了Lambda表达式书写方式,对于函数式接口实现可以使用方法引用来替换Lambda ,当对Lambda掌握后使用方法引用时代码会变得更加简洁、自然 。
感谢光临阅读小乐的Java8 ,敬请关注 乐字节 后续将继续讲述Java8、Java12等前沿知识技术。