浅谈对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, supplier collectionFactory) { DEST result = collectionFactory.get(); result.addAll(sourceCollection); return result; }调用这个方法//化简前 Set rosterSetLambda = transferElements(roster, () -> new HashSet()); //化简后 Set rosterSet = 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 List list = new ArrayList(); public void add(String string, Function 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 stringList = list.stream().map(item -> item.getId()).collect(Collectors.toList()); //化简后 List 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中的String是不可变对象 在面向对象及函数编程语言中,不...
String, StringBuffer 和 StringBuilder 可变性 String不可变...
序列化:把对象转换为字节序列的过程称为对象的序列化. 反序...
先说结论,是对象!可以继续往下看 数组是不是对象 什么是对...
为什么浮点数 float 或 double 运算的时候会有精度丢失的风险...
面试题引入 这里引申出一个经典问题,看下面代码 Integer a ...