Java8 中引入方法引用新特性,用于简化应用对象方法的调用, 方法引用是用来直接访问类或者实例的已经存在的方法或者构造方法。 方法引用提供了一种引用而不执行方法的方式,它需要由兼容的函数式接口构成的目标类型上下文。

计算时,方法引用会创建函数式接口的一个实例。 当 Lambda 表达式中只是执行一个方法调用时,不用 Lambda 表达式,直接通过方法引用的形式可读性更高一些。方法引用是一种更简洁易懂的 Lambda 表达式。

一、语法

目标引用:方法名称

目标引用:类名、实例对象

方法名称:实例方法名、静态方法名

等效 Lambda 的方法引用示例如下:

二、方法引用分类

Java8 中对于方法引用主要分为四大类:

  • 构造器引用

    Class::new

  • 静态方法引用

    Class::static_method

  • 实例对象方法引用

    Instance::method

  • 特定类型任意对象的实例方法引用

    Class:: method

三、构造器引用

语法是Class::new,或者更一般的Class< T >::new实例如下:

/**
* Lambda 表达式 实例化User 对象
*/
Supplier s01 =()->new User();
Function  f01 = (id)->{
return new User(id);
};
f01.apply(2019);
BiFunction bf01 = (id,uname)->{
return new User(id,uname);
};
bf01.apply(2019,"admin");

//方法引用 等价写法
Supplier s02 = User::new;
Function f02 = User::new;
f02.apply(2019);
BiFunction bf02 = User::new;
bf02.apply(2019,"admin");

四、静态方法引用

语法是Class::static_method,实例如下:

// 判断字符串是否为空串
Function f03 = (str)-> StringUtils.isBlank(str);
System.out.println(f03.apply(""));
// Base64 对输入字符串执行编码操作
Supplier s01 = ()->Base64.getEncoder();
s01.get().encodeToString("java8 is so easy!!!".getBytes());

Function f04 = StringUtils::isBlank;
System.out.println(f04.apply(""));

Supplier s02 = Base64::getEncoder;
s02.get().encodeToString("java8 is so easy!!!".getBytes());

五、实例对象方法引用

语法是Instance::method ,此时引用方法时必须存在实例,示例如下:

//方法引用构造User 对象
BiFunction bf02 = User::new;
User u02 = bf02.apply(2019,"admin");
// 调用实例对象方法
Supplier s01 = ()->u02.getUserName();
System.out.println(s01.get());

// 方法引用 等价Lambda 写法
Supplier s02 = u02::getUserName;
System.out.println(s02.get());

六、特定类型任意对象的实例方法引用

语法是Class:: method,第四种类型比较特殊,这里特定类型指多个对象类型说的,对象类型属于同一类类型通常是一个集合,元素类型一致,此时可以对类方法实现引用。

/**
 准备测试数据
**/
List list= Arrays.asList(10,2,30,5,8,10,60,99,101);
List emails = Arrays.asList("[email protected]"," ","","[email protected]");
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= 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 掌握后使用方法引用时代码会变得更加简洁、自然 。