浅谈对Java双冒号::的理解

作者:步步咏凉天 时间:2023-09-20 02:19:21 

本文为个人理解,不保证完全正确。
官方文档中将双冒号的用法分为4类,按照我的个人理解可以分成2类来使用。

官方文档

官方文档中将双冒号的用法分为了以下4类:

用法举例
引用静态方法ContainingClass::staticMethodName
引用特定对象的实例方法containingObject::instanceMethodName
引用特定类型的任意对象的实例方法ContainingType::methodName
引用构造函数ClassName::new

以下是我的理解

个人理解

双冒号的作用

在使用双冒号前我们要先搞清楚一个问题:为什么要使用双冒号?也就是双冒号的作用是什么。
双冒号的设计初衷是为了化简Lambda表达式,不熟悉Lambda表达式的同学可以先了解一下。
Lambda表达式的形式有两种:

包含单独表达式 :parameters -> an expression


list.forEach(item -> System.out.println(item));

包含代码块:parameters -> { expressions }


list.forEach(item -> {
int numA = item.getNumA();
int numB = item.getNumB();
System.out.println(numA + numB);
});

使用双冒号可以省略第一种Lambda表达式中的参数部分,即item ->和调用方法的参数这两部分。

例如:


//不使用双冒号
list.forEach(item -> System.out.println(item));
//使用双冒号
list.forEach(System.out::println);

双冒号的使用条件

使用双冒号有两个条件:

条件1
条件1为必要条件,必须要满足这个条件才能使用双冒号。
Lambda表达式内部只有一条表达式(第一种Lambda表达式),并且这个表达式只是调用已经存在的方法,不做其他的操作。

条件2
由于双冒号是为了省略item ->这一部分,所以条件2是需要满足不需要写参数item也知道如何使用item的情况。
有两种情况可以满足这个要求,这就是我将双冒号的使用分为2类的依据。

情况举例
Lambda表达式的参数与调用函数的参数完全一致list.forEach(item -> System.out.println(item))
调用的函数是参数item对象的方法且没有参数list.stream().map(item -> item.getId())

一些栗子

Lambda表达式的参数与调用函数的参数完全一致时

静态方法调用


//化简前
list.forEach(item -> System.out.println(item));
//化简后
list.forEach(System.out::println);

非静态方法调用


StringBuilder stringBuilder = new StringBuilder();
//化简前
IntStream.range(1, 101).forEach(item -> stringBuilder.append(item));
//化简后
IntStream.range(1, 101).forEach(stringBuilder::append);

调用构造方法

官方给出的例子

先定义一个方法,这个方法的作用是将一个集合的内容复制到另一个集合


public <T, SOURCE extends Collection<T>, DEST extends Collection<T>>
DEST transferElements(SOURCE sourceCollection, Supplier<DEST> collectionFactory) {
 DEST result = collectionFactory.get();
 result.addAll(sourceCollection);
 return result;
}

调用这个方法


//化简前
Set<Person> rosterSetLambda = transferElements(roster, () -> new HashSet<>());
//化简后
Set<Person> rosterSet = transferElements(roster, HashSet::new);

稍微解释一下:

调用时传入的Lambda表达式相当于是对Supplier的继承,并重写Supplier的get()方法,下面是Supplier的源码:


@FunctionalInterface
public interface Supplier<T> {

/**
  * Gets a result.
  *
  * @return a result
  */
 T get();
}

在transferElements()方法中调用collectionFactory.get()时相当于调用重写后的方法{return new HashSet<>();}

我自己写的一个例子

第一个类:


@Data
public class ModelA {
 private String id;

public ModelA(String id) {
   this.id = id;
 }

public ModelA() {
 }
}

第二个类


class ClassB {
 private final List<ModelA> list = new ArrayList<>();

public void add(String string, Function<String, ModelA> function) {
   list.add(function.apply(string));
 }
}

测试代码


ClassB classB = new ClassB();d
//化简前
classB.add("ddd", item -> new ModelA(item));
//化简后
classB.add("ddd", ModelA::new);

调用的函数是参数item对象的方法且没有参数时


//化简前
List<String> stringList = list.stream().map(item -> item.getId()).collect(Collectors.toList());
//化简后
List<String> stringList = list.stream().map(ModelA::getId).collect(Collectors.toList());

一种特殊情况

除了上述两种情况可以使用双冒号化简Lambda表达式外,还存在一种特殊情况也可以使用双冒号。
当Lambda表达式的参数有两个(形如(a,b) -> an expression)时,调用a的方法参数为b时,例如:


String[] stringArray = {"Barbara", "James", "Mary", "John"};
//化简前
Arrays.sort(stringArray, (a,b) -> a.compareToIgnoreCase(b));
//化简后
Arrays.sort(stringArray, String::compareToIgnoreCase);

来源:https://blog.csdn.net/u012784162/article/details/106538516

标签:Java,双冒号,::
0
投稿

猜你喜欢

  • Android Gradle依赖管理、去除重复依赖、忽略的方式

    2023-06-08 16:07:12
  • stream中使用peek一些陷阱避免及解决方法

    2021-09-02 20:58:21
  • 详解Java8 CompletableFuture的并行处理用法

    2023-07-27 11:48:06
  • Android自定义view仿淘宝快递物流信息时间轴

    2022-02-20 18:51:16
  • Android中Parcelable的作用实例解析

    2022-08-07 09:19:32
  • Java Scanner输入两个数组的方法

    2022-06-07 08:21:20
  • 详解Mybatis Generator的具体使用教程

    2022-01-16 22:39:13
  • Javaweb监听器实例之统计在线人数

    2023-02-08 11:47:23
  • Java实现五子棋的基础方法

    2021-07-11 12:32:08
  • C#遍历文件夹后上传文件夹中所有文件错误案例分析

    2022-11-03 09:28:27
  • Android 捕获运行时异常详解

    2023-12-22 21:16:30
  • java源码解析之String类的compareTo(String otherString)方法

    2023-11-11 23:10:00
  • Android开发之拖动条和评分组件用法分析

    2022-01-15 02:55:18
  • SpringBoot雪花算法主键ID传到前端后精度丢失问题的解决

    2022-07-18 02:30:47
  • 详解Java线程同步器CountDownLatch

    2023-08-23 18:42:39
  • 安卓输入框被虚拟键盘挡住的问题(微信开发)

    2023-12-01 15:42:43
  • 解析Android截取手机屏幕两种实现方案

    2023-12-16 21:28:06
  • 简单的观察者模式示例分享

    2023-02-11 12:52:12
  • spring cloud将spring boot服务注册到Eureka Server上的方法

    2023-12-08 19:42:09
  • 浅析Java内存模型与垃圾回收

    2023-11-23 06:11:58
  • asp之家 软件编程 m.aspxhome.com