(原)

方法引用:

方法引用有4种:

1、静态方法引用:类名::静态方法名

在java中,对集合的排序,我们常用java提供的

Collections.sort(List<T> list, Comparator<? super T> c)

这个方法,

而在java8中,排序方法被作为了一个默认方法加入了List集合中。

default void sort(Comparator<? super E> c) {
Object[] a = this.toArray();
Arrays.sort(a, (Comparator) c);
ListIterator<E> i = this.listIterator();
for (Object e : a) {
i.next();
i.set((E) e);
}
}

这个与Collections.sort(list,comparator)类似,需要传入一个Comparator比较器。(Comparator在java8中已经是一个函数式接口了。)

如果我有这样一个model类:

class Person7{
private String name;
private Integer age; public Person7(String name, Integer age) {
super();
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
} public static int sortByAge(Person7 p1,Person7 p2){
return p1.getAge() - p2.getAge();
} public static int sortByName(Person7 p1,Person7 p2){
return p1.getName().compareToIgnoreCase(p2.getName());
} public int objectSortByAge(Person7 p){
return this.age - p.getAge();
} }

然后对这个model对象的age,name排序,

先创建一个Person7的List集合:

Person7 p1 = new Person7("zhaoda", 25);
Person7 p2 = new Person7("wanger",19);
Person7 p3 = new Person7("zhangsan",46);
Person7 p4 = new Person7("lisi",23); List<Person7> list = Arrays.asList(p1,p2,p3,p4);

用常用的lambda表达式进行排序,然后输出:

list.sort((a , b) -> Person7.sortByAge(a, b));
list.forEach(a -> System.out.println(a.getAge()));

如果用静态方法引用,那么就应该这么写:

list.sort(Person7::sortByAge);

sortByAge方法接收二个同类型参数,并且返回一个int类型,这种结构完全符合Comparator中的唯一抽象方法:int compare(T o1, T o2);

所以这里的Person7::sortByAge(a , b) -> Person7.sortByAge(a, b)是等价的。

(注意:方法引用是不能传任何参数的)

2、对象名::实例方法名

接上面的例子,现在将上面排序的方法拿出来,单独放在SortUtil7类中。

class SortUtil7{
public int sortByAge(Person7 p1,Person7 p2){
return p1.getAge() - p2.getAge();
} public int sortByName(Person7 p1,Person7 p2){
return p1.getName().compareToIgnoreCase(p2.getName());
}
}

我们先创建一个SortUtil7对象,然后用lambda表达式进行排序并输出:

SortUtil7 sort = new SortUtil7();
list.sort((a,b) -> sort.sortByAge(a, b));
list.forEach(a -> System.out.println(a.getAge()));

然后我们换对象方法引用去做同样的事:

list.sort(sort::sortByAge);
list.forEach(a -> System.out.println(a.getAge()));

再看看这段代码:

list.forEach(a -> System.out.println(a.getAge()));

这里可以替换成:

list.forEach(System.out::println);

这里的System.out是一个静态的对象

public final static PrintStream out = null;

这个out对象的println方法是这样的:

public void println(Object x) {
String s = String.valueOf(x);
synchronized (this) {
print(s);
newLine();
}
}

接收一个Object类型的参数没有返回值,与forEach方法的参数Consume中的抽象方法void accept(T t)完全一致,所以这里

list.forEach(System.out::println);

这种方法引用属于第里的第二种,对象名::实例方法。

3、类名::实例方法名

这种方法引用是4种方法引用中最难理解的一种,这种写法与第一种写没有必然的联系。

接上面的例子,在Person7中有一个实例方法,

public int objectSortByAge(Person7 p){
return this.age - p.getAge();
}

这个方法让一个实例与传进参数的那个实例进行比较。

然后用方法去调用并输出:

list.sort(Person7::objectSortByAge);
list.forEach(a -> System.out.println(a.getAge()));

这种方法引用之所以难以理解,是因为实例方法被调用了,却在代码中看到不是哪个实例调用的。但是要知道的是实例方法被调用了,一定是有一个实例对象,那么这个实例对象是哪个呢?

