java基础学习_JDK新特性_day28总结

java基础学习_JDK新特性_day28总结

=============================================================================
=============================================================================
涉及到的知识点有:
    0:JDK5 & JDK7 新特性
    1:Java 8 新特性
    2:Java 9 新特性
    3:Java 10 新特性
=============================================================================
=============================================================================
0:JDK5 & JDK7 新特性
    (1) JDK5新特性
        自动装箱和拆箱(day13)
        泛型(day16)
        增强for循环(day16)
        静态导入(day16)
        可变参数(day16)
        枚举(day27)
--------------------------------------        
    (2) JDK7新特性
        二进制字面量(二进制的表现形式)
        数字字面量可以出现下划线(用_分隔数据)
        switch语句的表达式可是用字符串(day04)
        泛型简化(泛型推断(也叫菱形泛型))
        异常的多个catch合并(多catch的使用)(day19)
        try-with-resources 语句(自动释放资源的用法)
-----------------------------------------------------------------------------
1:Java 8 新特性
    (1) 接口中有方法了(可以有默认方法、静态方法、私有方法(JDK9))。
-----------------------------------------------------------------------------
    (2) Lambda表达式
        A: 接口的实现类的目的是:为了给接口中的方法找方法体。即为了使用方法体,不得不去定义一个实现类。
            你以为这是理所当然的,其实不是。
            
            对于某些场景,我能不能有一种更加先进,更加清亮的方法呢? 
            答:有,函数式编程。(Lambda表达式编程)
--------------------------------------        
        B: 从Java 8 开始,没有实现类的接口,也可以直接使用接口了。如何使用? 
            答:Lambda表达式。
            其实就像Lambda表达式替代了实现类一样。
--------------------------------------        
        C: Java语言中使用Lambda表达式的前提是:必须有"函数式接口"
    
            函数式接口:有且仅有一个抽象方法的接口,叫做函数式接口。

            如何才能万无一失地检测一下当前接口是不是函数式接口呢?
            用一个固定的格式写在public interface的前一行即可。
            
            格式:
                @FunctionalInterface 
                public interface 函数式接口名 {
                ...
                }
                
                @FunctionalInterface 
                注解:在java.lang包中
                
            注意:
                不管写不写@FunctionalInterface,只要有且仅有一个抽象方法的接口就是函数式接口!
                但是呢?建议永远写上。
--------------------------------------                
        D: Lambda表达式要想使用,一定要有函数式接口的推断环境。
            推断环境?
                方式1. 要么通过方法的参数类型,来确定是哪个函数式接口。
                方式2. 要么通过赋值操作,赋值语句左侧的数据类型,来确定是哪个函数式接口。
                    
                    // 通过方法的参数类型,来确定是哪个函数式接口。
                    // 在调用方法的时候,方法的参数类型是函数式接口,所以Lambda可以推断出来是哪个接口。
                    method((int a, int b) -> {
                        return a + b;
                    });
                    
                    // 通过赋值操作,赋值语句左侧的数据类型,来确定是哪个函数式接口。(赋值语句左侧的类型来进行Lambda上下文推断)
                    Calculator cal = (int a, int b) -> {
                        return a + b;
                    };
                    method(cal);
                    
                    // 错误写法!没有上下文环境,Lambda就无法推断是哪个函数式接口。
                    /*
                    (int a, int b) -> {
                        return a + b;
                    };
                    */
        
            Lambda的格式就是为了将抽象方法,翻译成以下三点:
                1. 一些参数(方法参数)
                2. 一个箭头
                3. 一些代码(方法体)
            例如这个抽象方法:
                public abstract int sum(int a, int b);
                翻译成为Lambda的标准格式:
                (int a, int b) -> { 
                    return a + b; 
                } 
--------------------------------------                
        E: Lambda的简便格式
            在Lambda表达式当中,凡是可以推断的,都是可以省略的。
                1. Lambda表达式当中的参数类型可以省略不写。
                2. 如果参数有且仅有一个,那么可以省略小括号。
                3. 如果语句只有一条,那么大括号、分号和return也可以省略。
                
                // Lambda表达式的标准格式
                method((int x) -> {
                    return ++x;
                });

                // 简化格式1:省略参数类型:Lambda表达式当中的参数类型可以省略不写。
                method((x) -> {
                    return ++x;
                });

                // 简化格式2:再省略小括号:如果参数有且仅有一个,那么可以省略小括号。
                method(x -> {
                    return ++x;
                });

                // 简化格式3:再省略大括号和return:如果语句只有一条,那么大括号、分号和return也可以省略。
                method(x -> ++x);                            
