浅谈对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 = NumA();
int numB = NumB();
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 -> Id())
⼀些栗⼦
Lambda表达式的参数与调⽤函数的参数完全⼀致时
静态⽅法调⽤
//化简前
list.forEach(item -> System.out.println(item));
奇瑞5一8万车//化简后
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 = ();
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()⽅法中调⽤()时相当于调⽤重写后的⽅法{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 -> Id()).List());
//化简后
List<String> stringList = list.stream().map(ModelA::getId).List());
⼀种特殊情况
除了上述两种情况可以使⽤双冒号化简Lambda表达式外,还存在⼀种特殊情况也可以使⽤双冒号。
当Lambda表达式的参数有两个(形如(a,b) -> an expression)时,调⽤a的⽅法参数为b时,例如:
String[] stringArray = {"Barbara", "James", "Mary", "John"};
//化简前
Arrays.sort(stringArray, (a,b) -> a.compareToIgnoreCa(b));
//化简后
许嵩歌曲大全
Arrays.sort(stringArray, String::compareToIgnoreCa);
到此这篇关于浅谈对Java双冒号::的理解的⽂章就介绍到这了,更多相关Java双冒号::内容请搜索以前的⽂章或继续浏览下⾯的相关⽂章希望⼤家以后多多⽀持!