专业的JAVA编程教程与资源

网站首页 > java教程 正文

Java8新特性之方法引用中的双冒号

temp10 2025-04-09 20:28:03 java教程 9 ℃ 0 评论

目录

简介

双冒号运算操作符是类方法的引用lambda 表达式的一种简写,这种简写的学名叫 eta-conversion 或者叫 η-conversion。

Java8新特性之方法引用中的双冒号

先看一个简单的例子初步了解下:

把 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

本文暂时没有评论,来添加一个吧(●'◡'●)

欢迎 发表评论:

最近发表
标签列表