让我们回过头来看看sort方法,这个方法对应的抽象接口是int compare(T o1, T o2),也就是说Person7::objectSortByAge是与int compare(T o1, T o2)对应起来的,这里的实例对象就是T o1,可以这么理解:o1.objectSortByAge(o2),如果有多个参数,那么方法引用的实例就是第一个参数,后面的参数会按顺序当做实例方法的参数传进来。

4、构造方法引用:类名::new

这种方法引用比较好理解了。

接上面的例子,在Person7中创建一个方法:

public String getString(Supplier<String> supplier){
return supplier.get() + " hello";
}

然后创建一个Person7对象,用lambda的方法这个方法调用一下。

Test7 t = new Test7();
System.out.println(t.getString(() -> "123"));

这里换成构造方法引用:

System.out.println(t.getString(String::new));

因为这里的Supplier<String>不接收参数,返回一个String类型,所以可以用String::new的方法,只是这里方法引用传不了参数。

如果我们想用构造方法引用的写法,又想把参数写进去,我们这可以将getString方法改成这样:

public String getString2(String p, Function<String, String> f){
return f.apply(p) + " hello";
}

然后通过:

System.out.println(t.getString2("hehe", String::new));

这样的方法调用,这样就能将参数传入进来。

注意这里的getString2中的String::new不是调用String的new String()这个构造方法,而且调用的new String(String original)这个构造方法,因为Function中的抽象方法R apply(T t)是接收一个参数,返回一个参数。

继承:

由于java8中,抽象类新加入了default关键字,可以使得抽象方法中有了方法体。

如果有二个方法,都用了default方法,并且它们的方法名也一样,有一个类又同时实现了这二个接口,会出现什么情况呢?

比如这样:

interface MyInter1{
default void sayHello(){
System.out.println("hello");
}
} interface MyInter2{
default void sayHello(){
System.out.println("你好");
}
}

然后我的Person7同时实现了这二个接口

public class Test7 implements MyInter1, MyInter2

然后编译器此时就会报错了,说Test7 继承了一个无法关联的default sayHello()

此时我们必需重载此方法

@Override
public void sayHello(){ }

如果我们想用MyInter2中的sayHello(),

则我们可以这么写:

@Override
public void sayHello(){
MyInter2.super.sayHello();
}

这里   接口.super 可以找到特定的接口,然后再调用它的方法。用这种法就可以实现一个类实现多个接口,而这些接口中又有同相同的default方法名时,可以选取任意一个指定的接口中的default方法进行调用。

还有一种情况,类A实现了MyInter1接口,B继承了A同时又实现了MyInter2接口,这时你会发现,编译器通过了,没有报错,并且在调用B中的sayHello方法时,实际上是调用的类A中的sayHello方法,通过这里可以看出,其实这里存在一个优先级关系,即 类的优先级高于接口。

例子请看这里:https://github.com/LeeScofield/java8

