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

相关文章

如何理解java基础中的Reference和引用类型?

首先要大致了解 Java 的几种引用类型。如下图所示,JDK 1.2 之后新增了 Reference 的概念,给开发人员提供了与 GC 交互的一种渠道。《深入理解 Java 虚拟机》中对于几种引用类型...

java的四种引用

java 中的引用分为 4种1.强引用 引用存在就不会被GC *2.软引用 heap memory(堆内存)满了就会被GC掉 *3.弱引用 每次GC就会回收掉(应用有:ThreadLocal) *4....

详细介绍一下Java中的什么是值传递?什么是引用传递?

Java中的参数传递是通过值传递进行的,即使是对象也是遵循这个规则,想要了解这个原理,首先我们就需要了解什么是值传递,什么是引用传递?值传递值传递是在调用方法的时候,方法接收到的参数是实际参数的一个副...

Java 值传递详解

开始之前,我们先来搞懂下面这两个概念:形参&实参值传递&引用传递形参&实参方法的定义可能会用到 参数(有参的方法),参数在程序语言中分为:实参(实际参数,Arguments):用于传递给函数/方法的参...

Java开发者必知的15个核心概念,第8个让你恍然大悟!

你是不是经常在面试中被问到Java的基础概念,却总是回答得不够全面?别担心,今天我们就来聊聊Java开发中那些必须掌握的核心知识点,帮你轻松应对各种技术面试! 1. ClassLoader:Java类...

Java8特性之方法引用

方法引用简介如果在Lambda表达式的具体逻辑处理和某个方法的处理逻辑相同,则可以直接将方法名称指代具体的处理逻辑,从而使得整个Lambda表达式更加的简洁,逻辑更加清晰。比如: System.out...