Java中的函数式编程(三)lambda表达式
写在前面
lambda表达式与匿名内部类
无参的函数式接口
public static void createThreadWithAnonymousClass() {
// Runnable 是接口名。我们通过匿名内部类的方式,构造了一个 Runnable 的实例。
Thread t = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Thread is running");
}
});
t.start();
}
使用匿名内部类的一个重要目的,就是为了减轻程序员的代码负担,不需要额外再定义一个类,而且这个类是一个一次性的类,没有太多的重用价值。但是,我们会发现,这个对象看起来也是多余的,因为我们实际上并不是要传入一个对象,而只是想传入一个方法。
public static void createThreadWithLambda() {
// 在Java 8中,Runnable 是一个函数式接口,因此我们可以使用 lambda 表达式来实现它。
Thread t = new Thread(() -> {
System.out.println("Thread is running");
});
t.start();
}
带参的函数式接口
@FunctionalInterface
public interface Comparator<T> {
int compare(T o1, T o2); ....
}
假设一个场景:给定一个省份的拼音列表,需要对该列表中的省份进行排序,排序规则是字母长度最小的省份排在前面,如果两个省份字母长度一样,则按字母顺序排序。
public static void sortProvincesWithAnonymousClass() {
List list = Arrays.asList("Guangdong", "Zhejiang", "Jiangsu", "Xizang", "Fujian", "Hunan", "Guangxi");
list.sort(new Comparator<String>() {
@Override
public int compare(String first, String second) {
int lenDiff = first.length() - second.length();
return lenDiff == 0 ? first.compareTo(second) : lenDiff;
}
});
list.forEach(s -> System.out.println(s));
}
上述代码输出为:
public static void sortProvincesWithLambda() {
List list = Arrays.asList("Guangdong", "Zhejiang", "Jiangsu", "Xizang", "Fujian", "Hunan", "Guangxi");
// 下面的参数列表 first 和 second ,即方法 Comparator.compare 的参数列表
list.sort((first, second) -> {
int lenDiff = first.length() - second.length();
return lenDiff == 0 ? first.compareTo(second) : lenDiff;
});
list.forEach(s -> System.out.println(s));
}
注意到,带参数的lambda表达式,甚至不需要声明类型,因为编译器可以通过上下文来推断出参数的类型。当然,我们也可以显式指定参数类型,尤其是在参数类型推断失败的时候:
(String first, String second) -> {
int lenDiff = first.length() - second.length();
return lenDiff == 0 ? first.compareTo(second) : lenDiff;
}
this关键字的作用域
public class ThisScopeExample {
public static void main(String[] args) {
ThisScopeExample example = new ThisScopeExample();
// 输出 "I am Anonymous Class."
example.runWithAnonymousClass();
// 输出 "I am ThisScopeExample Class."
example.runWithLambda();
}
public void runWithAnonymousClass() {
// 以匿名类的方式运行
run(new Runnable() {
@Override
public void run() {
// this 是实现了接口 Runnable 的匿名内部类的实例
System.out.println(this);
}
@Override
public String toString() {
return "I am Anonymous Class.";
}
});
}
public void runWithLambda() {
// 以lambda表达式的方式运行
run(() -> {
// this 是类 ThisScopeExample 的实例
System.out.println(this);
});
}
public void run(Runnable runnable) {
runnable.run();
}
@Override
public String toString() {
return "I am ThisScopeExample Class.";
}
}
lambda表达式的语法
(String first, String second) -> {
int lenDiff = first.length() - second.length();
return lenDiff == 0 ? first.compareTo(second) : lenDiff;
}
上述是一个典型的而且完整的lambda表达式。
Supplier supplier = () -> {
return new Random().nextInt(100);
}
对于上面的lambda表达式,可以发现它的方法体只有一个表达式,所以,它可以省略大括号,甚至return关键字也省略了,因为编译器可以根据上下文推断是否需要返回值:如果需要,那么就返回该唯一表达式的返回值,如果不需要,则在该唯一表达式后直接return。例如:
// Supplier 是需要返回值的,所以下面的lambda表达式等同于:
// () -> { return new Random().nextInt(100); }
Supplier supplier = () -> new Random().nextInt(100); // Runnable 是不需要返回值的,所以下面的lambda表达式等同于:
// () -> { new Random().nextInt(100); return; }
Runnable runnable = () -> new Random().nextInt(100);
如果编译器可以推断出lambda表达式的参数类型,则可以忽略其类型:
// 在这里,编译器可以推断出 first 和 second 的类型是 String。
Comparator comp = (first, second) -> {
int lenDiff = first.length() - second.length();
return lenDiff == 0 ? first.compareTo(second) : lenDiff;
};
如果lambda表达式只有一个参数,那么参数列表中的小括号也可以省略掉:
// 这里的 value ,等同于 (value)
Consumer consumer = value -> System.out.println(value);
与普通的函数不一样,lambda表达式不需要指定返回类型,它总是由编译器自行推断出返回类型。如果推断失败,则默认为Object类型。
lambda表达式与闭包
public class ClosureExample {
public static void main(String[] args) {
// 平方
IntUnaryOperator square = getPowOperator(2);
// 立方
IntUnaryOperator cube = getPowOperator(3);
// 四次方
IntUnaryOperator fourthPower = getPowOperator(4);
// 5的平方
System.out.println(square.applyAsInt(5));
// 5的立方
System.out.println(cube.applyAsInt(5));
// 5的四次方
System.out.println(fourthPower.applyAsInt(5));
}
public static IntUnaryOperator getPowOperator(int exp) {
return base -> {
// 变量 exp 是 getPowOperator 的参数,属于lambda 表达式定义时的自由变量,
// 它的生命周期会延长到和返回的 lambda 表达式一样长。
return (int) Math.pow(base, exp);
};
}
}
上述代码的输出是:
public static IntUnaryOperator getPowOperator(int exp) {
// 尝试修改 exp 的值,但编译器会在lambda表达式中报错
exp++;
return base -> {
// 如果尝试修改 exp 的值,会在此处报错:
// Error: 从lambda 表达式引用的本地变量必须是final变量或实际上的final变量
return (int) Math.pow(base, exp);
};
}
但这种限制也是有限的,因为我们可以通过将变量声明为一个数组或一个类就可以修改其中的值。例如:
public static IntUnaryOperator getPowOperator(int[] exp) {
// exp 是一个int数组:exp = new int[1];
exp[0]++;
return base -> {
// 此时不会报错,可以正常运行
return (int) Math.pow(base, exp[0]);
};
}
结语
为方便大家在移动端浏览,已注册微信公众号【员说】,欢迎关注。第一时间更新技术文章,也会不定时分享圈内热门动态和一线大厂内幕。
感谢您阅读本篇文章,如果觉得本文对您有帮助,欢迎点击推荐和关注,您的支持是我最大的写作动力。
文章欢迎转载,但需在文章页面明显位置,给出作者和原文链接,否则保留追究法律责任的权利!
注意!应各位朋友的邀请,创建了一个技术交流群,(聊技术/看内幕/找内推/读书分享等,拒绝水群,保证品质),可添加微信号【yuanshuo824】,备注:交流,即可入群。
Java中的函数式编程(三)lambda表达式的更多相关文章
- Java 中的函数式编程(Functional Programming):Lambda 初识
Java 8 发布带来的一个主要特性就是对函数式编程的支持. 而 Lambda 表达式就是一个新的并且很重要的一个概念. 它提供了一个简单并且很简洁的编码方式. 首先从几个简单的 Lambda 表达式 ...
- Java 函数式编程(Lambda表达式)与Stream API
1 函数式编程 函数式编程(Functional Programming)是编程范式的一种.最常见的编程范式是命令式编程(Impera Programming),比如面向过程.面向对象编程都属于命令式 ...
- Java 函数式编程和Lambda表达式
1.Java 8最重要的新特性 Lambda表达式.接口改进(默认方法)和批数据处理. 2.函数式编程 本质上来说,编程关注两个维度:数据和数据上的操作. 面向对象的编程泛型强调让操作围绕数据,这样可 ...
- Java中的函数式编程(五)Java集合框架中的高阶函数
写在前面 随着Java 8引入了函数式接口和lambda表达式,Java 8中的集合框架(Java Collections Framework, JCF)也增加相应的接口以适应函数式编程. 本文的 ...
- Java中的函数式编程(六)流Stream基础
写在前面 如果说函数式接口和lambda表达式是Java中函数式编程的基石,那么stream就是在基石上的最富丽堂皇的大厦. 只有熟悉了stream,你才能说熟悉了Java 的函数式编程. 本文主要介 ...
- Java函数式编程和lambda表达式
为什么要使用函数式编程 函数式编程更多时候是一种编程的思维方式,是种方法论.函数式与命令式编程的区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉代码要怎么做.说白了,函数式编程是基于 ...
- 函数式编程--使用lambda表达式
前面一篇博客我们已经说到了,lambda表达式允许使用更简洁的代码来创建只有一个抽象方法的接口的实例.现在我们来写一段java的命令者模式来自己研究下lambda表达式的语法. 这里重复下命令者模式: ...
- Java8函数式编程和lambda表达式
文章目录函数式编程JDK8接口新特性函数接口方法引用函数式编程函数式编程更多时候是一种编程的思维方式,是一种方法论.函数式与命令式编程区别主要在于:函数式编程是告诉代码你要做什么,而命令式编程则是告诉 ...
- Java8函数式编程以及Lambda表达式
第一章 认识Java8以及函数式编程 尽管距离Java8发布已经过去7.8年的时间,但时至今日仍然有许多公司.项目停留在Java7甚至更早的版本.即使已经开始使用Java8的项目,大多数程序员也仍然采 ...
随机推荐
- thrift的介绍及其使用
什么是thrift Thrift是Facebook于2007年开发的跨语言的rpc服框架,提供多语言的编译功能,并提供多种服务器工作模式:用户通过Thrift的IDL(接口定义语言)来描述接口函数及数 ...
- 🏆【Alibaba工具型技术系列】「EasyExcel技术专题」摒除OOM!让你的Excel操作变得更加优雅和安全
前提概要 针对于后端开发者而言的,作为报表的导入和导出是一个很基础且有很棘手的问题!之前常用的工具和方案大概有这么几种: JXL(Java Excel API 工具服务),此种只支持xls的文件格式, ...
- docker数据卷(Data Volumes)
Docker宿主机和容器之间文件拷贝docker copy 前言: Docker 数据管理 在生产环境中使用 Docker ,往往需要对数据进行持久化,或者需要在多个容器之间进行 数据共享,这必然涉及 ...
- noip模拟42
A. 卷 发现乘积足以爆 \(long\) \(long\),但是数据随机,可以略忽略精度问题 一个快速降低数的级别的方法是取对数,由于有性质 \(log(x * y)=logx+logy\),合并时 ...
- vue 路由视图,router-view嵌套跳转
实现功能:制作一个登录页面,跳转到首页,首页包含菜单栏.顶部导航栏.主体,标准的后台网页格式.菜单栏点击不同菜单控制主体展示不同的组件(不同的页面). 配置router-view嵌套跳转需要准备两个主 ...
- 《通过刷leetcode学习Go语言》之(1):序言
Author : Email : vip_13031075266@163.com Date : 2021.03.07 Version : 北京 C ...
- ubuntu 20.04 发邮件配置
安装sendmail后,发邮件一直没有成功,因此卸载sendmail后,安装heirloom-mailx. # unbuntu 18.04和20.04移除了heirloom-mailx,需要另外配置软 ...
- 编写一个应用程序,利用数组或者集合, 求出"HELLO",“JAVA”,“PROGRAM”,“EXCEPTION”四个字符串的平均长度以及字符出现重复次数最多的字符串。
public class Number { public static void main(String[] args) { String[] arr = { "HELLO", & ...
- private关键字理解
private 意思: 私有的 私人的 不公开的 private 是一个修饰符可以用来修饰成员变量和方法 被private修饰的成员变量或成员方法,只能在本类中访问,针对private修饰的成员变量, ...
- C# 中 async 和 await 的基本使用
C# 中 async 和 await 的基本使用 前言 经常在 C# 的代码中看到以 Async 结尾的方法,大概知道意为异步方法,但不知道怎么使用,也不知道如何定义. 对于"同步" ...