读书笔记——《Java 8 实战》系列之方法引用

本贴最后更新于 2707 天前,其中的信息可能已经时异事殊

上一篇博客中,我们继续介绍了一些关于 Lambda 表达式的相关知识。在本篇博客中,我将向大家介绍一种通过调用特定方法来使 Lambda 表达式更简捷的办法——方法引用。事实上,方法引用就是让你根据已有的方法实现来创建 Lambda 表达式。

简单来说,方法引用的语法就是:

目标引用::方法名称

需要注意的是这里的方法名称后面不加圆括号(),因为我们并没有实际调用这个方法

方法引用大概分为三类:

类型方法引用示例
指向静态方法的方法引用Class::staticMethod(如 Integer::parseInt)
指向任意类型实例方法的方法引用Class::instanceMethod(如 String::length)
指向现有对象的实例方法的方法引用object::instanceMethod(如 student::getHeight)

第二种方法引用与第三种方法引用乍一看会让人有些迷惑,实际上第二种方法引用主要是用在当我们 Lambda 表达式的主体中调用参数对象的某个方法时,如:

(String s) -> s.length() 可以改写成 String::length

而第三种方法引用主要时用在当我们在调用一个外部对象的方法时,如:

() -> student.getHeight() 可以改写成 student::getHeight

下图是为三种不同类型的 Lambda 表达式构建方法引用的办法

e4ebadd76a1247e5a12993f2fadbef18.png

接下来我就用一个例子——给学生利用不同策略来排序为大家总结一下,行为参数化Lambda 表达式(一)Lambda 表达式(二)方法引用这四篇博客的重要内容。

  • 首先是我们的 Student 实体类
