(JDK1.8New)Java8版本总结

Java 8 新特性

Java 8 (又称为 jdk 1.8) 是 Java 语言开发的一个主要版本。 Oracle 公司于 2014 年 3 月 18 日发布 Java 8 ,它支持函数式编程,新的 JavaScript 引擎,新的日期 API,新的Stream API 等。


新特性总结

Java8 新增了非常多的特性,我们主要讨论以下几个:

  • Lambda 表达式 − Lambda允许把函数作为一个方法的参数(函数作为参数传递进方法中。

  • 方法引用 − 方法引用提供了非常有用的语法,可以直接引用已有Java类或对象(实例)的方法或构造器。与lambda联合使用,方法引用可以使语言的构造更紧凑简洁,减少冗余代码。

  • 默认方法 − 默认方法就是一个在接口里面有了一个实现的方法。

  • 新工具 − 新的编译工具,如:Nashorn引擎 jjs、 类依赖分析器jdeps。

  • Stream API −新添加的Stream API(java.util.stream) 把真正的函数式编程风格引入到Java中。

  • Date Time API − 加强对日期与时间的处理。

  • Optional 类 − Optional 类已经成为 Java 8 类库的一部分,用来解决空指针异常。

  • Nashorn, JavaScript 引擎 − Java 8提供了一个新的Nashorn javascript引擎,它允许我们在JVM上运行特定的javascript应用。

  • Map集合的优化及并发包的拓展

更多的新特性可以参阅官网:What's New in JDK 8

在关于 Java 8 文章的实例,我们均使用 jdk 1.8 环境,你可以使用以下命令查看当前 jdk 的版本:

$ java -version
java version "1.8.0_31"
Java(TM) SE Runtime Environment (build 1.8.0_31-b13)
Java HotSpot(TM) 64-Bit Server VM (build 25.31-b07, mixed mode)

Java8新增的功能中,要数lambda表达式和流API最为重要了.这篇文章主要介绍流API的基础,了解简单的集合聚合操作,也是Java8系列的第一篇文章,(后续会有其他文章更新)话不多说,直奔主题.

什么是流API? 它能做一些什么?

我们应该知道(绝对知道~)API是一个程序向使用者提供的一些方法,通过这些方法就能实现某些功能.所以对于流API来说,重点是怎么理解"流"这个概念,所谓的流:就是数据的渠道,所以,流代表的是一个对象的序列.它和Java I/O类里使用的"流"不同.虽然在概念上与java.util.stream中定义的流是类似的,但它们是不同的.流API中的流是描述某个流类型的对象.

流API中的流操作的数据源,是数组或者是集合.它本身是不存储数据的,只是移动数据,在移动过程中可能会对数据进行过滤,排序或者其它操作.但是,一般情况下(绝大数情况下),流操作本身不会修改数据源.比如,对流排序不会修改数据源的顺序.相反,它会创建一个新的流,其中包含排序后的结果.

从一个简单的例子,体验流API的强大与优雅

这个简单的Demo,主要是对一个由1-6乱序组成的List对应的流进行操作,然后通过这个流,就可以获取到列表里面最大最小值,排序,过滤某些元素等等的操作.并且这此操作不会改变原List里面的数据.Demo里面需要注意的地方就是流API里面的"终端操作"和"中间操作"的区别:其实也很简单,终端操作会消费流,一个被消费过的流是不能被再次利用的,但我们在实际应用的时候,并不会受到太大的影响(Ps:如果你们能动手实践一下我相信你体验更好,强烈推荐!)

public class Main {

    
public static void main(String[] args) {
        learnStream();
 }

private static void learnStream() {
        
//首先,创建一个1-6乱序的List
        
List<Integer> lists = new
 
ArrayList<>();
        lists.add(4);
        lists.add(3);
        lists.add(6);
        lists.add(1);
        lists.add(5);
        lists.add(2);

        
//看看List里面的数据是什么样子的先
        
System.out.print("List里面的数据:");
        
for (Integer elem : lists) {
System.out.print(elem + " ");
// 4 3 6 1 5 2
}
        
System.out.println();

        
//最小值
        
System.out.print("List中最小的值为:");
        
Stream<Integer> stream = lists.stream();
        
Optional<Integer> min = stream.min(Integer::compareTo);
        
if(min.isPresent()) {         
    System.out.println(min.get());
//1
}


        
//最大值
        
System.out.print("List中最大的值为:");
lists.stream().max(Integer::compareTo).ifPresent(
    System.out::println);
//6

        
//排序
        
System.out.print("将List流进行排序:");
        
Stream<Integer> sorted = lists.stream().sorted();
        
sorted.forEach(elem -> System.out.print(elem + " "));
// 1 2 3 4 5 6

        
System.out.println();

        
//过滤
System.out.print("过滤List流,只剩下那些大于3的元素:");
lists.stream()
            .filter(elem -> elem > 3)
            .forEach(elem -> System.out.print(elem + " "));
// 4 5 6

        
System.out.println();

        
//过滤
        
System.out.println("过滤List流,只剩下那些大于0并且小于4的元素:\n=====begin=====");
        
lists.stream()
               .filter(elem -> elem > 0)
               .filter(elem -> elem < 4)
               .sorted(Integer::compareTo)
               .forEach(System.out::println);
// 1 2 3

        
System.out.println("=====end=====");
        
//经过了前面的这么多流操作,我们再来看看List里面的值有没有发生什么改变
        
System.out.print("原List里面的数据:");
        
for (Integer elem : lists){
 System.out.print(elem + " ");
// 4 3 6 1 5 2
    
}  

如果刚才的Demo你认真读了,我相信你心里面多多少少都会产生一点点涟漪.没错,流API结合lambda表达式,就是这么优美!下面我详细介绍一下整个Demo,让大家更加清淅:

最小值

 
//最小值
System.out.print("List中最小的值为:");
Stream<Integer> stream = lists.stream();
Optional<Integer> min = stream.min(Integer::compareTo);
if(min.isPresent()) {
    System.out.println(min.get());//1
}

首先通过stream()方法获取List对应的流,如果你对Java8的集合框架有一定的了解,你应该知道stream()是由Collection接口提供的.然后就可以通过min()获取流中的最小值了,当然这个流中的最小值肯定也是List里面的最小值.

min()方法接收一个Comparator类型的参数,这个比较器是用于比较流中的两个元素的.我们这里把Integer的compareTo()的引用传递给了min().它返回的类型是Optional,Optional可谓是NullPointException的大杀器啊,感兴趣的同学,了解一下.然后判断最小值存不存在,如果存在,就通过Optional的get()读取出来.很简单有木有!

最大值

//最大值
System.out.print("List中最大的值为:");
lists.stream().max(Integer::compareTo).ifPresent(System.out::println);
//6

语法糖爽YY有木有,不过需要注意的一点,因为min()是一个终端操作,所以这个流是不可以再用了,因此我们需要通过stream()重新生成一个流,(但这其实并不影响我们的实际生产的:①方法功能单一原则②还有其它很多很强大的方法组合能让你实现各种功能啊.)ifPresent其实和上面的最小值的if判断是一定要,如果存在最大值,我们就打印一下,这里只不过用了一些函数式写法而已.

排序和遍历

 
//排序
System.out.print("将List流进行排序:");
Stream<Integer> sorted = lists.stream().sorted();
sorted.forEach(elem -> 
System.out.print(elem + " "));
// 1 2 3 4 5 6

通过上面的讲解,相信这个已经难不了你了,sorted()方法是用于排序的,它的一个重载方法可以接收一个Comparator类型的参数,让你自定义你的排序规则.forEach方法就遍历.

过滤

filter()是基于一个谓词过滤流,它返回一个只包含满足谓词的元素的新流.它的参数形式是Predicate,是在java.util.function包下的泛型函数式接口.并且filter是一个中间操作,而且还可以同时存在多个filter.这里的两个过滤器,我们都传递了lambda表达式.

小结一下

其实基本的流API使用就是这么简单,结合lambda表达式后,一切都变得特别清淅.这个简单的Demo展示了一些基础的功能,它或许就扩展了你操作数组或者集合框架的思路,让你操作集合和数组,变得更加的容易,简单和高效.

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章