Java8新特性之方法引用中的双冒号
目录
简介
双冒号运算操作符是类方法的引用,lambda 表达式的一种简写,这种简写的学名叫 eta-conversion 或者叫 η-conversion。
先看一个简单的例子初步了解下:
把 x -> System.out.println(x) 简化为 System.out::println 的过程称之为 eta-conversion
把 System.out::println 简化为 x -> System.out.println(x) 的过程称之为 eta-expansion
上面这里看不懂也没有关系,这里只是说我们先有一个大致的了解。
双冒号运算就是 java 中的 [方法引用]。 [方法引用] 格式为:类名::方法名。
注意是方法名哦,后面没有括号 “()” 的。为啥不要括号,因为这样的是式子并不代表一定会调用这个方法。这种式子一般是用作 Lambda 表达式,Lambda 有所谓懒加载嘛,不要括号就是说,看情况调用方法。
示例说明
比如说下面的表达式:
// 这里是说获取 person 类的 age 属性
person -> person.getAge();
// 按照方法引用的方式,可以改写为:
Person::getAge
这里就是调用的 Person 类中的 getAge 方法。接着,我们再来看一个简单的例子:
// 这里就是说创建一个 HashMap
() -> new HashMap<>();
// 可以替换为下面这样子
HashMap::new
使用示例
面这段代码,进行的操作是,把List里面的String全部大写并返还新的ArrayList,在前面的例子中我们是这么写的:
@Test
public void convertTest() {
List collected = new ArrayList<>();
collected.add("alpha");
collected.add("beta");
collected = collected.stream().map(string -> string.toUpperCase()).collect(Collectors.toList());
System.out.println(collected);
}
上面的代码比较简单,这里就不做特别多的解读了,我们直接看下怎么改写:
@Test
public void convertTest() {
List collected = new ArrayList<>();
collected.add("alpha");
collected.add("beta");
// 注意发生的变化
collected = collected.stream().map(String::toUpperCase).collect(Collectors.toCollection(ArrayList::new));
System.out.println(collected);
}
看到这里应该是可以知道这个具体是怎么回事了吧。
几种常见的用法
这里先小结下:
方法引用是Lambda表达式的一个非常有用的简化形式,它允许你直接引用现有方法而不必显式地创建Lambda表达式,方法引用使用双冒号(::)操作符。
双冒号操作符(::)在方法引用中有以下几种主要用法:
静态方法引用
语法格式:类名::静态方法名,当你需要引用一个静态方法时,你可以使用类名加上双冒号和静态方法名:
List list = Arrays.asList("Apple1", "Apple2", "Apple3");
list.forEach(System.out::println);
在上面的例子中,System.out::println 是对 System.out 类中的 println 静态方法的引用。
特定对象的实例方法引用
语法格式:特定对象::实例方法名,如果你有一个特定的对象,并且你想引用该对象的一个实例方法,你可以使用对象名加上双冒号和实例方法名
String str = "Hello";
Consumer consumer = str::toUpperCase;
consumer.accept(str); // 注意:这里的str::toUpperCase实际上并不会改变str的值
但通常,当你使用特定对象的实例方法作为方法引用时,更常见的是作为某种函数的参数(如 Function 或 Predicate ),而不是直接使用它(因为上面的例子实际上没有改变str的值)。
特定类型的任意对象的实例方法引用
语法格式:类名::实例方法名,如果你想要引用某个类型对象的实例方法,而不指定具体的对象,你可以使用类名加上双冒号和实例方法名。这通常与函数式接口的方法参数类型相匹配。
List list = Arrays.asList("apple", "banana", "cherry");
list.stream().map(String::toUpperCase).forEach(System.out::println);
List list = Arrays.asList("a1", "a2", "a3");
list.sort(String::compareToIgnoreCase);
在上面的例子中,
String::compareToIgnoreCase 是对 String 类中的 compareToIgnoreCase 实例方法的引用,它用于比较两个字符串(忽略大小写)。
构造器引用
语法格式:类名::new,也可以使用双冒号来引用构造器。这在你需要创建对象实例时特别有用
Supplier stringSupplier = String::new;
String newString = stringSupplier.get(); // 创建一个新的空字符串
Function stringFromInt = Integer::toString;
String stringFromIntResult = stringFromInt.apply(123); // 结果是 "123"
在上面的例子中,String::new 是对 String 类的默认构造器的引用,而 Integer::toString 是对 Integer 类的 toString 方法的引用,该方法接受一个 int 参数并返回一个 String。