Java8初体验(一)lambda表达式语法
感谢同事【天锦】的投稿。投稿请联系 tengfei@ifeve.com
本文主要记录自己学习Java8的历程,方便大家一起探讨和自己的备忘。因为本人也是刚刚开始学习Java8,所以文中肯定有错误和理解偏差的地方,希望大家帮忙指出,我会持续修改和优化。本文是该系列的第一篇,主要介绍Java8对屌丝码农最有吸引力的一个特性—lambda表达式。
java8的安装
工欲善其器必先利其器,首先安装JDK8。过程省略,大家应该都可以自己搞定。但是有一点这里强调一下(Windows系统):目前我们工作的版本一般是java 6或者java 7,所以很多人安装java8基本都是学习为主。这样就在自己的机器上会存在多版本的JDK。而且大家一般是希望在命令行中执行java命令是基于老版本的jdk。但是在安装完jdk8并且没有设置path的情况下,你如果在命令行中输入:java -version,屏幕上会显示是jdk 8。这是因为jdk8安装的时候,会默认在C:/Windows/System32中增加java.exe,这个调用的优先级比path设置要高。所以即使path里指定是老版本的jdk,但是执行java命令显示的依然是新版本的jdk。这里我们要做的就是删除C:/Windows/System32中的java.exe文件(不要手抖!)。
Lambda初体验
下面进入本文的正题–lambda表达式。首先我们看一下什么是lambda表达式。以下是维基百科上对于”Lambda expression”的解释:
a function (or a subroutine) defined, and possibly called, without being bound to an identifier。
简单点说就是:一个不用被绑定到一个标识符上,并且可能被调用的函数。这个解释还不够通俗,lambda表达式可以这样定义(不精确,自己的理解):一段带有输入参数的可执行语句块。这样就比较好理解了吧?一例胜千言。有读者反馈:不理解Stream的含义,所以这里先提供一个没用stream的lambda表达式的例子。
1
|
//这里省略list的构造
|
2
|
List<String> names = ...;
|
3
|
Collections.sort(names, (o1, o2) -> o1.compareTo(o2));
|
1
|
//这里省略list的构造
|
2
|
List<String> names = ...;
|
3
|
Collections.sort(names, new Comparator<String>() {
|
4
|
@Override
|
5
|
public int compare(String o1, String o2) {
|
6
|
return o1.compareTo(o2);
|
7
|
}
|
8
|
});
|
上面两段代码分别是:使用lambda表达式来排序和使用匿名内部类来排序。这个例子可以很明显的看出lambda表达式简化代码的效果。接下来展示lambda表达式和其好基友Stream的配合。
1
|
List<String> names = new ArrayList<>();
|
2
|
names.add( "TaoBao" );
|
3
|
names.add( "ZhiFuBao" );
|
4
|
List<String> lowercaseNames = names.stream().map((String name) -> { return name.toLowerCase();}).collect(Collectors.toList());
|
这段代码就是对一个字符串的列表,把其中包含的每个字符串都转换成全小写的字符串(熟悉Groovy和Scala的同学肯定会感觉很亲切)。注意代码第四行的map方法调用,这里map方法就是接受了一个lambda表达式(其实是一个java.util.function.Function的实例,后面会介绍)。
为什么需要Lambda表达式呢?在尝试回答这个问题之前,我们先看看在Java8之前,如果我们想做上面代码的操作应该怎么办。
先看看普通青年的代码:
1
|
List<String> names = new ArrayList<>();
|
2
|
names.add( "TaoBao" );
|
3
|
names.add( "ZhiFuBao" );
|
4
|
List<String> lowercaseNames = new ArrayList<>();
|
5
|
for (String name : names) {
|
6
|
lowercaseNames.add(name.toLowerCase());
|
7
|
}
|
接下来看看文艺青年的代码(借助Guava):
1
|
List<String> names = new ArrayList<>();
|
2
|
names.add( "TaoBao" );
|
3
|
names.add( "ZhiFuBao" );
|
4
|
List<String> lowercaseNames = FluentIterable.from(names).transform( new Function<String, String>() {
|
5
|
@Override
|
6
|
public String apply(String name) {
|
7
|
return name.toLowerCase();
|
8
|
}
|
9
|
}).toList();
|
在此,我们不再讨论普通青年和文艺青年的代码风格孰优孰劣(有兴趣的可以去google搜索“命令式编程vs声明式编程”)。本人更加喜欢声明式的编程风格,所以偏好文艺青年的写法。但是在文艺青年代码初看起来看起来干扰信息有点多,Function匿名类的构造语法稍稍有点冗长。所以Java8的lambda表达式给我们提供了创建SAM(Single Abstract Method)接口更加简单的语法糖。
Lambda语法详解
我们在此抽象一下lambda表达式的一般语法:
1
|
(Type1 param1, Type2 param2, ..., TypeN paramN) -> {
|
2
|
statment1;
|
3
|
statment2;
|
4
|
//.............
|
5
|
return statmentM;
|
6
|
}
|
从lambda表达式的一般语法可以看出来,还是挺符合上面给出的非精确版本的定义–“一段带有输入参数的可执行语句块”。
上面的lambda表达式语法可以认为是最全的版本,写起来还是稍稍有些繁琐。别着急,下面陆续介绍一下lambda表达式的各种简化版:
1. 参数类型省略–绝大多数情况,编译器都可以从上下文环境中推断出lambda表达式的参数类型。这样lambda表达式就变成了:
1
|
(param1,param2, ..., paramN) -> {
|
2
|
statment1;
|
3
|
statment2;
|
4
|
//.............
|
5
|
return statmentM;
|
6
|
}
|
所以我们最开始的例子就变成了(省略了List的创建):
1
|
List<String> lowercaseNames = names.stream().map((name) -> { return name.toLowerCase();}).collect(Collectors.toList());
|
2. 当lambda表达式的参数个数只有一个,可以省略小括号。lambda表达式简写为:
1
|
param1 -> {
|
2
|
statment1;
|
3
|
statment2;
|
4
|
//.............
|
5
|
return statmentM;
|
6
|
}
|
所以最开始的例子再次简化为:
1
|
List<String> lowercaseNames = names.stream().map(name -> { return name.toLowerCase();}).collect(Collectors.toList());
|
3. 当lambda表达式只包含一条语句时,可以省略大括号、return和语句结尾的分号。lambda表达式简化为:
1
|
param1 -> statment
|
所以最开始的例子再次简化为:
1
|
List<String> lowercaseNames = names.stream().map(name -> name.toLowerCase()).collect(Collectors.toList());
|
4. 使用Method Reference(具体语法后面介绍)
1
|
//注意,这段代码在Idea 13.0.2中显示有错误,但是可以正常运行
|
2
|
List<String> lowercaseNames = names.stream().map(String::toLowerCase).collect(Collectors.toList());
|
Lambda表达式眼中的外部世界
我们前面所有的介绍,感觉上lambda表达式像一个闭关锁国的家伙,可以访问给它传递的参数,也能自己内部定义变量。但是却从来没看到其访问它外部的变量。是不是lambda表达式不能访问其外部变量?我们可以这样想:lambda表达式其实是快速创建SAM接口的语法糖,原先的SAM接口都可以访问接口外部变量,lambda表达式肯定也是可以(不但可以,在java8中还做了一个小小的升级,后面会介绍)。
1
|
String[] array = { "a" , "b" , "c" };
|
2
|
for (Integer i : Lists.newArrayList( 1 , 2 , 3 )){
|
3
|
Stream.of(array).map(item -> Strings.padEnd(item, i, '@' )).forEach(System.out::println);
|
4
|
}
|
上面的这个例子中,map中的lambda表达式访问外部变量Integer i。并且可以访问外部变量是lambda表达式的一个重要特性,这样我们可以看出来lambda表达式的三个重要组成部分:
- 输入参数
- 可执行语句
- 存放外部变量的空间
不过lambda表达式访问外部变量有一个非常重要的限制:变量不可变(只是引用不可变,而不是真正的不可变)。
1
|
String[] array = { "a" , "b" , "c" };
|
2
|
for ( int i = 1 ; i< 4 ; i++){
|
3
|
Stream.of(array).map(item -> Strings.padEnd(item, i, '@' )).forEach(System.out::println);
|
4
|
}
|
上面的代码,会报编译错误。因为变量i被lambda表达式引用,所以编译器会隐式的把其当成final来处理(ps:大家可以想象问什么上一个例子不报错,而这个报错。)细心的读者肯定会发现不对啊,以前java的匿名内部类在访问外部变量的时候,外部变量必须用final修饰。Bingo,在java8对这个限制做了优化(前面说的小小优化),可以不用显示使用final修饰,但是编译器隐式当成final来处理。
lambda眼中的this
在lambda中,this不是指向lambda表达式产生的那个SAM对象,而是声明它的外部对象。
方法引用(Method reference)和构造器引用(construct reference)
方法引用
前面介绍lambda表达式简化的时候,已经看过方法引用的身影了。方法引用可以在某些条件成立的情况下,更加简化lambda表达式的声明。方法引用语法格式有以下三种:
- objectName::instanceMethod
- ClassName::staticMethod
- ClassName::instanceMethod
前两种方式类似,等同于把lambda表达式的参数直接当成instanceMethod|staticMethod的参数来调用。比如System.out::println等同于x->System.out.println(x);Math::max等同于(x, y)->Math.max(x,y)。
最后一种方式,等同于把lambda表达式的第一个参数当成instanceMethod的目标对象,其他剩余参数当成该方法的参数。比如String::toLowerCase等同于x->x.toLowerCase()。
构造器引用
构造器引用语法如下:ClassName::new,把lambda表达式的参数当成ClassName构造器的参数 。例如BigDecimal::new等同于x->new BigDecimal(x)。
吐槽一下方法引用
表面上看起来方法引用和构造器引用进一步简化了lambda表达式的书写,但是个人觉得这方面没有Scala的下划线语法更加通用。比较才能看出,翠花,上代码!
1
|
List<String> names = new ArrayList<>();
|
2
|
names.add( "TaoBao" );
|
3
|
names.add( "ZhiFuBao" );
|
4
|
names.stream().map(name -> name.charAt( 0 )).collect(Collectors.toList());
|
上面的这段代码就是给定一个String类型的List,获取每个String的首字母,并将其组合成新的List。这段代码就没办法使用方法引用来简化。接下来,我们简单对比一下Scala的下划线语法(不必太纠结Scala的语法,这里只是做个对比):
1
|
//省略List的初始化
|
2
|
List[String] names = ....
|
3
|
names.map(_.charAt( 0 ))
|
在Scala中基本不用写lambda表达式的参数声明。
下期预告
Java8初体(二)Sream语法详解
引用文档
- 《Java SE 8 for the Really Impatient》
- Java 8 Tutorial
- Java 8 API doc
Java8初体验(一)lambda表达式语法相关推荐
- Java8初体验(二)Stream语法详解(转)
本文转自http://ifeve.com/stream/ Java8初体验(二)Stream语法详解 感谢同事[天锦]的投稿.投稿请联系 tengfei@ifeve.com 上篇文章Java8初体验( ...
- java8 lambda python_【学习笔记】java8 Lambda表达式语法及应用
本文是慕课网大牧莫邪老师的视频教程一课掌握Lambda表达式语法及应用的学习笔记.如果觉得内容对你有用,可以购买老师的课程支持一下,课程价格1元,十分良心了. 1. 课程介绍 2. 为什么引入Lamb ...
- 【java8新特性】——lambda表达式与函数式接口详解(一)
一.简介 java8于2014年发布,相比于java7,java8新增了非常多的特性,如lambda表达式.函数式接口.方法引用.默认方法.新工具(编译工具).Stream API.Date Time ...
- 夯实Java基础(二十二)——Java8新特性之Lambda表达式
1.前言 Java 8于14年发布到现在已经有5年时间了,经过时间的磨练,毫无疑问,Java 8是继Java 5(发布于2004年)之后的又一个非常最重要的版本.因为Java 8里面出现了非常多新的特 ...
- Java8函数式接口与Lambda表达式
摘要 何为函数式接口? 什么是lambda表达式,lambda表达式的本质: 函数式接口与lambda表达式的联系:lambda是实现函数式接口的一个快捷方式,可以作为函数式接口的一个实例: 常用Ja ...
- java8新特性之lambda表达式--超级详细版本
java8新特性之lambda表达式 1.什么是lambda表达式?为什么用它? 2. 为什么Java需要lambda表达式? 3. lambda表达式的语法 4.函数式接口 4.1 什么是函数式接口 ...
- java8新特性总结——lambda表达式
最近看尚硅谷java8新特性视屏,总结一下学习知识. Lambda表达式:是一个匿名函数,我们可以把Lambda理解为一段可以传递的代码(将代码像数据一样传递),可以写出更简洁更灵活的代码.作为一种更 ...
- Java8 新特性:Lambda 表达式、方法和构造器引用、Stream API、新时间与日期API、注解
Java8新特性:Lambda 表达式.方法和构造器引用.Stream API.新时间与日期API.注解 1.Java8新特性 1.1.主要的新特性: 1.2.编程风格 2.Lambda 表达式 2. ...
- Java新特性:Java8函数式接口与Lambda表达式(一)
摘要 何为函数式接口? 什么是lambda表达式,lambda表达式的本质: 函数式接口与lambda表达式的联系:lambda是实现函数式接口的一个快捷方式,可以作为函数式接口的一个实例: 常用Ja ...
- java8新特性: lambda表达式:直接获得某个list/array/对象里面的字段集合
java8新特性: lambda表达式:直接获得某个list/array/对象里面的字段集合 比如,我有一张表: entity Category.java service CategoryServic ...
最新文章
- 关于按键消抖以及LED灯控制的一个实例
- Atitit.异步编程 java .net php python js 对照
- Python 抖音用户粉丝(公开)--分析与实现
- 反射中使用 BindingFlags.IgnoreCase
- 【直播预约】线上Greentea JUG Meetup
- 精品软件 推荐 360 安全卫士
- PWN-PRACTICE-BUUCTF-22
- 神仙情侣:一起读研读博,然后结婚……
- c语言 链表首部插入数据,在链表中插入数据!求助!!!
- java manualbuffer_java从很大的buffer数组中每次读128字节 读到最后不足128字节的用0xff补齐 这个代码怎么写 ?没有思路哎...
- fat jar包缺点_什么是fat JAR?
- CentOS 关闭蜂鸣器声音
- mysql 15位升18位_身份证号码15位升18位规则
- samtools提取指定位置核酸或氨基酸序列
- http301重定向IIS 301重定向域名转向新旧域名更替
- python编程实现人民币和美元的互相转换_java人民币转换美元的实验报告
- 【大数据】SQL的作业题(仅适合初学者)
- Android 8.1/GO GMS认证 (GMS认证群:738320435)
- 读书笔记:不可能的技艺,巅峰人生需要凶猛的起点
- LVGL+RTTHREAD KEIL下出现.\build\debug.axf: Error: L6218E: Undefined symbol __aeabi_assert怎么办
热门文章
- 技术改变世界以及减速慢行
- 别怕,“卷积“其实很简单
- linux下输入法安装设置及中文字体安装
- pandas.DataFrame对根据某些列对某些列进行修改
- html jQuery 光标位置,jquery实现在光标位置插入内容的方法
- 私有5g网络_面向企业的私有5G网络
- 计算机考试数据库相关知识点,计算机等级考试四级数据库工程师知识点总结
- Android ADB 你想要的都在这
- Android 手势放大图片
- Futter基础第15篇: 实现类似闲鱼App底部导航凸起按钮