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

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

上一篇博客中,我们继续介绍了一些关于 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 引用 • 3454 回帖 • 189 关注
  • Java

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

    3190 引用 • 8214 回帖 • 1 关注
  • 方法引用
    3 引用

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • Ruby

    Ruby 是一种开源的面向对象程序设计的服务器端脚本语言,在 20 世纪 90 年代中期由日本的松本行弘(まつもとゆきひろ/Yukihiro Matsumoto)设计并开发。在 Ruby 社区,松本也被称为马茨(Matz)。

    7 引用 • 31 回帖 • 216 关注
  • 以太坊

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

    34 引用 • 367 回帖 • 1 关注
  • 爬虫

    网络爬虫(Spider、Crawler),是一种按照一定的规则,自动地抓取万维网信息的程序。

    106 引用 • 275 回帖 • 1 关注
  • SOHO

    为成为自由职业者在家办公而努力吧!

    7 引用 • 55 回帖 • 5 关注
  • CodeMirror
    1 引用 • 2 回帖 • 129 关注
  • Swagger

    Swagger 是一款非常流行的 API 开发工具,它遵循 OpenAPI Specification(这是一种通用的、和编程语言无关的 API 描述规范)。Swagger 贯穿整个 API 生命周期,如 API 的设计、编写文档、测试和部署。

    26 引用 • 35 回帖 • 5 关注
  • WiFiDog

    WiFiDog 是一套开源的无线热点认证管理工具,主要功能包括:位置相关的内容递送;用户认证和授权;集中式网络监控。

    1 引用 • 7 回帖 • 591 关注
  • 旅游

    希望你我能在旅途中找到人生的下一站。

    93 引用 • 899 回帖 • 3 关注
  • FlowUs

    FlowUs.息流 个人及团队的新一代生产力工具。

    让复杂的信息管理更轻松、自由、充满创意。

    1 引用
  • Mac

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

    166 引用 • 595 回帖
  • 博客

    记录并分享人生的经历。

    273 引用 • 2388 回帖 • 1 关注
  • Angular

    AngularAngularJS 的新版本。

    26 引用 • 66 回帖 • 536 关注
  • QQ

    1999 年 2 月腾讯正式推出“腾讯 QQ”,在线用户由 1999 年的 2 人(马化腾和张志东)到现在已经发展到上亿用户了,在线人数超过一亿,是目前使用最广泛的聊天软件之一。

    45 引用 • 557 回帖 • 44 关注
  • Markdown

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

    167 引用 • 1520 回帖 • 1 关注
  • 黑曜石

    黑曜石是一款强大的知识库工具,支持本地 Markdown 文件编辑,支持双向链接和关系图。

    A second brain, for you, forever.

    16 引用 • 130 回帖
  • Bug

    Bug 本意是指臭虫、缺陷、损坏、犯贫、窃听器、小虫等。现在人们把在程序中一些缺陷或问题统称为 bug(漏洞)。

    76 引用 • 1737 回帖
  • 数据库

    据说 99% 的性能瓶颈都在数据库。

    343 引用 • 723 回帖
  • 又拍云

    又拍云是国内领先的 CDN 服务提供商,国家工信部认证通过的“可信云”,乌云众测平台认证的“安全云”,为移动时代的创业者提供新一代的 CDN 加速服务。

    21 引用 • 37 回帖 • 548 关注
  • LeetCode

    LeetCode(力扣)是一个全球极客挚爱的高质量技术成长平台,想要学习和提升专业能力从这里开始,充足技术干货等你来啃,轻松拿下 Dream Offer!

    209 引用 • 72 回帖
  • JWT

    JWT(JSON Web Token)是一种用于双方之间传递信息的简洁的、安全的表述性声明规范。JWT 作为一个开放的标准(RFC 7519),定义了一种简洁的,自包含的方法用于通信双方之间以 JSON 的形式安全的传递信息。

    20 引用 • 15 回帖 • 6 关注
  • 阿里巴巴

    阿里巴巴网络技术有限公司(简称:阿里巴巴集团)是以曾担任英语教师的马云为首的 18 人,于 1999 年在中国杭州创立,他们相信互联网能够创造公平的竞争环境,让小企业通过创新与科技扩展业务,并在参与国内或全球市场竞争时处于更有利的位置。

    43 引用 • 221 回帖 • 106 关注
  • Spring

    Spring 是一个开源框架,是于 2003 年兴起的一个轻量级的 Java 开发框架,由 Rod Johnson 在其著作《Expert One-On-One J2EE Development and Design》中阐述的部分理念和原型衍生而来。它是为了解决企业应用开发的复杂性而创建的。框架的主要优势之一就是其分层架构,分层架构允许使用者选择使用哪一个组件,同时为 JavaEE 应用程序开发提供集成的框架。

    943 引用 • 1460 回帖 • 3 关注
  • 微信

    腾讯公司 2011 年 1 月 21 日推出的一款手机通讯软件。用户可以通过摇一摇、搜索号码、扫描二维码等添加好友和关注公众平台,同时可以将自己看到的精彩内容分享到微信朋友圈。

    132 引用 • 795 回帖
  • 星云链

    星云链是一个开源公链,业内简单的将其称为区块链上的谷歌。其实它不仅仅是区块链搜索引擎,一个公链的所有功能,它基本都有,比如你可以用它来开发部署你的去中心化的 APP,你可以在上面编写智能合约,发送交易等等。3 分钟快速接入星云链 (NAS) 测试网

    3 引用 • 16 回帖 • 5 关注
  • App

    App(应用程序,Application 的缩写)一般指手机软件。

    91 引用 • 384 回帖 • 2 关注
  • Node.js

    Node.js 是一个基于 Chrome JavaScript 运行时建立的平台, 用于方便地搭建响应速度快、易于扩展的网络应用。Node.js 使用事件驱动, 非阻塞 I/O 模型而得以轻量和高效。

    139 引用 • 269 回帖 • 28 关注
  • Postman

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

    4 引用 • 3 回帖 • 7 关注