Lambda03 方法引用、类型判断、变量引用
1 方法引用
1.1 方法引用的好处
方法引用结合 Lambda 可以引用已存在的方法,省略很多编码,而且可读性更强,它可以自动装配参数与返回值。
在编写lambda表达式的时候可以通过方法引用的方式来简化编写流程,例如:
1.2 静态方法引用
格式 -> 类名 :: 方法名
1.2.1 创建一个Student类
class Student { private String name = "王杨帅"; public static void info(Student student) {
System.out.println(student.name + "正在学习IT技能");
} @Override
public String toString() {
return this.name;
}
}
1.2.2 引用
在Lambda表达式中引用Student类中的静态方法info
Student :: info
技巧01:info 方法是一个静态方法,它的参数中没有this
技巧02:info 方法接收一个Student类型的参数,没有返回参数;所以 Student :: info 这个引用的结果是一个Consumer类型的实例
package demo_test; import java.util.function.Consumer;
import java.util.function.Supplier; /**
* @author 王杨帅
* @create 2018-07-30 9:54
* @desc
**/
public class TestDemo02 { public static void main(String[] args) { Student student = new Student(); Consumer<Student> consumer = Student :: info; consumer.accept(new Student()); } } class Student { private String name = "王杨帅"; public static void info(Student student) {
System.out.println(student.name + "正在学习IT技能");
} @Override
public String toString() {
return this.name;
}
}
1.3 实例方法引用
格式 -> 实例名 :: 方法名
1.3.1 重构Student类
新增一个money成员变量,新增一个useMoney成员方法
class Student { private String name = "王杨帅";
private Double money = 100d; public static void info(Student student) {
System.out.println(student.name + "正在学习IT技能");
} public Double useMoney(Double money) {
System.out.println(this.name + "花了" + money + "元");
return this.money -= money;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Double getMoney() {
return money;
} public void setMoney(Double money) {
this.money = money;
} @Override
public String toString() {
return this.name;
}
}
1.3.2 引用01【推荐方式】
利用实例变量引用
》创建一个Student实例student
》引用格式:student :: useMmoney
》技巧01:useMoney方法接收一个Double类型参数,返回一个Double类型的参数;所以 student :: useMmoney 引用返回的结果应该是一个Function类型的实例
package demo_test; import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier; /**
* @author 王杨帅
* @create 2018-07-30 9:54
* @desc
**/
public class TestDemo02 { public static void main(String[] args) { // demo01(); Student student = new Student();
Function<Double, Double> function = student :: useMoney;
System.out.println(student.getName() + "还剩下" + function.apply(10d) + "元"); } /**
* 静态方法的引用
*/
public static void demo01() {
Student student = new Student(); Consumer<Student> consumer = Student :: info; consumer.accept(new Student());
} } class Student { private String name = "王杨帅";
private Double money = 100d; public static void info(Student student) {
System.out.println(student.name + "正在学习IT技能");
} public Double useMoney(Double money) {
System.out.println(this.name + "花了" + money + "元");
return this.money -= money;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Double getMoney() {
return money;
} public void setMoney(Double money) {
this.money = money;
} @Override
public String toString() {
return this.name;
}
}
1.3.3 引用02【不推荐使用】
利用类来引用
》实体类不做任何修改
》引用格式:Student :: useMoney
》技巧01:由于useMoney里面用到了this关键字,所以在利用类名来引用实例方法时必须传入一个Student类型的实参,Student类中的userMoney方法不需要进行更改是因为JDK会默认给实例方法第一个参数设为this;所以 Student :: useMoney 返回的是一个 BiFunction 类型的实例
package demo_test; import java.util.function.BiFunction;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier; /**
* @author 王杨帅
* @create 2018-07-30 9:54
* @desc
**/
public class TestDemo02 { public static void main(String[] args) { // demo01(); // demo02(); BiFunction<Student, Double, Double> biFunction = Student :: useMoney;
Student student = new Student();
System.out.println(student.getName() + "还剩下" + biFunction.apply(student, 11d) + "元"); } /**
* 实例引用01【推荐方式】
*/
public static void demo02() {
Student student = new Student();
Function<Double, Double> function = student :: useMoney;
System.out.println(student.getName() + "还剩下" + function.apply(10d) + "元");
} /**
* 静态方法的引用
*/
public static void demo01() {
Student student = new Student(); Consumer<Student> consumer = Student :: info; consumer.accept(new Student());
} } class Student { private String name = "王杨帅";
private Double money = 100d; public static void info(Student student) {
System.out.println(student.name + "正在学习IT技能");
} public Double useMoney(Double money) {
System.out.println(this.name + "花了" + money + "元");
return this.money -= money;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Double getMoney() {
return money;
} public void setMoney(Double money) {
this.money = money;
} @Override
public String toString() {
return this.name;
}
}
1.4 构造器的引用
格式 -> Student :: new
1.4.1 重构Student类
添加一个无参构造器和有参构造器
class Student { private String name = "王杨帅";
private Double money = 100d; public Student() {
} public Student(String name, Double money) {
this.name = name;
this.money = money;
} public static void info(Student student) {
System.out.println(student.name + "正在学习IT技能");
} public Double useMoney(Double money) {
System.out.println(this.name + "花了" + money + "元");
return this.money -= money;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Double getMoney() {
return money;
} public void setMoney(Double money) {
this.money = money;
} @Override
public String toString() {
return this.name;
}
}
1.4.2 无参构造器的引用
》引用格式:Student :: new
》无参构造器没有输入参数,输出参数是一个Student类型,所以 Student :: new 的返回的是一个 Supplier 类型的实例
package demo_test; import java.util.function.BiFunction;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier; /**
* @author 王杨帅
* @create 2018-07-30 9:54
* @desc
**/
public class TestDemo02 { public static void main(String[] args) { // demo01(); // demo02(); // demo03(); Supplier<Student> supplier = Student :: new;
System.out.println(supplier.get()); } /**
* 实例引用02【不推荐使用】
*/
public static void demo03() {
BiFunction<Student, Double, Double> biFunction = Student :: useMoney;
Student student = new Student();
System.out.println(student.getName() + "还剩下" + biFunction.apply(student, 11d) + "元");
} /**
* 实例引用01【推荐方式】
*/
public static void demo02() {
Student student = new Student();
Function<Double, Double> function = student :: useMoney;
System.out.println(student.getName() + "还剩下" + function.apply(10d) + "元");
} /**
* 静态方法的引用
*/
public static void demo01() {
Student student = new Student(); Consumer<Student> consumer = Student :: info; consumer.accept(new Student());
} } class Student { private String name = "王杨帅";
private Double money = 100d; public Student() {
} public Student(String name, Double money) {
this.name = name;
this.money = money;
} public static void info(Student student) {
System.out.println(student.name + "正在学习IT技能");
} public Double useMoney(Double money) {
System.out.println(this.name + "花了" + money + "元");
return this.money -= money;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Double getMoney() {
return money;
} public void setMoney(Double money) {
this.money = money;
} @Override
public String toString() {
return this.name;
}
}
1.4.3 有参构造器的引用
》引用格式:Student :: new
疑惑:为什么有参构造器和无参构造器的输入参数都不一样,为什么引用格式一样呢?
解惑:lambda表达式的引用会自动进行以引用类型判断,自己去寻找符合条件的方法执行,开发人员无需担心弄错
》有参构造器有两个输入参数,输出参数也是一个Student类型,所以 Student :: new 返回的是一个 BiFunction 类型的实例
package demo_test; import java.util.function.BiFunction;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Supplier; /**
* @author 王杨帅
* @create 2018-07-30 9:54
* @desc
**/
public class TestDemo02 { public static void main(String[] args) { // demo01(); // demo02(); // demo03(); BiFunction<String, Double, Student> biFunction = Student :: new;
System.out.println(biFunction.apply("三少", 100d)); } /**
* 实例引用02【不推荐使用】
*/
public static void demo03() {
BiFunction<Student, Double, Double> biFunction = Student :: useMoney;
Student student = new Student();
System.out.println(student.getName() + "还剩下" + biFunction.apply(student, 11d) + "元");
} /**
* 实例引用01【推荐方式】
*/
public static void demo02() {
Student student = new Student();
Function<Double, Double> function = student :: useMoney;
System.out.println(student.getName() + "还剩下" + function.apply(10d) + "元");
} /**
* 静态方法的引用
*/
public static void demo01() {
Student student = new Student(); Consumer<Student> consumer = Student :: info; consumer.accept(new Student());
} } class Student { private String name = "王杨帅";
private Double money = 100d; public Student() {
} public Student(String name, Double money) {
this.name = name;
this.money = money;
} public static void info(Student student) {
System.out.println(student.name + "正在学习IT技能");
} public Double useMoney(Double money) {
System.out.println(this.name + "花了" + money + "元");
return this.money -= money;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Double getMoney() {
return money;
} public void setMoney(Double money) {
this.money = money;
} @Override
public String toString() {
return this.name;
}
}
2 类型判断
lambda表达式的返回结果是一个函数式接口类型的实例,lambda表达式的结果具体对应哪一个函数式接口类型有JDK自己进行判断;当然开发者如果清楚自己在做什么可以对lambda表达式的记过进行强制类型转换。
2.1 准备代码
》创建两个函数式接口
interface IMatch {
Integer add(Integer a, Integer b);
} interface IMatch02 {
Integer add(Integer a, Integer b);
}
2.2 类型判断分类
2.2.1 变量赋值
2.2.2 数组定义
2.2.3 强制转换
2.2.4 方法返回值
2.2.5 实际使用
一个方法的参数是一个函式接口,我们就可以利用lambda表达式了类型推断来实现;这种方式其实和类型定义是一样的
技巧01:重载方法引起lambda表达式不知道应该对应哪个函数接口时可以通过强制类型转化来实现
3 变量引用
3.1 在lambda表达式中使用this关键字
3.1.1 lambda表达式中this指向的是当前类
原因:lambda 表达式是函数式接口的实现类实例,所以,定义 lambda 表达式,实际上要经历两件事情。第一件事情是定义函数式接口实现类,第二件事情是创建该实现类实例。 this 称之为当前对象,但是, 定义 lambda 表达式时,也就是定义函数式接口实现类时, lambda 表达式代表的实现类本身没有 this 对象,此时若是使用 this 对象,指的是把 lambda 表达式围住的类的当前对象,而不是 lambda 表达式代表的实现类的当前对象。
/**
* notes:
* 1 java是传值的
* 2 匿名内部类中的this是指向内部类,lambda表达式的this是执行当前类
* 3 匿名内部类和lambda表达式引用当前类的变量时都需要时final修饰的成员,java8开始
* 可以不用final修饰,是因为jdk在编译的时候帮我们做了
* 4 引用的成员需要final修饰的原因时保证当前类的成员变量和匿名内部类或者labmda表达
* 式中应用的变量都指向同一个对象,如果没有final修饰,就很容易导致两个变量执行不同的变量,
* 从而导致程序出现一些bug
*/
Lambda03 方法引用、类型判断、变量引用的更多相关文章
- php引用计数与变量引用
每个php5.5变量都存储在一个叫做zval的变量容器中. 一个zval变量容器,除了包含变量的类型与值外,还包含两个字节的额外信息: 1.第一个是“is_ref”,是个bool型,用来标识这个变量是 ...
- console.log、toString方法与js判断变量类型
Java调用system.print.out()是会调用toString方法打印js里的console.log也是控制台打印,很多时候,我们以为也是调用toString方法,其实并不是.我们在chro ...
- JavaScript中判断变量类型最简洁的实现方法以及自动类型转换(#################################)
这篇文章主要介绍了JavaScript中判断整字类型最简洁的实现方法,本文给出多个判断整数的方法,最后总结出一个最短.最简洁的实现方法,需要的朋友可以参考下 我们知道JavaScript提供了type ...
- C# 异常:从作用域“”引用了“FiasHostApp.Entity.DBEntity.FIAS_RM_v1.ITraNetMgrUnitBaseInfoRecord”类型的变量“w”,但该变量未定义
C# 异常:从作用域“”引用了“FiasHostApp.Entity.DBEntity.FIAS_RM_v1.ITraNetMgrUnitBaseInfoRecord”类型的变量“w”,但该变量未定义 ...
- python list类型的变量相当于全局变量 可以被函数与类引用
python list类型的变量相当于全局变量 可以被函数与类引用
- Java 中 父类变量访问子类方法 需要使用 类型转换 (instenceof)关键字 /类型判断/
通过数组元素访问方法的时候只能访问在 Animal中定义的方法,对 于 Tiger类和 Fish中定义的方法时却不能调用,例如语句 animal[2].swim();就是不正确的.当 需要访问这些 ...
- JS 中对变量类型判断的几种方式
文章整理搬运,出处不详,如有侵犯,请联系~ 数据类型判断和数据类型转换代码工具 在 JS 中,有 5 种基本数据类型和 1 种复杂数据类型,基本数据类型有:Undefined, Null, Boo ...
- 关于java内存泄露的总结--引用的类型:强引用,弱引用,软引用
今天面试了一家公司的java开发方面的实习生,被问到一个问题:如何处理java中的内存泄露问题,保证java的虚拟机内存不会被爆掉,当时其实觉得面试官的问题有点泛,所以也没有很好领会他的意思,答案也不 ...
- 2.8 补充:shell变量引用方式
一 变量 变量:本质上是程序中保存用户数据的一块内存空间,变量名就是内存空间地址. Shell中:变量可由字母数字和下划线组成,以字母或下划线开头. 命名:PATH=/sbin ...
随机推荐
- php session在高并发时可能存在的问题。
如果同一个客户端并发发送多个请求,而每个请求都使用了Session,那么PHP Session锁的存在会导致服务器串行响应这些请求,而不是并行.这是因为在默认情况下,PHP使用文件存储Session数 ...
- kafka系列之(3)——Coordinator与offset管理和Consumer Rebalance
from:http://www.jianshu.com/p/5aa8776868bb kafka系列之(3)——Coordinator与offset管理和Consumer Rebalance 时之结绳 ...
- 第13篇 PSR-2代码样式
这个规范原文如下: 1. Overview Code MUST follow a "coding style guide" PSR [PSR-1]. Code MUST use 4 ...
- ZipArchive扩展的使用和Guzzle依赖的安装使用
在项目开发的过程中,需要去远程下载录音文件 然后保存到自己的项目中,然后再把录音文件压缩打包,最后再下载给用户 1.Guzzle依赖的安装 guzzle官方文档:http://guzzle-cn.re ...
- FPGA前世今生(一)
关于FPGA,我想做硬件的或多或少都听过.从上世纪80年代算来,FPGA已走过了30多个年头.我们以FPGA两大生产厂商,两大巨头之一的INTEL(altera)公司的FPGA为例,为大家逐步介绍FP ...
- Mac电脑Tomcat下载及安装(详细)
下载Tomcat 1.打开Apache Tomcat官网,选择你需要的版本进行下载: 地址http://tomcat.apache.org/download-70.cgi 2.解压apache-t ...
- 让memcached分布式
memcached是应用最广的开源cache产品,它本身不提供分布式的解决方案,我猜想一方面它想尽量保持产品简单高效,另一方面cache的key-value的特性使得让memcached分布式起来比较 ...
- ByteBuf 类——Netty 的数据容器
1.堆缓冲区 2.直接缓冲区 3.复合缓冲区 —CompositeByteBuf——实现了这个模式,它提供了一 个将多个缓冲区表示为单个合并缓冲区的虚拟表示 适用于 JDK 所使用的一种称为分散/收集 ...
- 1137 Final Grading
题意:排序题. 思路:通过unordered_map来存储考生姓名与其成绩信息结构体的映射,成绩初始化为-1,在读入数据时更新各个成绩,最后计算最终成绩并把符合条件的学生存入vector,再排序即可. ...
- PHP中的精确计算bcadd,bcsub,bcmul,bcdiv
引言:一定要确保数据的准确性.这是一个好的程序员的基本素养. <?php /** * 精确加法 * @param [type] $a [description] * @param [type] ...