-----------------------------------------------------------------------------                
    (3) 方法引用
        Lambda表达式的冗余场景:
            在某些场景下,Lambda表达式要做的事情,在另一个地方已经写过了(即已有方法体了,已经实现了)。
            那么此时如果通过Lambda表达式重复编写相同的代码,就是浪费。
    
        那么此时如何才能复用已经存在的方法逻辑呢?
            JDK8中新特性:方法引用(为了简化Lambda而存在)
            即:如果Lambda表达式需要做的事情,在另一个类当中已经做过了,那么就可以直接拿过来替换Lambda。这种方式叫做方法引用。
        
        方法引用的格式:
            1.通过类名引用静态方法,格式:类名::静态方法名    
        方法引用的格式:
            2.通过对象名引用静态方法,格式:对象名::成员方法名    
        还有其他的格式:
            ...
        小结:
            1. 一定要先有函数式接口,才能使用Lambda表达式。
            2. 对于Lambda表达式使用的冗余场景,可以使用方法引用来进行简化。
-----------------------------------------------------------------------------
    (4) Stream API 流式操作
        集合for遍历的冗余场景:
--------------------------------------        
        Stream流式思想概述:
            像流水线操作一般执行。从其他语言中借鉴过来的。可以很方便的操作多个元素。
            即简化了普通的集合操作。
--------------------------------------            
        获取Stream流对象的常用方式:
            Java 8当中的“流”其实就是Stream接口的对象。
            JDK提供了一个流接口:java.util.stream.Stream<T>
        如何获取Stream流对象?
            1. 根据集合获取流对象
                集合名称.stream();
            2. 根据数组获取流对象,数组当中的元素必须是引用类型才行
                Stream.of(数组名称);
--------------------------------------                
        获取Stream流对象后如何使用map方法呢?
            我们获取流对象后,可以使用映射方法:map(用于转换的Lambda表达式)
            映射:就是将一个对象转换成另一个对象,把老对象映射到新对象上。
            举例:"赵丽颖,98"    转换成为    "98"    把一个长字符串转换为一个短字符串
                    "98"        转换成为    98        把一个字符串转化成为一个int数字    
--------------------------------------        
        如果希望对流对象当中的元素进行过滤,可以使用过滤方法:filter(能产生boolean结果的lambda表达式)
            如果参数Lambda产生了true,则要该元素;如果产生了false,则不要该元素。    
--------------------------------------
        如果希望在流当中进行元素的遍历操作,可以使用forEach方法:forEach(Lambda表达式)
            意思是:对流当中的每一个元素都要进行操作。
            注意:参数Lambda表达式必须是一个能够消费一个参数,而且不产生数据结果(无返回值)的Lambda表达式。
            例如:
                Lambda:        s -> System.out.println(s);
                方法引用:        System::println            
--------------------------------------        
        流当中的元素如果特别多,那么只有一个人在逐一、挨个处理,肯定比较慢、费劲。
        如果对流当中的元素,使用多个人同时处理,这就是“并行”。
            parallel    并行,平行
        Java 8中将并行进行了优化,我们可以很容易的对数据进行并行操作。
        Stream API可以声明性地通过parallel()与sequential()在并行流与顺序流之间进行切换。            
        
        如何获取“并行流”?
            .parallelStream()        // 直接获取
            .stream.parallel()       // 间接获取

        注意事项:
            1. 使用并行流操作的时候,到底有几个人进行同时操作呢?我们不用管,JDK自己处理。(用的是Fork/Join框架)
            2. 只要正确使用的话,就不会出现多个人抢到同一个元素的情况。    
-----------------------------------------------------------------------------            
2:Java 9 新特性                
    (1) 模块化思想概述
        1. 把整体分成部分(模块)。文件体积变小。
        2. 在模块中增加权限控制。权限控制更精确。
            如下图所示:

(
2) 认识module-info.java文件 如下图所示:
(
3) 将Eclipse项目改造为模块 选择项目右键 --> Configure --> Create module-info.java --> xxx.mod(模块命名不推荐使用数字) 再配置好module-info.java文件中的exports和requires。 (4) 设置模块间的依赖关系 1. 在module-info.java文件中先加入requires语句(最底层包名)。 2. 选中项目右击 --> Build Path --> Configure Build Path... --> Projects --> Modulepath --> Add... --> 选中所依赖的项目(该步骤是Ecplise所特有的) ----------------------------------------------------------------------------- 3:Java 10 新特性 Java 10 新特新链接:https://juejin.im/entry/5ab1c64951882555880545f2 =============================================================================

 

我的GitHub地址:https://github.com/hei*gjun
我的博客园地址:http://www.cnblogs.com/chenmingjun
我的蚂蚁笔记博客地址:http://blog.leanote.com/chenmingjun
Copyright ©2018 黑泽明军
【转载文章务必保留出处和署名,谢谢!】
上一篇:在Sharepoint Designer 2007 中加入定制的工作流动作(翻译)


下一篇:Oracle 11gR2新特性--延迟段创建(Deferred Segment Creation)