zoukankan      html  css  js  c++  java
  • JAVA8 新特性

     
     

    一、Lambda表达式

    Lambda表达式可以说是Java 8最大的卖点,她将函数式编程引入了Java,避免了匿名类的代码冗余问题 。Lambda允许把函数作为一个方法的参数,或者把代码看成数据。
    一个Lambda表达式可以由用逗号分隔的参数列表、–>符号与函数体三部分表示。例如:
     
    List<String> strList = Arrays.asList( "a", "d", "b" );
     strList.forEach( e -> System.out.println( e ) );
    假设我们想对一个List<String>按字符串长度进行排序,那么在Java8之前,可以借助匿名内部类来实现:
    List<String> words = Arrays.asList("apple", "banana", "pear");  
    words.sort(new Comparator<String>() {  
      
        @Override  
        public int compare(String w1, String w2) {  
            return Integer.compare(w1.length(), w2.length());  
        }  
      
    });
    改为lamada后代码量明显减少
     words.sort((String w1, String w2) -> {  
        return Integer.compare(w1.length(), w2.length());  //如果lambda的语句块只有一行,不需要return关键字,会自动返回
    });
    Lambda表达式,很像一个匿名的方法,只是圆括号内的参数列表和花括号内的代码被->分隔开了。垃圾代码写的越少,我们就有越多的时间去写真正的逻辑代码。为了使现有函数更好的支持Lambda表达式,Java 8引入了函数式接口的概念。函数式接口就是只有一个方法的普通接口。java.lang.Runnable与java.util.concurrent.Callable是函数式接口最典型的例子。为此,Java 8增加了一种特殊的注解
    @FunctionalInterface
    public interface FruitI {
    void juice(String height);
    default void defaultMethod1(String s){
    System.out.println("defaultMethod1"+s);
    }
    default void defaultMethod2(String s){
    System.out.println("defaultMethod2"+s);
    }
    static void statisticMethod1(String s){
    System.out.println("statisticMethod1"+s);
    }
    static void statisticMethod2(String s){
    System.out.println("statisticMethod2"+s);
    }
    }

     调用匿名类的juice方法

    FruitI fruit = new FruitI() {
    @Override
    public void juice(String height) {
    System.out.print(height);
    }
    };
    fruit.juice("5kg");

     调用函数式接口的juice方法

    FruitI fruit = height -> System.out.print(height);
    fruit.juice("5kg");
    fruit.defaultMethod1("5kg");
    fruit.defaultMethod1("5kg");
    FruitI.statisticMethod1("5kg");
    FruitI.statisticMethod2("5kg");

    二、接口的默认方法与静态方法

    我们可以在接口中定义默认方法,使用default关键字,并提供默认的实现。所有实现这个接口的类都会接受默认方法的实现,除非子类提供的自己的实现。
    接口的默认方法和静态方法的引入,其实可以认为引入了C++中抽象类的理念,以后我们再也不用在每个实现类中都写重复的代码了。

    三、方法引用

    通常与Lambda表达式联合使用,可以直接引用已有Java类或对象的方法。一般有四种不同的方法引用:
    1. 构造器引用。语法是Class::new,或者更一般的Class< T >::new,要求构造器方法是没有参数;
    2. 静态方法引用。语法是Class::static_method,要求接受一个Class类型的参数;
    3. 特定类的任意对象方法引用。它的语法是Class::method。要求方法是没有参数的;
    4. 特定对象的方法引用,它的语法是instance::method。要求方法接受一个参数,与3不同的地方在于,3是在列表元素上分别调用方法,而4是在某个对象上调用方法,将列表元素作为参数传入;

    四、重复注解

    在Java 5中使用注解有一个限制,即相同的注解在同一位置只能声明一次。Java 8引入重复注解,这样相同的注解在同一地方也可以声明多次。重复注解机制本身需要用@Repeatable注解。Java 8在编译器层做了优化,相同注解会以集合的方式保存,因此底层的原理并没有变化。

    五、扩展注解的支持

    Java 8扩展了注解的上下文,几乎可以为任何东西添加注解,包括局部变量、泛型类、父类与接口的实现,连方法的异常也能添加注解。

    六、Optional

    Java 8引入Optional类来防止空指针异常,Optional类最先是由Google的Guava项目引入的。Optional类实际上是个容器:它可以保存类型T的值,或者保存null。使用Optional类我们就不用显式进行空指针检查了。
    public static String getName(User u) {
        if (u == null)
            return "Unknown";
        return u.name;
    }
    我们可以安心的进行链式调用,而不是一层层判断了。看一段代码:
    public static String getName(User u) {
        return Optional.ofNullable(u)
                        .map(user->user.name)
                        .orElse("Unknown");
    }

    七、Stream

    Stream API是把真正的函数式编程风格引入到Java中。其实简单来说可以把Stream理解为MapReduce,当然Google的MapReduce的灵感也是来自函数式编程。她其实是一连串支持连续、并行聚集操作的元素。从语法上看,也很像linux的管道、或者链式编程,代码写起来简洁明了,非常酷帅!

    八、Date/Time API

    Java 8新的Date-Time API (JSR 310)受Joda-Time的影响,提供了新的java.time包,可以用来替代 java.util.Date和java.util.Calendar。一般会用到Clock、LocaleDate、LocalTime、LocaleDateTime、ZonedDateTime、Duration这些类,对于时间日期的改进还是非常不错的。

    九、JavaScript引擎Nashorn

    Nashorn允许在JVM上开发运行JavaScript应用,允许Java与JavaScript相互调用。

    十、Base64

    在Java 8中,Base64编码成为了Java类库的标准。Base64类同时还提供了对URL、MIME友好的编码器与解码器。
    除了这十大新特性之外,还有另外的一些新特性:
    • 更好的类型推测机制:Java 8在类型推测方面有了很大的提高,这就使代码更整洁,不需要太多的强制类型转换了。
    • 编译器优化:Java 8将方法的参数名加入了字节码中,这样在运行时通过反射就能获取到参数名,只需要在编译时使用-parameters参数。
    • 并行(parallel)数组:支持对数组进行并行处理,主要是parallelSort()方法,它可以在多核机器上极大提高数组排序的速度。
    • 并发(Concurrency):在新增Stream机制与Lambda的基础之上,加入了一些新方法来支持聚集操作。
    • Nashorn引擎jjs:基于Nashorn引擎的命令行工具。它接受一些JavaScript源代码为参数,并且执行这些源代码。
    • 类依赖分析器jdeps:可以显示Java类的包级别或类级别的依赖。
    • JVM的PermGen空间被移除:取代它的是Metaspace(JEP 122)。

    十一、HashMap的内部实现使用Node来替换原来的Entry

    Node可能是链表结构,也可能是红黑树结构。如果插入的key的hashcode相同,那么这些key也会被定位到Node数组的同一个格子里。如果同一个格子里的key不超过8个,使用链表结构存储。如果超过了8个,那么会调用treeifyBin函数,将链表转换为红黑树。那么即使hashcode完全相同,由于红黑树的特点,查找某个特定元素,也只需要O(log n)的开销也就是说put/get的操作的时间复杂度最差只有O(log n)。

  • 相关阅读:
    江西理工大学南昌校区cool code竞赛
    喵哈哈村的魔法考试 Round #3 (Div.2) ABCDE
    项目管理概要记录
    JS开发引用HTML DOM的location和document对象
    Linux下触摸屏驱动程序分析
    敦泰FT6X06单层自容调屏
    FT5X06 如何应用在10寸电容屏(linux-3.5电容屏驱动简析&移植10寸电容屏驱动到Android4.2) (by liukun321咕唧咕唧)
    基于FT5x06嵌入式Linux电容触摸屏驱动
    Linux/Android多点触摸协议
    高通 8x12 添加 TP和按键
  • 原文地址:https://www.cnblogs.com/wzj4858/p/8204960.html
Copyright © 2011-2022 走看看