- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章深入理解Java8双冒号::的使用由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
1、方法引用 。
java8允许我们使用lambda表达式创建匿名方法。但有时lambda表达式除了调用现有方法之外什么也不做。在这些情况下,通过名称引用现有的方法,通常能更直白的表现出方法的调用过程。对于已经存在的且具有方法名称的方法,它其实是简洁且易于读取的一种lambda表达式,或者说是对lambda表达式的一种进一步简化.
现在我们来看看下面这个“person”类:
public class Person { public enum Sex { MALE, FEMALE } String name; LocalDate birthday; Sex gender; String emailAddress; public int getAge() { // ... } public Calendar getBirthday() { return birthday; } public static int compareByAge(Person a, Person b) { return a.birthday.compareTo(b.birthday); }}
假设你的社交网络应用程序的成员包含在一个数组中,并且你希望按年龄对数组进行排序。你可以使用以下代码:
Person[] rosterAsArray = roster.toArray(new Person[roster.size()]);class PersonAgeComparator implements Comparator<Person> { public int compare(Person a, Person b) { return a.getBirthday().compareTo(b.getBirthday()); }} Arrays.sort(rosterAsArray, new PersonAgeComparator());
上面最后一行代码:中的sort方法源码如下:
static <T> void sort(T[] a, Comparator<? super T> c)
注意,接口comparator是一个功能接口。因此,你可以使用lambda表达式,而不是定义并创建实现comparator的类的新实例,所以上上面的代码可以用lambda表达式改写成下面这样:
Arrays.sort(rosterAsArray, (Person a, Person b) -> { return a.getBirthday().compareTo(b.getBirthday()); });
但是,我们已经在Person bean对象中提前写好了用来比较两个person对象的出生日期的方法:,所以你其实可以在lambda表达式的主体中直接调用这个方法方法:
Arrays.sort(rosterAsArray, (Person a, Person b) -> { return a.getBirthday().compareTo(b.getBirthday()); });
因为这个上面这个lambda表达式是在调用现有的方法,所以我们这里就可以使用上面提到的使用方法引用方式(及双冒号 ::),而不是之前我们熟悉的lambda表达式:
Arrays.sort(rosterAsArray, Person::compareByAge);
方法引用 person::comparebyage 在语义上与lambda表达式(a,b)->person.comparebyage(a,b)相同。他们都有以下特点:
(Person, Person)
。2、方法引用的种类(哪些场景可以使用方法引用) 。
有四种方法引用:
。
种类 。 |
案例 |
---|---|
引用静态方法 | ContainingClass::staticMethodName |
对特定对象的实例方法的引用 | containingObject::instanceMethodName |
对特定类型的任意对象的实例方法的引用 | ContainingType::methodName |
对构造函数的引用 | ClassName::new |
。
1、引用静态方法 。
例如:
Person::comparebyage 是对Person类的静态方法 comparebyage 的引用.
2、引用特定对象的实例方法 。
以下是对特定对象的实例方法的引用示例:
class ComparisonProvider { public int compareByName(Person a, Person b) { return a.getName().compareTo(b.getName()); } public int compareByAge(Person a, Person b) { return a.getBirthday().compareTo(b.getBirthday()); }}ComparisonProvider myComparisonProvider = new ComparisonProvider();Arrays.sort(rosterAsArray, myComparisonProvider::compareByName);
方法引用 myComparisonProvider::compareByName 调用对象myComparisonProvider的一部分方法 compareByName。然后JRE会自动推断方法类型参数,在这种情况下为(Person, Person).
3、引用特定类型的任意对象的实例方法 。
以下是对特定类型的任意对象的实例方法的引用示例:
String[] stringArray = { "Barbara", "James", "Mary", "John", "Patricia", "Robert", "Michael", "Linda" };Arrays.sort(stringArray, String::compareToIgnoreCase);
方法引用 string::compareTogignoreCase 的等效lambda表达式会有形参列表(String a,String b),其中a和b是用于更好地描述此示例的任意名称。这次的方法引用,将调用方法a.CompareTognoreCase(b).
这里官网原文可能说的不是很清楚,也有点拗口,多说一句:
如何理解:对特定类型的 任意对象的 实例方法的引用 这句话呢?
实际上上面代码没有在特定实例(也就是没有new一个指定的String 实例出来,像这样:String s = new String() ,s就是一个指定的String对象的实例)上引用方法 compareTogignoreCase ,而是在String类自身引用的.
我们进入Arrays.sort(xx,xx)方法源码,我们可以看到这个数组排序方法底层算法涉及循环,那么这行代码我们可以想象它会在每循环到数组的一个String元素时,以这个String元素所属类型对象(其实就是String对象)的身份去调用compareTogignoreCase方法,即:
会调用的compareTogignoreCase --》String Barbara = new String(); Barbara.compareTogignoreCase 。
会调用的compareTogignoreCase --》String James= new String();James.compareTogignoreCase 。
以此类推。。.
我可能解释的也不太好,主要就是想语言上理解一下:对特定类型的 任意对象的 实例方法 的引用 。
4、引用构造函数 。
你可以通过使用name new 的方式引用构造函数,这与静态方法的引用方式类似.
下面这个示例方法,是将元素从一个集合复制到另一个集合。我们将以这个方法为例,讲解如何使用方法引用的方式引用构造函数.
public static <T, SOURCE extends Collection<T>, DEST extends Collection<T>> DEST transferElements(SOURCE sourceCollection,Supplier<DEST> collectionFactory) { DEST result = collectionFactory.get(); for (T t : sourceCollection) { result.add(t); } return result;}
函数接口supplier包含一个无参但返回一个对象的get方法,源码可见,他返回的类型就是泛型的类型,如下:
因此,可以使用lambda表达式调用方法transferElements,如下所示:
Set<Person> rosterSetLambda = transferElements(roster, () -> { return new HashSet<>(); });
可以使用构造函数引用代替lambda表达式,如下所示:
Set<Person> rosterSet = transferElements(roster, HashSet::new);
上面代码,Java编译器执行的时候会自动推断出这里想要创建一个包含Person类型元素的hashset集合。当然你也可以显示的指定类型,如下:
Set<Person> rosterSet = transferElements(roster, HashSet<Person>::new);
到此这篇关于深入理解Java8双冒号::的使用的文章就介绍到这了,更多相关Java8双冒号::内容请搜索我以前的文章或继续浏览下面的相关文章希望大家以后多多支持我! 。
原文链接:https://blog.csdn.net/aigoV/article/details/102586953 。
最后此篇关于深入理解Java8双冒号::的使用的文章就讲到这里了,如果你想了解更多关于深入理解Java8双冒号::的使用的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
最近做一个项目,由于是在别人框架里开发app,导致了很多限制,其中一个就是不能直接引用webservice 。 我们都知道,调用webserivice 最简单的方法就是在 "引用"
这是SDL2代码的一部分 SDL主函数 int main(int argc,char *argv[]) { ... ... bool quit=false; S
c 中的函数: PHPAPI char *php_pcre_replace(char *regex, int regex_len, ch
我有以下映射: public class SecurityMap : ClassMap { public SecurityMap() {
我在vue-lic3中使用了SCSS,但是有一个奇怪的错误,使用/ deep /会报告错误,我不想看到它。 代码运行环境 vue-cli3 + vant + scss 的CSS /deep/ .van
我在深入阅读 C# 时遇到了这个我能理解的内容: 当它被限制为引用类型时,执行的比较类型完全取决于类型参数被限制为什么。 但是不能理解这个: 如果进一步限制派生自重载 == 和 != 运算符的特定类型
Closed. This question is opinion-based。它当前不接受答案。 想改善这个问题吗?更新问题,以便editing this post用事实和引用来回答。 3年前关闭。
有人可以详细介绍关于自赋值的运算符重载中的 *this 和 const 例如: Class& Class::operator=(const Class& other) { a = other.
在向树中插入新节点时,如何填充闭包表的深度/长度列? ancestor 和 descendant 中的值是来自另一个表的 ID,表示要以树结构排列的页面。 关闭表: ancestor desce
现在我正在阅读“深入了解 C#”。缺少的一件事是完成一章后我可以解决的一系列问题。那会帮助我理解我刚刚学到的概念。 哪里可以找到适合 C#3.0 的问题集? 谢谢 最佳答案 你可以试试LINQ 101
TypeScript 给 JavaScript 扩展了类型的语法,我们可以给变量加上类型,在编译期间会做类型检查,配合编辑器还能做更准确的智能提示。此外,TypeScript 还支持了高级类型用
是否有一个单行代码来获取生成器并生成该生成器中的所有元素?例如: def Yearly(year): yield YEARLY_HEADER for month in range(1, 13)
所以我阅读了一些与“什么是方法组”相关的 StackOverflow 问题以及其他互联网文章,它们在底线都说了同样的话——方法组是“一组重载方法” ". 但是,在阅读 Jon Skeet 的“C# 深
有什么方法可以从子组件中获取子组件吗? 想象一下以下组件树: 应用程序 问题 问题选项(包含复选框) 问题选项(包含复选框) 问题选项(包含复选框) 我想从 App 访问问题选项以选中所有复选框。 参
class_eval 和 instance_eval 在定义方法等情况下是完全可以预测的。我也理解类的实例和类的单例(又名特征类)之间的区别。 但是 我无法弄清楚以下唯一的事情:比方说,出于某些策略目
我想出了如何将符号 rwx 部分读取/转换为 421 个八进制部分,这非常简单。但是当涉及到特殊字符时,我感到非常困惑。我们知道 -r-xr---wx 转换为 0543,但 -r-sr---wt 或
我怀疑我系统的 Java 版本有问题。某些应用程序出现段错误或内存不足或存在链接错误。如果我从源代码安装了 JDK,我会做类似“make test”的事情,看看哪些测试失败了。但是,看起来从源代码构建
如何克隆一个 repo(使用 libgit2 ) 我想做什么git clone确实,但有 libgit2 .我可能要问的是什么 git clone确实很深入。 这是我目前正在做的: 初始化一个repo
00、头痛的JS闭包、词法作用域? 被JavaScript的闭包、上下文、嵌套函数、this搞得很头痛,这语言设计的,感觉比较混乱,先勉强理解总结一下😂😂😂.
我开始玩 lubridate R 中的包。我注意到 now(tzone="EST")计算为: [1] "2015-08-25 13:01:08 EST" 而 now(tzone="PST")导致警告:
我是一名优秀的程序员,十分优秀!