java中双冒号可以传参数吗_浅谈对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 , DEST extends Collection>

DEST transferElements(SOURCE sourceCollection, SuppliercollectionFactory) {

DEST result = collectionFactory.get();

result.addAll(sourceCollection);

return result;

}

调用这个方法

//化简前

SetrosterSetLambda = transferElements(roster, () -> new HashSet<>());

//化简后

SetrosterSet = transferElements(roster, HashSet::new);

稍微解释一下:

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

@FunctionalInterface

public interface Supplier{

/**

* 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 Listlist = new ArrayList<>();

public void add(String string, Functionfunction) {

list.add(function.apply(string));

}

}

测试代码

ClassB classB = new ClassB();d

//化简前

classB.add("ddd", item -> new ModelA(item));

//化简后

classB.add("ddd", ModelA::new);

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

//化简前

ListstringList = list.stream().map(item -> item.getId()).collect(Collectors.toList());

//化简后

ListstringList = 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双冒号::内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值