Java8新特性

第二十二章 新特性

22.1 Java8新特性

22.1.1 Java8的概述

Java8是 Java 语言的一个重要版本,该版本于2014年3月发布,是自Java5以来最具革命性的版本,这个版本包含语言、编译器、库、工具和JVM等方面的十多个新特性。

22.1.2 函数式接口

函数式接口主要指只包含一个抽象方法的接口,如:java.lang.Runnable、java.util.Comparator接口等。

Java8提供@FunctionalInterface注解来定义函数式接口,若定义的接口不符合函数式的规范便会报错。

Java8中增加了java.util.function包,该包包含了常用的函数式接口,具体如下:

接口名称 方法声明 功能介绍
Consumer void accept(T t) 根据指定的参数执行操作
Supplier T get() 得到一个返回值
Function<T,R> R apply(T t) 根据指定的参数执行操作并返回
Predicate boolean test(T t) 判断指定的参数是否满足条件

22.1.3 Lambda表达式

Lambda 表达式是实例化函数式接口的重要方式,使用 Lambda 表达式可以使代码变的更加简洁紧凑。

lambda表达式:参数列表、箭头符号->和方法体组成,而方法体中可以是表达式,也可以是语句块。

语法格式:(参数列表) -> { 方法体; } - 其中()、参数类型、{} 以及return关键字 可以省略。

22.1.4 方法引用

方法引用主要指通过方法的名字来指向一个方法而不需要为方法引用提供方法体,该方法的调用交给函数式接口执行。

方法引用使用一对冒号 :: 将类或对象与方法名进行连接,通常使用方式如下:

  • 对象的非静态方法引用 ObjectName :: MethodName

  • 类的静态方法引用 ClassName :: StaticMethodName

  • 类的非静态方法引用 ClassName :: MethodName

  • 构造器的引用 ClassName :: new

  • 数组的引用 TypeName[] :: new

方法引用是在特定场景下lambda表达式的一种简化表示,可以进一步简化代码的编写使代码更加紧凑简洁,从而减少冗余代码。

22.1.5 Stream接口

(1)基本概念

java.util.stream.Stream接口是对集合功能的增强,可以对集合元素进行复杂的查找、过滤、筛选等操作。

Stream接口借助于Lambda 表达式极大的提高编程效率和程序可读性,同时它提供串行和并行两种模式进行汇聚操作,并发模式能够充分利用多核处理器的优势。

(2)使用步骤

  • 创建Stream,通过一个数据源来获取一个流。

  • 转换Stream,每次转换返回一个新的Stream对象。

  • 对Stream进行聚合操作并产生结果。

(3)创建方式

方式一:通过调用集合的默认方法来获取流,如:default Stream stream()

方式二:通过数组工具类中的静态方法来获取流,如:static IntStream stream(int[] array)

方式三:通过Stream接口的静态方法来获取流,如:static Stream of(T... values)

方式四:通过Stream接口的静态方法来获取流,static Stream generate(Supplier<? extends T> s)

(4)中间操作

筛选与切片的常用方法如下:

方法声明 功能介绍
Stream filter(Predicate<? super T> predicate) 返回一个包含匹配元素的流
Stream distinct() 返回不包含重复元素的流
Stream limit(long maxSize) 返回不超过给定元素数量的流
Stream skip(long n) 返回丢弃前n个元素后的流

映射的常用方法如下:

方法声明 功能介绍
Stream map(Function<? super T,? extends R> mapper) 返回每个处理过元素组成的流
Stream flatMap(Function<? super T,? extends Stream<? extends R>> mapper) 返回每个被替换过元素组成的流,并将所有流合成一个流

排序的常用方法如下:

方法声明 功能介绍
Stream sorted() 返回经过自然排序后元素组成的流
Stream sorted(Comparator<? super T> comparator) 返回经过比较器排序后元素组成的流

(5)终止操作

匹配与查找的常用方法如下:

方法声明 功能介绍
Optional findFirst() 返回该流的第一个元素
boolean allMatch(Predicate<? super T> predicate) 返回所有元素是否匹配
boolean noneMatch(Predicate<? super T> predicate) 返回没有元素是否匹配
Optional max(Comparator<? super T> comparator) 根据比较器返回最大元素
Optional min(Comparator<? super T> comparator) 根据比较器返回最小元素
long count() 返回元素的个数
void forEach(Consumer<? super T> action) 对流中每个元素执行操作

规约的常用方法如下:

方法声明 功能介绍
Optional reduce(BinaryOperator accumulator) 返回结合后的元素值

收集的常用方法如下:

方法声明 功能介绍
<R,A> R collect(Collector<? super T,A,R> collector) 使用收集器对元素进行处理

22.1.6 Option类

  • 基本概念

    • java.util.Optional类可以理解为一个简单的容器,其值可能是null或者不是null,代表一个值存在或不存在。

    • <font color=#00ffff size=72>该类的引入很好的解决空指针异常,不用显式进行空值检测。</font>

  • 常用方法

方法声明 功能介绍
static Optional ofNullable(T value) 根据参数指定数值来得到Optional类型的对象
Optional map(Function<? super T,? extends U> mapper) 根据参数指定规则的结果来得到Optional类型的对象
T orElse(T other) 若该值存在就返回,否则返回other的数值。
  • 代码demo
