本文为个人理解,不保证完全正确。 官方文档中将双冒号的用法分为了以下4类: 以下是我的理解 在使用双冒号前我们要先搞清楚一个问题:为什么要使用双冒号?也就是双冒号的作用是什么。 使用双冒号可以省略第一种Lambda表达式中的参数部分,即 使用双冒号有两个条件: 条件1为必要条件,必须要满足这个条件才能使用双冒号。 由于双冒号是为了省略 先定义一个方法,这个方法的作用是将一个集合的内容复制到另一个集合 调用这个方法 稍微解释一下: 在 第一个类: 第二个类 测试代码 除了上述两种情况可以使用双冒号化简Lambda表达式外,还存在一种特殊情况也可以使用双冒号。
官方文档中将双冒号的用法分为4类,按照我的个人理解可以分成2类来使用。官方文档
用法
举例
引用静态方法
ContainingClass::staticMethodName
引用特定对象的实例方法
containingObject::instanceMethodName
引用特定类型的任意对象的实例方法
ContainingType::methodName
引用构造函数
ClassName::new
个人理解
双冒号的作用
双冒号的设计初衷是为了化简Lambda表达式,不熟悉Lambda表达式的同学可以先了解一下。
Lambda表达式的形式有两种:
list.forEach(item -> System.out.println(item));
list.forEach(item -> { int numA = item.getNumA(); int numB = item.getNumB(); System.out.println(numA + numB); });
item ->
和调用方法的参数这两部分。
例如://不使用双冒号 list.forEach(item -> System.out.println(item)); //使用双冒号 list.forEach(System.out::println);
双冒号的使用条件
条件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表达式的参数有两个(形如(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);
本网页所有视频内容由 imoviebox边看边下-网页视频下载, iurlBox网页地址收藏管理器 下载并得到。
ImovieBox网页视频下载器 下载地址: ImovieBox网页视频下载器-最新版本下载
本文章由: imapbox邮箱云存储,邮箱网盘,ImageBox 图片批量下载器,网页图片批量下载专家,网页图片批量下载器,获取到文章图片,imoviebox网页视频批量下载器,下载视频内容,为您提供.
阅读和此文章类似的: 全球云计算