浅谈对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


猜你喜欢
- Spring是什么?Spring是一个轻量级Java开发框架,最早有Rod Johnson创建,目的是为了解决企业级应用开发的业务逻辑层和其
- tomcat中文乱码问题这几天测试的兄弟发现了项目中存在乱码问题 经过排查发现是tomcat中的问题 于是在server.xml中添加了如下
- 建造者模式(Builder Pattern)使用多个简单的对象一步一步构建成一个复杂的对象。这种类型的设计模式属于创建型模式,它提供了一种创
- 目录一、环境搭建二、RetryTemplate2.1 RetryTemplate2.2 RetryListener2.3 回退策略2.3.1
- 1. 使用try-with-resources简化文件读取操作:修改前:FileInputStream fis = null;try { &
- 1.1 JDK 14详细概述JDK 8 已经在 2014年 3月 18日正式可用,JDK 8作为长期支持(Long-Term-Support
- 本文实例为大家分享了Android App自动更新通知栏下载的具体代码,供大家参考,具体内容如下版本更新说明这里有调用UpdateServi
- 1、使用pom安装依赖<dependency> <groupId>com.alibaba&
- Java 编程语言中时间的处理类有 Date类与 Calendar类。目前官方不推荐使用 Date类,因为其不利于国际化;而是推荐使用 Ca
- 目录简介为什么使用protobuf定义.proto文件编译协议文件详解生成的文件Builders 和 Messages序列化和反序列化协议扩
- 快速排序快速排序是一种比较高效的排序算法,采用“分而治之”的思想,通过多次比较和交换来实现排序,在一
- 简介Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程。该框架使用了特
- Kotlin使用SQLite首先确定我们的目标,SQLite只是一种工具,我们需要掌握就是增删改查就可以,我们真正需要动脑的还是项目中的业务
- 在intellij中忽略提交文件,分两种情况,文件没有纳入版本管理第一种方法文件还没有纳入版本管理,这种通过 svn的ignore配置ver
- 1 Android SDK自带的org.json解析解析原理: 基于文档驱动,需要把全部文件读入到内存中,然后遍历所有数据,根据需要检索想要
- 什么事读写分离读写分离,基本的原理是让主数据库处理事务性增、改、删操作(INSERT、UPDATE、DELETE),而从数据库处理SELEC
- using System;using System.Collections.Generic;namespace Demo{ &nb
- 我在Eclipse/MyEclipse环境下都测试过了,都好使。需要2个组件,分别是: ext-4.0.2a.jsb2 spke
- 前言最近对 base-spring-boot 项目进行了升级。在将其用于应用开发中时遇到java.lang.ArrayStoreE
- 对于现在的 App 来说,布局页面基本都会用到沉浸式状态栏,单纯的沉浸式状态栏很容易解决,但是在华为手机上存在一个底部虚拟按键的问题,会导致