System.out.println("-------------------常规的判断字符串是否为空的方式--------------");
String str = null;
if (null != str) {
    System.out.println("字符串长度为:" + str.length());
} else {
    System.out.println("字符串长度为:" + 0);
}

System.out.println("--------------------使用Optional类容器进行判断-----------------");
/**
 * 1. 将str撞到Optional对象中去
 */
Optional<String> stringOptional = Optional.ofNullable(str);

/**
 * 2. 建立映射关系 使用字符串的长度和字符串建立映射关系
 */
Optional<Integer> length = stringOptional.map(new Function<String, Integer>() {
    @Override
    public Integer apply(String s) {
        return s.length();
    }
});
System.out.println("length的长度为:" + length);

/**
 * 3. 使用lambda表达式
 */
Optional<Integer> length2 = stringOptional.map(s -> s.length());
System.out.println("length2的长度为:" + length2);

Optional<Integer> length3 = stringOptional.map(String::length);
System.out.println("length3的长度为:" + length3);

/**
 * 4. 若该值存在就返回,否则返回other的数值(此处other为4)
 *
 */
Integer integer = length3.orElse(4);
System.out.println(integer);

22.2 Java9的新特性

22.2.1 Java9的概述

  • Java9发布于2017年9月发布,带来了很多新特性,其中最主要的变化是模块化系统。
  • 模块就是代码和数据的封装体,模块的代码被组织成多个包,每个包中包含Java类和接口,模块的
    数据则包括资源文件和其他静态信息。

22.2.2 模块化的使用

(1)语法格式

在 module-info.java 文件中,我们可以用新的关键词module来声明一个模块,具体如下:
module 模块名称 {
}

(2)模块化的优势

  • 减少内存的开销。
  • 可简化各种类库和大型应用的 开发和维护。
  • 安全性,可维护性,提高性能。

22.2.3 钻石操作符的使用升级

在Java9中允许在匿名内部类的使用中使用钻石操作符。

// 实现匿名内部类和钻石操作符的搭配使用
//Comparator<Integer> comparator = new Comparator<Integer>() {
Comparator<Integer> comparator = new Comparator<>() {
    @Override
    public int compare(Integer o1, Integer o2) {
        return 0;
    }
};

22.2.4 集合工厂方法

(1)基本概念

  • Java9的List、Set和Map集合中增加了静态工厂方法of实现不可变实例的创建。
  • 不可变体现在无法添加、修改和删除它们的元素。
  • 不允许添加null元素对象。

(2)实际意义

  • 保证线程安全:在并发程序中既保证线程安全性,也大大增强了并发时的效率。
  • 被不可信的类库使用时会很安全。
  • 如果一个对象不需要支持修改操作,将会节省空间和时间的开销。
  • 可以当作一个常量来对待,并且这个对象在以后也不会被改变。

22.2.5 InputStream的增强

InputStream类中提供了transferTo方法实现将数据直接传输到OutputStream中。

本质:底层还是通过创建一个byte[]数组缓冲区进行不断的读写。

22.3 Java10的新特性

22.3.1 Java10的概述

  • Java10于2018年3月发布,改进的关键点包括一个本地类型推断、一个垃圾回收的增强。

  • Java10计划只是一个短期版本,因此公开更新将在六个月内结束,9月份发布的Java11将是Java的长期支持(LTS)版本,LTS版本的发布每三年发布一次。

22.3.2 局部变量类型推断

(1)基本概念

  • Java10可以使用var作为局部变量类型推断标识符,此符号仅适用于局部变量,增强for循环的索引,以及传统for循环的本地变量。

  • 它不能使用于方法形式参数,构造函数形式参数,方法返回类型,字段,catch形式参数或任何其他类型的变量声明。

  • 例如

// 由初始值可以推断出变量的类型,因此可以使用var取代
//int num = 10;
var num = 10;

//List<Integer> list = new LinkedList<>();
var list = new LinkedList<Integer>();
list.add(10);

for (var v : list) {
    System.out.println(v);
}

for (var i = 0; i < 10; i++) {}

类似与JavaScript中的var,只不过这里是Java编译器进行类型的推断

(2)实际意义

  • 标识符var不是关键字,只是一个保留的类型名称。这意味着var用作变量,方法名或包名的代码不会受到影响,但var不能作为类或则接口的名字。避免了信息冗余。

  • 对齐了变量名。

  • 更容易阅读。

22.4 Java11的新特性

22.4.1 Java11的概述

Java11于2018年9月正式发布,这是 Java 大版本周期变化 后的第一个长期支持版本,非常值得关注。

22.4.2 简化的编译运行操作

  • 在Java11中可以使用java命令一次性进行编译和运行操作。

  • 执行源文件中的第一个类必须包含主方法。

  • 不可以使用其它源文件中自定义的类。

方法声明 功能介绍
boolean isBlank() 判断字符串是否为空或只包含空白代码点
Optional map(Function<? super T,? extends U> mapper) 根据参数指定规则的结果来得到Optional类型的对象
T orElse(T other) 若该值存在就返回,否则返回other的数值。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,423评论 6 491
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,147评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,019评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,443评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,535评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,798评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,941评论 3 407
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,704评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,152评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,494评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,629评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,295评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,901评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,742评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,978评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,333评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,499评论 2 348

推荐阅读更多精彩内容