Java 8 特性 —— lambda 表达式
Lambda 表达式
Lambda表达式本质上是一个匿名方法。常见的一个例子是,在 IDEA + JDK8 的环境下按照Java传统的语法规则编写一个线程:
new Thread(new Runnable() {
@Override
public void run() {
System.out.println("Hello World!");
}
});
IDEA会给出提示 Anonymous new Runnable() can be replaced with lambda less... (Ctrl+F1) ,说匿名内部类可以使用 Lambda 表达式替换。
使用 Lambda 表达式则只需要使用一句话就可代替上面使用匿名类的方式。
new Thread(() -> System.out.println("Hello World!"));
在这个例子中,传统的语法规则,我们是将一个匿名内部类作为参数进行传递,我们实现了Runnable接口,并将其作为参数传递给Thread类,这实际上我们传递的是一段代码,也即我们将代码作为了数据进行传递,这就带来许多不必要的“样板代码”。
在JDK8之前,Java是不支持函数式编程的,所谓的函数编程,即可理解是将一个函数(也称为“行为”)作为一个参数进行传递。通常我们提及得更多的是面向对象编程,面向对象编程是对数据的抽象(各种各样的POJO类),而函数式编程则是对行为的抽象(将行为作为一个参数进行传递)。在JavaScript中这是很常见的一个语法特性,但在Java中将一个函数作为参数传递这却行不通,好在JDK8的出现打破了Java的这一限制。
如上例所见,Lambda表达式一共有三部分组成:参数列表,箭头(->),以及一个表达式或语句块。
Lambda 表达式和函数式接口结合
什么场景下可以使用 Lambda 表达式?
能够接收 Lambda 表达式的参数类型,是一个只包含一个抽象方法的接口。只包含一个抽象方法的接口为“函数式接口”。
例如上例,Runnable 接口只包含一个抽象方法,所以它被称为“函数式接口”,所以它可以使用 Lambda 表达式来代替匿名内部类,作为 Thread 的构造参数。
函数式接口有无入参和有无返回值,可分为四种情形:
无参无返回值
有参无返回值
无参有返回值
有参有返回值
Lambda 表达式和这四种形式的函数式接口结合也就有四种情况;
无参无返回值
package com.boomoom; @FunctionalInterface
public interface InterfaceWithNoParam {
void run();
}
package com.boomoom;
import org.junit.Test;
public class FunctionInterfaceTest {
//匿名内部类
InterfaceWithNoParam param1 = new InterfaceWithNoParam() {
@Override
public void run() {
System.out.println("通过匿名内部类实现run()");
}
};
//Lambda表达式
//空括号表示无参
InterfaceWithNoParam param = () -> System.out.println("通过Lambda表达式实现run()") ;
@Test
public void testInterfaceWithNoParam() {
this.param.run();
this.param1.run();
}
}
通过Lambda表达式实现run()
通过匿名内部类实现run()
有参无返回值
package com.boomoom; @FunctionalInterface
public interface InterfaceWithParam {
void run(String s);
}
package com.boomoom;
import org.junit.Test;
public class FunctionInterfaceTest {
InterfaceWithParam param = new InterfaceWithParam() {
@Override
public void run(String s) {
System.out.println("通过" + s + "实现run(String)");
}
};
InterfaceWithParam param1 = (String s) -> System.out.println("通过" + s + "实现run(String)");
@Test
public void testInterfaceWithParam() {
this.param.run("匿名类");
this.param1.run("Lambda");
}
}
通过匿名类实现run(String)
通过Lambda实现run(String)
Lambda 表达式“(String) -> Sysout.out.println("通过" + s + "实现run(String)")”,左边传递的是参数,此处指明参数类型,也可不指定,通过上下文进行类型推导;但在有些情况下不能推导出参数类型(在编译时不能推导通常IDE会提示),此时则需要指明参数类型。我个人建议,任何情况下指明函数的参数类型。
哪种情况不能推导出参数类型呢?就是函数接口是一个泛型的时候。
@FunctionalInterface
public interface InterfaceWithParam<T> {
void run(T param);
}
无参有返回值
package com.boomoom; @FunctionalInterface
public interface InterfaceUnVoidWithNoParam {
String run();
}
package com.boomoom;
import org.junit.Test;
public class FunctionInterfaceTest {
InterfaceUnVoidWithNoParam interfaceUnVoidWithNoParam = new InterfaceUnVoidWithNoParam() {
@Override
public String run() {
return "Hello 匿名内部类!";
}
};
InterfaceUnVoidWithNoParam interfaceUnVoidWithNoParam1 = () -> "Hello Lambda!";
@Test
public void test() {
System.out.println("返回结果是:"+ this.interfaceUnVoidWithNoParam.run());
System.out.println("返回结果是:"+ this.interfaceUnVoidWithNoParam1.run());
}
}
返回结果是:Hello 匿名内部类!
返回结果是:Hello Lambda!
有参有返回值
package com.boomoom; @FunctionalInterface
public interface InterfaceUnVoidWithParam {
String run(Integer i);
}
package com.boomoom;
import org.junit.Test;
public class FunctionInterfaceTest {
InterfaceUnVoidWithParam interfaceWithParams = new InterfaceUnVoidWithParam() {
@Override
public String run(Integer integer) {
return String.valueOf(integer);
}
};
InterfaceUnVoidWithParam interfaceWithParams1 = (Integer integer) -> String.valueOf(integer);
@Test
public void test() {
System.out.println("您输入的是:"+ this.interfaceWithParams.run(1));
System.out.println("您输入的是:"+ this.interfaceWithParams1.run(2));
}
}
您输入的是:1
您输入的是:2
这四种基本情况已经大致清楚了,特别是需要弄清,什么时候可以使用 Lambda 表达式代替匿名内部类,也就是 Lambda 表达式的应用场景是函数式接口。Lambda 表达式这一新特性在 JDK8 中的引入,更大的好处则是集合 API 的更新,新增的 Stream 类库,使得我们在遍历使用集合时不再像以往那样不断地使用 for 循环。
JDK8 使用集合
示例:计算来自“hangzhou”的学生数量有多少。
在JDK8 前的代码:
for (Student student : studentList) {
if (student.getCity().equals("hangzhou")) {
count++;
}
}
JDK8 使用集合的正确姿势:
count = studentList.stream().filter((student -> student.getCity().equals("hangzhou"))).count();
API 的使用“难度”恰似提高了,实际只是不熟悉而已。传统迭代的方式需要阅读完整个循环才能明白代码逻辑,JDK8 通过流的方式则可以望文生义且代码量大大减小。
其中最为重要的是 Stream 流。Stream 的是通过函数式编程方式实现的在集合类上进行复杂操作的工具。关于 Stream 的用法和实现方式,详见此篇。使用JDK8的开发环境进行开发,尽量使用新的集合操作 API。
参考:https://www.cnblogs.com/yulinfeng/p/8452379.html
http://www.runoob.com/java/java8-lambda-expressions.html
https://www.cnblogs.com/yw0219/p/7302597.html
https://blog.csdn.net/io_field/article/details/54380200
Java 8 特性 —— lambda 表达式的更多相关文章
- 关于java新特性lambda表达式的理解即使用
Lambda 表达式的使用 1.举例: (o1,o2) -> Integer.compare(o1,o2); 2.格式: -> : lambda操作符 或 箭头操作符 ->左边 : ...
- Java 8 新特性 - Lambda表达式
Lambda表达式 vs 匿名类既然lambda表达式即将正式取代Java代码中的匿名内部类,那么有必要对二者做一个比较分析.一个关键的不同点就是关键字 this.匿名类的 this 关键字指向匿名类 ...
- Lambda 表达式,Java中应用Lambda 表达式
一.Lambda 表达式 简单来说,编程中提到的 lambda 表达式,通常是在需要一个函数,但是又不想费神去命名一个函数的场合下使用,也就是指匿名函数. 链接:知乎 先举一个普通的 Python 例 ...
- Java8新特性-Lambda表达式是什么?
目录 前言 匿名内部类 函数式接口 和 Lambda表达式语法 实现函数式接口并使用Lambda表达式: 所以Lambda表达式是什么? 实战应用 总结 前言 Java8新特性-Lambda表达式,好 ...
- Java 终于有 Lambda 表达式啦~Java 8 语言变化——Lambda 表达式和接口类更改【转载】
原文地址 en cn 下载 Demo Java™ 8 包含一些重要的新的语言功能,为您提供了构建程序的更简单方式.Lambda 表达式 为内联代码块定义一种新语法,其灵活性与匿名内部类一样,但样板文件 ...
- 乐字节-Java8新特性-Lambda表达式
上一篇文章我们了解了Java8新特性-接口默认方法,接下来我们聊一聊Java8新特性之Lambda表达式. Lambda表达式(也称为闭包),它允许我们将函数当成参数传递给某个方法,或者把代码本身当作 ...
- java 8 中lambda表达式学习
转自 http://blog.csdn.net/renfufei/article/details/24600507 http://www.jdon.com/idea/java/10-example-o ...
- Java 8中Lambda表达式默认方法的模板方法模式,你够了解么?
为了以更简单的术语描述模板方法,考虑这个场景:假设在一个工作流系统中,为了完成任务,有4个任务必须以给定的执行顺序执行.在这4个任务中,不同工作流系统的实现可以根据自身情况自定义任务的执行内容. 模板 ...
- JDK新特性-Lambda表达式的神操作
一.Lambda表达式的介绍 Lambda表达式是 Java8 中最重要的新功能之一.使用 Lambda 表达 式可以替代只有一个抽象函数的接口实现,告别匿名内部类,代码看 起来更简洁易懂.Lambd ...
随机推荐
- jQuery(八)、ajax
1.jQuery.ajax(url[, settings]) 通过HTTP请求加载远程数据. 注意:所有的settings选择都可以通过$.ajaxSetup()函数来全局指定. 回调函数 在实际开发 ...
- 折腾Java设计模式之建造者模式
博文原址:折腾Java设计模式之建造者模式 建造者模式 Separate the construction of a complex object from its representation, a ...
- 基于html5 plus + Mui 移动App开发(一)
使用Html5 plus + Mui 进行移动App开发,有一段时间了,这几日得空,做个资讯App分享给大家. 今天主要分享主页实现,首先看下效果: 此界面主要分为:标题.内容分类列表.搜索及设置按钮 ...
- 软件工程实践-WC项目之C实现
1.Github项目地址 https://github.com/ShadowEvan/homework 基本功能 -c 统计文件字符数(实现) -w 统计文件词数(实现) -l 统计文件行数(实现) ...
- Linux学习历程——Centos 7 chmod命令
一.命令介绍 chmod 命令,是Linux管理员最常用的命令之一,用于修改文件或目录的访问权限. Linux系统中,每一个文件都有文件所有者和所属群组,并且规定文件的所有者,所属群组,以及其他人队问 ...
- kali中的postgres怎么连接
metasploit,默认使用的是postgresql数据库.在BT5或是Kali等专业的渗透系统中,postgresql已经被默认安装. 执行msfdb init会自动创建一个默认的用户名密码都是m ...
- Conway生命游戏
版权申明:本文为博主窗户(Colin Cai)原创,欢迎转帖.如要转贴,必须注明原文网址 http://www.cnblogs.com/Colin-Cai/p/9986679.html 作者:窗户 Q ...
- shiro过滤器详解分析
(原) shiro最核心的2个操作,一个是登录的实现,一就是过滤器了.登录有时间再补录说明,这里分析下shiro过滤器怎样玩的. 1.目标 这里会按如下顺序逐一看其实原理,并尽量找出其出处. 先看一下 ...
- MyBatis学习日记(三):戏说MyBatis配置文件
properties标签 properties标签可以用来加载别的配置文件,比如可以加载数据库的配置文件,jdbc.properties. 下面是jdbc.properties jdbc.driver ...
- Redis报错 Server started, Redis version 3.2.13 Can't handle RDB format version 9 Fatal error loading the DB: Invalid argument. Exiting.
在/usr/local/etc 目录下 运行 redis-server 命令重启 redis 服务发现报错,报错信息如下: 如上报错的含义是:当前的redis的版本是3.2.13版本,无法处理 ver ...