class Student{ private String name; //学生姓名 private Integer avgScore; //平均成绩 private Integer height; //学生身高 public Student(String name, Integer avgScore, Integer height) { super(); this.name = name; this.avgScore = avgScore; this.height = height; } public String getName() { return name; } public void setName(String name) { this.name = name; } public Integer getAvgScore() { return avgScore; } public void setAvgScore(int avgScore) { this.avgScore = avgScore; } public Integer getHeight() { return height; } public void setHeight(int height) { this.height = height; } @Override public String toString() { return this.getName(); } }
  • Java 的 API 类库中已经提供了 List 的 sort 方法,我们不用去自己实现它,只需要传入一个 Comparator 类型的对象,list 就可以按照它的规则来排序
void sort(Comparator<? super E> c)
  • Comparator是一个函数式接口,它提供了一个比较两个对象大小的方法 compare(T o1, T o2),返回一个 int 类型的值,该返回值可能为负数,0,或者正数,分别对应着 o1 小于,等于,或者大于 o2。
//返回值可能为负数,0,或者正数,分别对应着o1小于,等于,或者大于o2 int compare(T o1, T o2);
  • 要想将一个 List 中的学生以某种规则排序,我们以行为参数化的角度来考虑,第一种解决方案应该如下:
//实现一个Comparator接口的实体类StudentHeightComparator,通过比较两个学生的身高,来比较两个student对象的大小 class StudentHeightComparator implements Comparator<Student>{ @Override public int compare(Student s1, Student s2) { // TODO Auto-generated method stub return s1.getHeight().compareTo(s2.getHeight()); } public static void main(String[] args) { // TODO Auto-generated method stub List<Student> students = new ArrayList<>(); students.add(new Student("a",90,180)); students.add(new Student("b",80,175)); students.add(new Student("c",70,190)); //实例化StudentHeightComparator,并将其作为参数传入到sort方法中去 students.sort(new StudentHeightComparator()); System.out.println(students.toString()); } }
  • 匿名类的机制可以一定程度上优化我们的第一种解决方案,它的解决方案应该如下:
public static void main(String[] args) { // TODO Auto-generated method stub List<Student> students = new ArrayList<>(); students.add(new Student("a",90,180)); students.add(new Student("b",80,175)); students.add(new Student("c",70,190)); //使用匿名类,我们就无需实现一个只需实例化一次的类StudentHeightComparator students.sort(new Comparator<Student>() { @Override public int compare(Student s1, Student s2) { // TODO Auto-generated method stub return s1.getHeight().compareTo(s2.getHeight()); } }); System.out.println(students.toString()); }
  • 尽管匿名类一定程度上减少了方案一中啰嗦的代码,但我们可以使用 Lambda 表达式来使方案二变得更加简单:
public static void main(String[] args) { // TODO Auto-generated method stub List<Student> students = new ArrayList<>(); students.add(new Student("a",90,180)); students.add(new Student("b",80,175)); students.add(new Student("c",70,190)); //Comparator实际上代表了(T,T) -> int 的函数描述符 students.sort((s1,s2) -> s1.getHeight().compareTo(s2.getHeight())); System.out.println(students.toString()); }
  • 现在我们来为实体类 Student 增加一个静态方法,来写出一个不一样的 Lambda 表达式:
class Student{ ...... public static int compareStudentByHeight(Student s1, Student s2) { return s1.getHeight().compareTo(s2.getHeight()); } } public static void main(String[] args) { List<Student> students = new ArrayList<>(); students.add(new Student("a",90,180)); students.add(new Student("b",80,175)); students.add(new Student("c",70,190)); //使用Student类的静态方法 students.sort((s1,s2) -> Student.compareStudentByHeight(s1,s2)); System.out.println(students.toString()); }
  • 最后,我们使用方法引用来将上面的代码变得更加简洁,使得代码阅读起来就像问题描述的一样:
public static void main(String[] args) { List<Student> students = new ArrayList<>(); students.add(new Student("a",90,180)); students.add(new Student("b",80,175)); students.add(new Student("c",70,190)); //使用Student类的静态方法的方法引用 students.sort(Student::compareStudentByHeight); System.out.println(students.toString()); }
  • 最后的这种解决方案呢,其实是用了我们上面提到的指向静态方法的方法引用,那么同学们能不能自己写出剩下的两种类型的方法引用呢?

指向任意类型实例方法的方法引用实例

class Student{ ...... //一个普通的实例方法 public int compareToStudentByHeight(Student s) { return this.getHeight().compareTo(s.getHeight()); } } public static void main(String[] args) { // TODO Auto-generated method stub List<Student> students = new ArrayList<>(); students.add(new Student("a",90,180)); students.add(new Student("b",80,175)); students.add(new Student("c",70,190)); //使用任意类型对象实例方法的Lambda表达式形式,等价于下一行代码 students.sort((s1,s2) -> s1.compareToStudentByHeight(s2)); //使用指向任意类型对象实例方法的方法引用,进一步简化了上一行代码 students.sort(Student::compareToStudentByHeight); System.out.println(students.toString()); }

指向现有对象的实例方法的方法引用

public static void main(String[] args) { // TODO Auto-generated method stub List<Student> students = new ArrayList<>(); students.add(new Student("a",90,180)); students.add(new Student("b",80,175)); students.add(new Student("c",70,190)); StudentComparatorProvider provider = new StudentComparatorProvider(); //使用指向现有对象的实例方法的Lambda表达式形式,等价于下一行代码 students.sort((s1,s2) -> provider.compareStudentByHeight(s1, s2)); //使用指向现有对象的实例方法的方法引用,进一步简化了上一行代码 students.sort(provider::compareStudentByHeight); System.out.println(students.toString()); }
  • 除此之外,对于一个现有的构造函数,你可以利用它的名称和关键字 new 来创建一个它的引用,首先我们先稍微修改下我们的 Student 实体类,在实体类中分别提供无参,一个参数和两个参数的构造函数。
class Student{ private String name; //学生姓名 private Integer avgScore; //平均成绩 public Student() { this.name = "defaultName"; this.avgScore = 0; } public Student(String name) { this.name = name; this.avgScore = 0; } public Student(String name, Integer avgScore) { this.name = name; this.avgScore = avgScore; } public String getName() { return name; } public void setName(String name) { this.name = name; } public Integer getAvgScore() { return avgScore; } public void setAvgScore(int avgScore) { this.avgScore = avgScore; } @Override public String toString() { return "姓名:"+this.getName()+" 平均分: "+this.getAvgScore(); } }
  • 假如使用 new Student()构造函数,它的函数签名是() -> Student, 符合 Supplier 的() -> T 的函数签名,因此我们可以用以下几种不同的方法创建一个 Student 实例:
public static void main(String[] args) { //使用无参构造函数创建实例 Student student1 = new Student(); System.out.println(student1.toString()); //使用Lambda表达式创造实例,等价于下面使用方法引用创建实例 Supplier<Student> s1 = () -> new Student(); Student student2 = s1.get(); System.out.println(student2.toString()); //使用方法引用进一步简化Lambda表达式的写法 Supplier<Student> s= Student::new; Student student3 = s.get(); System.out.println(student3.toString()); }
  • 假如使用 new Student(String name)构造函数,它的函数签名是(String) -> Student,符合 Function<T,R> 的(T) -> R 的函数签名,因此我们可以用以下几种不同的方法创建一个 Student 实例:
public static void main(String[] args) { //使用一个参数构造函数创建实例 Student student1 = new Student("Jesmin"); System.out.println(student1.toString()); //使用Lambda表达式创造实例,等价于下面使用方法引用创建实例 Function<String,Student> f = s -> new Student(s); Student student2 = f.apply("Jesmin"); System.out.println(student2.toString()); //使用方法引用进一步简化Lambda表达式的写法 Function<String,Student> f2= Student::new; Student student3 = f2.apply("Jesmin"); System.out.println(student3.toString()); }
  • 假如使用 new Student(String name, Integer avgScore)构造函数,他的函数签名是(String,Integer) -> Student,符合 BiFunction<T,U,R> 的 <T,U> -> R 的函数签名,因此我们可以用以下几种不同的方法创建一个 Student 实例:
public static void main(String[] args) { //使用两个参数构造函数创建实例 Student student1 = new Student("Jesmin",90); System.out.println(student1.toString()); //使用Lambda表达式创造实例,等价于下面使用方法引用创建实例 BiFunction<String,Integer,Student> bf = (s,i) -> new Student(s,i); Student student2 = bf.apply("Jesmin",90); System.out.println(student2.toString()); //使用方法引用进一步简化Lambda表达式的写法 BiFunction<String,Integer,Student> bf2= Student::new; Student student3 = bf2.apply("Jesmin",90); System.out.println(student3.toString()); }
  • B3log

    B3log 是一个开源组织,名字来源于“Bulletin Board Blog”缩写,目标是将独立博客与论坛结合,形成一种新的网络社区体验,详细请看 B3log 构思。目前 B3log 已经开源了多款产品:SymSoloVditor思源笔记

    1063 引用 • 3455 回帖 • 165 关注
  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3194 引用 • 8214 回帖
  • 方法引用
    3 引用

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • jsoup

    jsoup 是一款 Java 的 HTML 解析器,可直接解析某个 URL 地址、HTML 文本内容。它提供了一套非常省力的 API,可通过 DOM,CSS 以及类似于 jQuery 的操作方法来取出和操作数据。

    6 引用 • 1 回帖 • 488 关注
  • 职场

    找到自己的位置,萌新烦恼少。

    127 引用 • 1708 回帖
  • 生活

    生活是指人类生存过程中的各项活动的总和,范畴较广,一般指为幸福的意义而存在。生活实际上是对人生的一种诠释。生活包括人类在社会中与自己息息相关的日常活动和心理影射。

    230 引用 • 1454 回帖 • 1 关注
  • Ant-Design

    Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

    17 引用 • 23 回帖 • 1 关注
  • 设计模式

    设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结出来的。

    200 引用 • 120 回帖
  • BAE

    百度应用引擎(Baidu App Engine)提供了 PHP、Java、Python 的执行环境,以及云存储、消息服务、云数据库等全面的云服务。它可以让开发者实现自动地部署和管理应用,并且提供动态扩容和负载均衡的运行环境,让开发者不用考虑高成本的运维工作,只需专注于业务逻辑,大大降低了开发者学习和迁移的成本。

    19 引用 • 75 回帖 • 663 关注
  • Access
    1 引用 • 3 回帖 • 4 关注
  • Jenkins

    Jenkins 是一套开源的持续集成工具。它提供了非常丰富的插件,让构建、部署、自动化集成项目变得简单易用。

    54 引用 • 37 回帖
  • 30Seconds

    📙 前端知识精选集,包含 HTML、CSS、JavaScript、React、Node、安全等方面,每天仅需 30 秒。

    • 精选常见面试题,帮助您准备下一次面试
    • 精选常见交互,帮助您拥有简洁酷炫的站点
    • 精选有用的 React 片段,帮助你获取最佳实践
    • 精选常见代码集,帮助您提高打码效率
    • 整理前端界的最新资讯,邀您一同探索新世界
    488 引用 • 384 回帖 • 5 关注
  • 运维

    互联网运维工作,以服务为中心,以稳定、安全、高效为三个基本点,确保公司的互联网业务能够 7×24 小时为用户提供高质量的服务。

    151 引用 • 257 回帖 • 1 关注
  • Kubernetes

    Kubernetes 是 Google 开源的一个容器编排引擎,它支持自动化部署、大规模可伸缩、应用容器化管理。

    116 引用 • 54 回帖 • 5 关注
  • RYMCU

    RYMCU 致力于打造一个即严谨又活泼、专业又不失有趣,为数百万人服务的开源嵌入式知识学习交流平台。

    4 引用 • 6 回帖 • 53 关注
  • jsDelivr

    jsDelivr 是一个开源的 CDN 服务,可为 npm 包、GitHub 仓库提供免费、快速并且可靠的全球 CDN 加速服务。

    5 引用 • 31 回帖 • 93 关注
  • 国际化

    i18n(其来源是英文单词 internationalization 的首末字符 i 和 n,18 为中间的字符数)是“国际化”的简称。对程序来说,国际化是指在不修改代码的情况下,能根据不同语言及地区显示相应的界面。

    8 引用 • 26 回帖 • 1 关注
  • 新人

    让我们欢迎这对新人。哦,不好意思说错了,让我们欢迎这位新人!
    新手上路,请谨慎驾驶!

    52 引用 • 228 回帖
  • 宕机

    宕机,多指一些网站、游戏、网络应用等服务器一种区别于正常运行的状态,也叫“Down 机”、“当机”或“死机”。宕机状态不仅仅是指服务器“挂掉了”、“死机了”状态,也包括服务器假死、停用、关闭等一些原因而导致出现的不能够正常运行的状态。

    13 引用 • 82 回帖 • 83 关注
  • Caddy

    Caddy 是一款默认自动启用 HTTPS 的 HTTP/2 Web 服务器。

    12 引用 • 54 回帖 • 165 关注
  • 分享

    有什么新发现就分享给大家吧!

    247 引用 • 1794 回帖
  • 禅道

    禅道是一款国产的开源项目管理软件,她的核心管理思想基于敏捷方法 scrum,内置了产品管理和项目管理,同时又根据国内研发现状补充了测试管理、计划管理、发布管理、文档管理、事务管理等功能,在一个软件中就可以将软件研发中的需求、任务、bug、用例、计划、发布等要素有序的跟踪管理起来,完整地覆盖了项目管理的核心流程。

    6 引用 • 15 回帖 • 34 关注
  • Markdown

    Markdown 是一种轻量级标记语言,用户可使用纯文本编辑器来排版文档,最终通过 Markdown 引擎将文档转换为所需格式(比如 HTML、PDF 等)。

    169 引用 • 1527 回帖
  • 锤子科技

    锤子科技(Smartisan)成立于 2012 年 5 月,是一家制造移动互联网终端设备的公司,公司的使命是用完美主义的工匠精神,打造用户体验一流的数码消费类产品(智能手机为主),改善人们的生活质量。

    4 引用 • 31 回帖 • 2 关注
  • GAE

    Google App Engine(GAE)是 Google 管理的数据中心中用于 WEB 应用程序的开发和托管的平台。2008 年 4 月 发布第一个测试版本。目前支持 Python、Java 和 Go 开发部署。全球已有数十万的开发者在其上开发了众多的应用。

    14 引用 • 42 回帖 • 805 关注
  • Mac

    Mac 是苹果公司自 1984 年起以“Macintosh”开始开发的个人消费型计算机,如:iMac、Mac mini、Macbook Air、Macbook Pro、Macbook、Mac Pro 等计算机。

    167 引用 • 595 回帖 • 1 关注
  • Postman

    Postman 是一款简单好用的 HTTP API 调试工具。

    4 引用 • 3 回帖
  • 以太坊

    以太坊(Ethereum)并不是一个机构,而是一款能够在区块链上实现智能合约、开源的底层系统。以太坊是一个平台和一种编程语言 Solidity,使开发人员能够建立和发布下一代去中心化应用。 以太坊可以用来编程、分散、担保和交易任何事物:投票、域名、金融交易所、众筹、公司管理、合同和知识产权等等。

    34 引用 • 367 回帖 • 1 关注
  • Visio
    1 引用 • 2 回帖
  • OpenResty

    OpenResty 是一个基于 NGINX 与 Lua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态 Web 应用、Web 服务和动态网关。

    17 引用 • 55 关注