Java 8 新特性7-方法引用、继承的更多相关文章

  1. Java 8新特性-4 方法引用

    对于引用来说我们一般都是用在对象,而对象引用的特点是:不同的引用对象可以操作同一块内容! Java 8的方法引用定义了四种格式: 引用静态方法     ClassName :: staticMetho ...

  2. 乐字节-Java8新特性之方法引用

    上一篇小乐介绍了<Java8新特性-函数式接口>,大家可以点击回顾.这篇文章将接着介绍Java8新特性之方法引用. Java8 中引入方法引用新特性,用于简化应用对象方法的调用, 方法引用 ...

  3. JDK8新特性04 方法引用与构造器引用

    import java.io.PrintStream; import java.util.Comparator; import java.util.function.*; /** * 一.方法引用 * ...

  4. Java(43)JDK新特性之方法引用

    作者:季沐测试笔记 原文地址:https://www.cnblogs.com/testero/p/15228461.html 博客主页:https://www.cnblogs.com/testero ...

  5. 2020你还不会Java8新特性?方法引用详解及Stream 流介绍和操作方式详解(三)

    方法引用详解 方法引用: method reference 方法引用实际上是Lambda表达式的一种语法糖 我们可以将方法引用看作是一个「函数指针」,function pointer 方法引用共分为4 ...

  6. Java8新特性之方法引用&Stream流

    Java8新特性 方法引用 前言 什么是函数式接口 只包含一个抽象方法的接口,称为函数式接口. 可以通过 Lambda 表达式来创建该接口的对象.(若 Lambda 表达式抛出一个受检异常(即:非运行 ...

  7. jdk1.8新特性之方法引用

    方法引用其实就是方法调用,符号是两个冒号::来表示,左边是对象或类,右边是方法.它其实就是lambda表达式的进一步简化.如果不使用lambda表达式,那么也就没必要用方法引用了.啥是lambda,参 ...

  8. JDK8新特性之方法引用

    什么是方法引用 方法引用是只需要使用方法的名字,而具体调用交给函数式接口,需要和Lambda表达式配合使用. 如: List<String> list = Arrays.asList(&q ...

  9. Java8新特性之方法引用

    <Java 8 实战>学习笔记系列 定义 方法引用让你可以重复使用现有的方法定义,并像Lambda一样传递它,可以把方法引用看作针对仅仅涉及单一方法的Lambda的语法糖,使用它将减少自己 ...

随机推荐

  1. IntelliJ IDEA中创建Web聚合项目(Maven多模块项目)

    Eclipse用多了,IntelliJ中创建Maven聚合项目可能有小伙伴还不太熟悉,我们今天就来看看. IntelliJ中创建普通的Java聚合项目相对来说比较容易,不会涉及到web操作,涉及到we ...

  2. 前端笔记之jQuery(上)加载函数的区别&对象&操作HTML/CSS&动画&选择器

    一.jQuery简介 1.0 JavaScript编程比较恶心的地方 恶心1:选择元素麻烦,全线兼容的方法只有getElementById()和getElementsByTagName()两个.其他的 ...

  3. centos7下vim8.1的编译安装教程

    之前安装YouCompleteMe的时候遇到vim版本不兼容的问题,看网上说是需要将vim版本提升到8.0及以上,然后就开始安装最新版本的vim,安装过程中的遇到了不少问题主要集中在配置方面和缺少插件 ...

  4. HBase学习-HBase原理

    1.系统架构 1.1 图解   从HBase的架构图上可以看出,HBase中的组件包括Client.Zookeeper.HMaster.HRegionServer.HRegion.Store.MemS ...

  5. Spring Boot(十三)RabbitMQ安装与集成

    一.前言 RabbitMQ是一个开源的消息代理软件(面向消息的中间件),它的核心作用就是创建消息队列,异步接收和发送消息,MQ的全程是:Message Queue中文的意思是消息队列. 1.1 使用场 ...

  6. win10连接无线网,开启移动热点,手机连接它手机一直显示获取ip地址中。

    *必须要有无线网卡才能设置WIFI首先打开电脑,选中“计算机”或者“我的电脑”,右击进入“管理”选项“.打开“计算机管理”窗口之后,在左栏菜单选项中找到“服务和应用程序”下的“服务”选项,如图点击进入 ...

  7. [争什么! 掺在一起做撒尿牛丸啊! 笨蛋]ASP.NET Core 2.0 + EF6 + Linux +MySql混搭

    好消息!特好消息!同时使用ASP.NET Core 2.0和.NET Framework类库还能运行在linux上的方法来啦! 是的,你没有看错!ASP.NET Core 2.0,.NET Frame ...

  8. 史上最全python面试题详解(四)(附带详细答案(关注、持续更新))

    python高级进阶-网络编程和并发(?道题详解) 1.简述 OSI 七层协议. OSI是Open System Interconnection的缩写,意为开放式系统互联. OSI七层协议模型主要是: ...

  9. Web 性能优化: 图片优化让网站大小减少 62%

    摘要: 压缩各种格式的图片. 原文:Web 性能优化: 图片优化让网站大小减少 62% 作者:前端小智 Fundebug经授权转载,版权归原作者所有. 这是 Web 性能优化的第二篇,上一篇在下面看点 ...

  10. HTML和CSS在IE7中常见的兼容性问题

    IE7及以下版本都会有这些问题 1.IE7块转内联块问题 问题描述:好像块转内联块失败,依然不在一行排列 解决办法:给元素添加如下css    *display:inline;*zoom:1; *di ...