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

 更新时间:2020年06月18日 14:49:31   作者:步步咏凉天  
这篇文章主要介绍了浅谈对Java双冒号::的理解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

本文为个人理解,不保证完全正确。
官方文档中将双冒号的用法分为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);

到此这篇关于浅谈对Java双冒号::的理解的文章就介绍到这了,更多相关Java双冒号::内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 编译大型Java项目class冲突导致报错的解决方案

    编译大型Java项目class冲突导致报错的解决方案

    这篇文章给大家盘点编译大型项目class冲突导致报错的解决方案,文中通过代码示例介绍的非常详细,具有一定的参考价值,需要的朋友可以参考下
    2023-10-10
  • java不解压直接读取压缩包中文件的实现方法

    java不解压直接读取压缩包中文件的实现方法

    这篇文章主要介绍了java不解压直接读取压缩包中文件的实现方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-04-04
  • JDBC中PreparedStatement详解以及应用场景实例介绍

    JDBC中PreparedStatement详解以及应用场景实例介绍

    PreparedStatement对象代表的是一个预编译的SQL语句,用它提供的setter方法可以传入查询的变量,这篇文章主要给大家介绍了关于JDBC中PreparedStatement详解以及应用场景实例介绍的相关资料,需要的朋友可以参考下
    2024-02-02
  • Java实现用位运算维护状态码

    Java实现用位运算维护状态码

    位运算是一种非常高效的运算方式,在算法考察中比较常见,那么业务代码中我们如何使用位运算呢,感兴趣的小伙伴快跟随小编一起学习一下吧
    2024-03-03
  • springboot集成Swagger的方法(让你拥有属于自己的api管理器)

    springboot集成Swagger的方法(让你拥有属于自己的api管理器)

    在大型的项目中,如果你有非常多的接口需要统一管理,或者需要进行接口测试,那么我们通常会在繁杂地api中找到需要进行测试或者管理的接口,接下来通过本文给大家介绍springboot集成Swagger的方法让你拥有属于自己的api管理器,感兴趣的朋友一起看看吧
    2021-11-11
  • java用split分割字符串的一个有趣现象

    java用split分割字符串的一个有趣现象

    最近在项目中使用了java中的split分割字符串,发现了一个bug,充分了展示了自己对java底层的认知有很多的不足和欠缺。下面将这次的经过总结出来分享给大家,有需要的朋友们可以参考借鉴,下面来一起看看吧。
    2016-12-12
  • 简单易懂Java反射的setAccessible()方法

    简单易懂Java反射的setAccessible()方法

    本文主要介绍了简单易懂Java反射的setAccessible()方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • 简介Java的Hibernate框架中的Session和持久化类

    简介Java的Hibernate框架中的Session和持久化类

    这篇文章主要介绍了Java的Hibernate框架中的Session和持久化类,Hibernate是Java的SSH三大web开发框架之一,需要的朋友可以参考下
    2015-12-12
  • springboot优雅获取前端参数的方法详解

    springboot优雅获取前端参数的方法详解

    现在的项目基本上都是前后端分离的项目,如何打通前后端,接收前端传过来的参数呢,这篇文章小编就来和大家详细介绍一下springboot如何优雅的获取前端参数吧
    2024-03-03
  • Java中计算集合中元素的出现次数统计

    Java中计算集合中元素的出现次数统计

    本文主要介绍了Java中计算集合中元素的出现次数统计,使用Collections类配合HashMap来统计和java lamb 计算这两种方式,具有一定的参考价值,感兴趣可以了解一下
    2024-02-02

最新评论