java8(一)Lambda表达式
其实很久前已经学习过了Lambda表达式,但是学习后没有多少使用的机会,久而久之也就忘记(惭愧)。最近新的项目用的jdk8所以准备再学习一次,写下文章也是为了记录,方便以后再忘的时候,不用到处找资料(旁白:这么丢人的事,能别到处说吗)。
语法
(参数) -> 表达式或(参数) -> {表达式}
第一个Lambda表达式
public class Java8Lambda { @Test public void test1() { // lambda表达式 Greeting helloLambda = (name) -> "Hello! " + name; // 调用方法 System.out.println(helloLambda.sayHello("Lambda")); } interface Greeting { String sayHello(String name); }}
输出结果:
Hello! Lambda
Lambda表达式可以看成是匿名类一点点演变过来
匿名类的写法
@Testpublic void test2() { // 匿名类的写法 Greeting helloLambda = new Greeting() { @Override public String sayHello(String name) { return "Hello! " + name; } }; // 调用方法 System.out.println(helloLambda.sayHello("Lambda"));}
设计匿名内部类的目的,就是为了方便java程序员将代码作为数据传递。不过匿名内部类还是不够简便。
去除外壳,保留方法参数和方法体,并在参数和方法体之间加上符号->(这里已经变成Lambda了,但不是最简洁的,请往下看)
@Testpublic void test3() { // 去除外壳,保留方法参数和方法体,并在参数和方法体之间加上符号-> Greeting helloLambda =(String name) -> { return "Hello! " + name; }; // 调用方法 System.out.println(helloLambda.sayHello("Lambda"));}
去除return和{}
@Testpublic void test4() { // 去除return和{} Greeting helloLambda =(String name) -> "Hello! " + name; // 调用方法 System.out.println(helloLambda.sayHello("Lambda"));}
去除参数类型和圆括号(只有一个参数的时候,才可以去掉圆括号)
@Testpublic void test5() { // 去除参数类型和圆括号(只有一个参数的时候,才可以去掉圆括号) Greeting helloLambda = name -> "Hello! " + name; // 调用方法 System.out.println(helloLambda.sayHello("Lambda"));}
从test2到test5代码一步步变的简介紧凑,与匿名类相比Lambda更加简洁。
函数式接口
函数式接口是只有一个抽象方法的接口,使用Lambda表达式的接口都是函数式接口。
Lambda表达式允许你直接以内联的形式为函数式接口的抽象方法提供实现,并把整个表达式作为函数式接口的实例。
函数式接口还带有一个注解@FunctionalInterface,这个注解帮助开发者判断接口是否是函数式接口,如果接口不是函数式接口,编译器将会返回一个提示错误。例如:Multiple non-overriding abstract methods found in interface com.straymalone.java8demo.Java8Lambda.Greeting,表明存在多个抽象方法。注意,@FunctionalInterface不是必须的,但是对于为此设计的接口而言,使用它是比较好的做法。它就像@Override注解表示方法被重写了。
类型推断
某些情况下,用户需要手动指明Lambda参数的类型,例如:
@Testpublic void test6() { BinaryOperator<Long> addLongs = (x, y) -> x + y; // 去掉<Long>会报错哦 // 调用方法 System.out.println(addLongs.apply(2L, 3L));}
这时将泛型<Long>去掉编译器将会报错,提示如下:
Operator '+' cannot be applied to 'java.lang.Object', 'java.lang.Object'
BinaryOperator 毕竟是一个具有泛型参数的函数接口, 该类型既是参数 x 和 y 的类型, 也是返回值的类型。 上面的例子中并没有给出变量add 的任何泛型信息, 给出的正是原始类型的定义。 因此, 编译器认为参数和返回值都是java.lang.Object 实例。
所以这个时候如果没有泛型就会编译报错。
Lambda表达式的重要特征
可选的类型声明:有时省略类型信息可以减少干扰,更易弄清状况;而有时却需要类型信息帮助理解代码
可选的参数圆括号:一个参数无需定义圆括号,但多个参数需要定义圆括号
可选的大括号:如果主题包含了一个语句,就不需要使用大括号
可选的返回关键字:如果主题只有一个表达式返回值则编译器会自动返回值,如果使用大括号说明不止一个语句,必须声明返回值。
可选的注解@FunctionalInterface:检测接口是否是函数式接口
参考资料
《Java 8函数式编程》
《Java 8实战》
java8(一)Lambda表达式的更多相关文章
- Java8中Lambda表达式的10个例子
Java8中Lambda表达式的10个例子 例1 用Lambda表达式实现Runnable接口 //Before Java 8: new Thread(new Runnable() { @Overri ...
- java8的lambda表达式,将List<DTO> 转为 List<DO>
将List<PhoneDTO>转为List<PhoneDO>,通过java8的lambda表达式来操作,比传统的for循环精简很多: /** * List<PhoneDT ...
- java8的lambda表达式
关于java8的lambda表达式 lambda表达式一般用于接口,因为lambda表达式是函数式编程. 1.有且仅有一个抽象方法被称为函数式接口,函数式接口可以显示的被@FunctionalInte ...
- java8中lambda表达式的应用,以及一些泛型相关
语法部分就不写了,我们直接抛出一个实际问题,看看java8的这些新特性究竟能给我们带来哪些便利 顺带用到一些泛型编程,一切都是为了简化代码 场景: 一个数据类,用于记录职工信息 public clas ...
- java8之lambda表达式(1)-基本语法
lambda表达式,即带有参数的表达式,为更清晰地理解lambda表达式,先看如下例子: (1) class Student{ private String name; private Double ...
- java8之lambda表达式入门
1.基本介绍 lambda表达式,即带有参数的表达式,为了更清晰地理解lambda表达式,先上代码: 1.1 两种方式的对比 1.1.1 方式1-匿名内部类 class Student{ privat ...
- JAVA8之lambda表达式具体解释,及stream中的lambda使用
前言: 本人也是学习lambda不久,可能有些地方描写叙述有误,还请大家谅解及指正! lambda表达式具体解释 一.问题 1.什么是lambda表达式? 2.lambda表达式用来干什么的? 3.l ...
- 十分钟学会Java8的lambda表达式和Stream API
01:前言一直在用JDK8 ,却从未用过Stream,为了对数组或集合进行一些排序.过滤或数据处理,只会写for循环或者foreach,这就是我曾经的一个写照. 刚开始写写是打基础,但写的多了,各种乏 ...
- java8中Lambda表达式和Stream API
一.Lambda表达式 1.语法格式 Lambda是匿名函数,可以传递代码.使用“->”操作符,改操作符将lambda分成两部分: 左侧:指定了 Lambda 表达式需要的所有参数 右侧:指定了 ...
- Java8之lambda表达式
一.什么是lambda表达式? Lambda 是一个匿名函数,我们可以把 Lambda 表达式理解为是一段可以传递的代码(将代码像数据一样进行传递).可以写出更简洁.更灵活的代码.作为一种更紧凑的代码 ...
随机推荐
- ASP.NET第一次访问慢的解决方法(MVC,Web Api)
问题现象 访问asp.net web项目的时候,第一次访问比较慢,当闲置一段时间后,再次访问还是会非常慢. 问题原因 这是IIS回收造成的,再次访问的时候会初始化操作,初始化需要耗费时间,所以访问会比 ...
- 关于自己"手机瘾"的切身体验
其实之前也喜欢看手机,只是从去年7月份开始,因为自控能力.新项目等原因,导致"手机瘾"这个病更严重了,控制不住,一看就是几小时.去年过年的时候,自己定的第一个目标就是戒掉手机瘾,到 ...
- JAVA超级简单的爬虫例子(1)
爬取整个页面的数据,并进行有效的提取信息,注释都有就不废话了: public class Reptile { public static void main(String[] args) { Stri ...
- .NET Core 3.0之深入源码理解Kestrel的集成与应用(二)
前言 前一篇文章主要介绍了.NET Core继承Kestrel的目的.运行方式以及相关的使用,接下来将进一步从源码角度探讨.NET Core 3.0中关于Kestrel的其他内容,该部分内容,我们 ...
- Java 7 源码学习系列(二)——Enum
Enum Enum类是java.lang包中一个类,他是Java语言中所有枚举类型的公共基类. 一.定义 public abstract class Enum<E extends Enum< ...
- mysql+mybatis存储超大json
1. 场景描述 因前端界面需存储元素较多,切割后再组装存储的话比较麻烦,就采用大对象直接存储到mysql字段中,根据mysql的介绍可以存放65535个字节,算了算差不多,后来存的时候发现: 一是基本 ...
- SpringCloud-Alibaba-Sentinel(1)初探
Sentinel 是什么? Sentinel 具有以下特征: 丰富的应用场景:Sentinel 承接了阿里巴巴近 10 年的双十一大促流量的核心场景,例如秒杀(即突发流量控制在系统容量可以承受的范围) ...
- CitusDB UPSERT
CitusDB的upsert功能 postgresql9.5 版本支持 "UPSERT" 特性, 这个特性支持 INSERT 语句定义 ON CONFLICT DO UPDATE/ ...
- stack用法,queue用法,
stack stack 模板类的定义在头文件中. stack 模板类需要两个模板参数,一个是元素类型,一个容器类型,但只有元素类型是必要 的,在不指定容器类型时,默认的容器类型为deque. 定义st ...
- [PTA] 数据结构与算法题目集 6-7 在一个数组中实现两个堆栈
//如果堆栈已满,Push函数必须输出"Stack Full"并且返回false:如果某堆栈是空的,则Pop函数必须输出"Stack Tag Empty"(其中 ...