1、新特性
- Lambda表达式
- 函数式接口
- 方法引用与构造器引用
- Stream API
- 接口的默认方法与静态方法
- 新时间日期API
- 其他新特性
2、优点
- 速度更快
- 代码更少(增加了新的语法Lambda表达式)
- 强大的Stream API
- 便于并行
- 最大化减少空指针异常Optional
3、Lambda表达式
Lambda表达式是一个匿名函数,我们可以这样理解Lambda表达式:Lambda是一段可以传递的代码(能够做到将代码像数据一样进行传递)。使用Lambda表达式能够写出更加简洁、灵活的代码。并且,使用Lambda表达式能够使Java的语言表达能力得到提升。 匿名内部类:使用匿名内部类实现了比较两个Integer类型数据的大小。 上述代码,在整个匿名内部类中,实际上真正有用的就是下面一行代码。
return Integer.compare(o1, o2);
其他的代码本质上都是“冗余”的。使用Lambda表达式:
Comparator<Integer> com = (x, y) -> Integer.compare(x, y);
Lambda表达式需要函数式接口的支持,什么是函数式接口?
只包含一个抽象方法的接口,称为函数式接口。 可以通过 Lambda 表达式来创建该接口的对象。(若 Lambda表达式抛出一个受检异常,那么该异常需要在目标接口的抽象方法上进行声明)。 可以在任意函数式接口上使用 @FunctionalInterface 注解,这样做可以检查它是否是一个函数式接口,同时 javadoc 也会包含一条声明,说明这个接口是一个函数式接口。 我们可以自定义函数式接口,并使用Lambda表达式来实现相应的功能。
4、java7和8知识点总结
1) Java7与Java8中的HashMap
JDK7 HashMap结构为数组+链表(发生元素碰撞时,会将新元素添加到链表开头) JDK8 HashMap结构为数组+链表+红黑树(发生元素碰撞时,会将新元素添加到链表末尾,当HashMap总容量大于等于64,并且某个链表的大小大于等于8,会将链表转化为红黑树(注意:红黑树是二叉树的一种))
2) JDK8 HashMap重排序
如果删除了HashMap中红黑树的某个元素导致元素重排序时,不需要计算待重排序的元素的HashCode码,只需要将当前元素放到(HashMap总长度+当前元素在HashMap中的位置)的位置即可。
3) 筛选与切片
- filter——接收 Lambda , 从流中排除某些元素。
- limit——截断流,使其元素不超过给定数量。
- skip(n) —— 跳过元素,返回一个扔掉了前 n 个元素的流。若流中元素不足 n 个,则返回一个空流。与 limit(n) 互补
- distinct——筛选,通过流所生成元素的 hashCode() 和 equals() 去除重复元素
4) 中间操作
- map——接收 Lambda , 将元素转换成其他形式或提取信息。接收一个函数作为参数,该函数会被应用到每个元素上,并将其映射成一个新的元素。
- flatMap——接收一个函数作为参数,将流中的每个值都换成另一个流,然后把所有流连接成一个流
- sorted()——自然排序
- sorted(Comparator com)——定制排序
5)终止操作(查找与匹配)
- allMatch——检查是否匹配所有元素【只有所有的元素都匹配条件时,allMatch()方法才会返回true】
- anyMatch——检查是否至少匹配一个元素【只要有任意一个元素符合条件,anyMatch()方法就会返回true】
- noneMatch——检查是否没有匹配的元素【只有所有的元素都不符合条件时,noneMatch()方法才会返回true】
- findFirst——返回第一个元素【fifindFirst()方法表示返回第一个元素】
- findAny——返回当前流中的任意元素【返回当前流中的任意元素】
- count——返回流中元素的总个数【返回流中元素总数】
- max——返回流中最大值【表示返回流中最大值】
- min——返回流中最小值【返回流中最小值】
6) 归约
- reduce(T identity, BinaryOperator) / reduce(BinaryOperator) ——可以将流中元素反复结合起来,得到一个值。
- collect——将流转换为其他形式。接收一个 Collector接口的实现,用于给Stream中元素做汇总的方法
注意:流进行了终止操作后,不能再次使用
7) Optional 容器类
用于尽量避免空指针异常
- Optional.of(T t) : 创建一个 Optional 实例
- Optional.empty() : 创建一个空的 Optional 实例
- Optional.ofNullable(T t):若 t 不为 null,创建 Optional 实例,否则创建空实例
- isPresent() : 判断是否包含值
- orElse(T t) : 如果调用对象包含值,返回该值,否则返回t
- orElseGet(Supplier s) :如果调用对象包含值,返回该值,否则返回 s 获取的值
- map(Function f): 如果有值对其处理,并返回处理后的Optional,否则返回 Optional.empty()
- flatMap(Function mapper):与 map 类似,要求返回值必须是Optional
8) 方法引用
当要传递给Lambda体的操作,已经有实现的方法了,可以使用方法引用!这里需要注意的是:实现抽象方法的参数列表,必须与方法引用方法的参数列表保持一致! 那么什么是方法引用呢?方法引用就是操作符“::”将方法名和对象或类的名字分隔开来。 有如下三种使用情况:
- 对象::实例方法
- 类::静态方法
- 类::实例方法
例如: compare((x, y) -> x.equals(y), “binghe”, “binghe”) 等同于 compare(String::equals, “binghe”, “binghe”) 注意: 当需要引用方法的第一个参数是调用对象,并且第二个参数是需要引用方法的第二个参数(或无参数)时: ClassName::methodName 。
5、Stream流
Java8中有两大最为重要的改变。第一个是 Lambda 表达式;另外一个则是 Stream API(java.util.stream.*)。 Stream 是 Java8 中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找、过滤和映射数据等操作。使用Stream API 对集合数据进行操作,就类似于使用 SQL 执行的数据库查询。也可以使用 Stream API 来并行执行操作。简而言之,Stream API 提供了一种高效且易于使用的处理数据的方式。
6、Optional类
Optional 类(java.util.Optional) 是一个容器类,代表一个值存在或不存在,原来用 null 表示一个值不存在,现在 Optional 可以更好的表达这个概念。并且可以避免空指针异常。 Optional类常用方法:
- Optional.of(T t) : 创建一个 Optional 实例。
- Optional.empty() : 创建一个空的 Optional 实例。
- Optional.ofNullable(T t):若 t 不为 null,创建 Optional 实例,否则创建空实例。
- isPresent() : 判断是否包含值。
- orElse(T t) : 如果调用对象包含值,返回该值,否则返回t。
- orElseGet(Supplier s) :如果调用对象包含值,返回该值,否则返回 s 获取的值。
- map(Function f): 如果有值对其处理,并返回处理后的Optional,否则返回
- Optional.empty()。
- flflatMap(Function mapper):与 map 类似,要求返回值必须是Optional。
7、本地时间和时间戳
- now:静态方法,根据当前时间创建对象
- of:静态方法,根据指定日期/时间创建对象
- plusDays,plusWeeks,plusMonths,plusYears:向当前LocalDate 对象添加几天、几周、几个月、几年
- minusDays,minusWeeks,minusMonths,minusYears:从当前LocalDate 对象减去几天、几周、几个月、几年
- plus,minus:添加或减少一个Duration 或Period
- withDayOfMonth,withDayOfYear,withMonth,withYear:将月份天数、年份天数、月份、年份修改为指定的值并返回新的LocalDate 对象
- getDayOfYear:获得年份天数(1~366)
- getDayOfWeek:获得星期几(返回一个DayOfWeek枚举值)
- getMonth:获得月份, 返回一个Month 枚举值
- getMonthValue:获得月份(1~12)
- getYear:获得年份
- until:获得两个日期之间的Period 对象,或者指定ChronoUnits 的数字
- isBefore,isAfter:比较两个LocalDate
- isLeapYear:判断是否是闰年
|