前言最近在公司比较新的项目中,看到了很多java8新特性的用法。由于之前对java8的新特性了解不多,也没有深入研究,所以最近系统的研究了一下,然后总结出一篇文章尽快分享给大家。在了解一个新的之前技术,我们需要明白我们为什么要学习它,以及它的优势。以下是我的总结:Java8(又称jdk1.8)是java语言发展的一个大版本,而Java8是oracal公司于2014年3月发布的,可以说是自java5以来最具革命性的版本。新特性的优点:速度更快,代码更少,易于并行化,最小化空指针异常函数式编程提供更高层次的抽象排序:ListrolesListSort=rolesList.stream().sorted(Comparator.comparing(RoleEntity::getCreateDate)).collect(Collectors.toList());Consumer是一个函数式接口,参数是Consumer类型,Consumer中的泛型类型表示泛型类型要么是Integer,要么是Integer的父类,super表示它在它上面,也就是父类。下面这段代码是Iterable接口中的默认方法,jdk8之后的新方法,默认方法(引入默认方法很大程度上是为了保证向后兼容)defaultvoidforEach(Consumeraction){Objects.requireNonNull(动作);对于(Tt:this){action.accept(t);}}关于Java8的新特性,我总结了以下6个方面,可以借鉴:1、Lambda表达式我的理解是,lambda表达式其实是一套新的语法规则,主要是文法的要求。那么为什么我们需要Lambda表达式呢?在Java中,我们不能将函数作为参数传递给方法,也不能声明返回函数的方法;在JavaScript中,函数参数是一个函数而返回值是另一个函数是很常见的;JavaScript是一种非常典型的函数式语言。addUser(e->Sysout.out.println("hello"))e代表参数,->箭头符号代表分隔符。它的作用是左右分离。sysout.out.println("hello")是执行体,即代码块(如果执行体中的代码不止一行,可以加花括号括起来),所以Lambda表达式分为三个部分。Lambda表达式结构的基础知识:Lambda表达式可以有零个或多个参数,参数的类型可以显式声明或从上下文推断。例如,(inta)与(a)具有相同的效果;所有参数必须用括号括起来,参数之间用逗号隔开;空括号表示参数集为空。例如:()->42当只有一个参数且可以推导出其类型时,括号()可以省略。例如:a->returna*aLambda表达式的主体,即主体可以包含0个或多个语句。如果表达式体只有一条语句,大括号{}可以省略,匿名函数的返回类型与表达式体保持一致。如果表达式的主体包含多个语句,则必须将其包含在花括号{}中以形成一个代码块。匿名函数的返回类型与函数体表达式一致,没有返回则为空。statement和expression的区别,expression只有一句话,不需要用花括号括起来,也不需要return;语句需要用花括号括起来,如果有返回值,则必须返回(argument)->{body}也可以:(arg1,arg2)->{body}(typearg1,typearg2)->{body}(这是最完整的语法)(param1,param2,param3)->{}左括号代表方法的参数,右花括号代表方法的参数方法的具体实现()->{}类型是从上下文中推断出来的。其实就是在目标函数式接口中找到具体唯一的抽象方法,看抽象方法中的参数和返回类型,而抽象方法的名字对于Lambda表达式来说是没有意义的Lambda表达式的作用:Lambda表达式为Java添加了缺少的函数式编程特性,使我们能够将函数视为一等公民。在Citizen的语言中,lambda表达式的类型是函数。但是在Java中,Lambda表达式是对象,必须依附于一种特殊类型的对象——函数式接口(functionalinterface)传递的是行为,而不仅仅是值(之前的方式,是先定义好了,行为(thebehavior)behavioralreadyexists),然后调用这个behavior来使用它,但是现在相反,这个behavior是预先不存在的,通过方法来通知)//内部迭代integerList.forEach(newConsumer(){//匿名内部类@Overridepublicvoidaccept(Integerinteger){System.out.println(integer);}});二、函数式(Functional)接口函数式接口可以通过三种方式实现:Lambda表达式、方法引用和构造函数引用通过Lambda表达式、方法引用或构造函数引用创建函数式接口的实例关于函数式接口:如果一个接口只有一个抽象方法,那么这个接口就是一个函数式接口。如果我们在接口上声明@FunctionalInterface注解,编译器将根据功能接口的定义要求该接口。如果一个接口只有一个抽象方法,但是接口上没有声明@FunctionalInterface注解,编译器仍然会把这个接口当作一个函数式接口,Java8中引入的很多函数式接口都位于java.util.function下。下面是一些常用的函数式接口:位于java.util.function包下Consumer接受一个参数,不返回结果publicinterfaceConsumer{voidaccept(Tt);}函数接受一个参数并返回一个结果publicinterfaceFunction{Rapply(Tt);}BiFunction接收两个参数并返回一个结果(其中BI是bidirectional的缩写,双向的意思)publicinterfaceBiFunction{Rapply(Tt,Uu);}Supplier提供者,供应商,不接收任何参数,返回一个结果publicinterfaceSupplier{Tget();}Predicatepredicate,接收一个参数,返回一个布尔值(根据给定的参数,返回布尔值)publicinterfacePredicate{booleantest(Tt);}3.方法引用方法引用是Lambda表达式的一种特例(或者说是Lambda表达式的语法糖),可以理解为方法引用和Lambda表达式实现的功能其实是一样的,完全等价的,只是方法参考更简洁。我们可以把方法引用看成是函数指针(Functionpointer)方法引用(methodreferences):ListintegerList=Arrays.asList(1,2,3,4,5);//方法参考方法integerList.forEach(System.out::println);方法引用有4种:1.类名::静态方法名下面两种形式是完全不等的.引用名(对象名)::实例方法名3.类名::实例方法名4.构造函数引用:类名::new4.强大的StreamAPI其实是JDK8提供给我们的新API。它经常与Lambda表达式和功能接口一起使用。分为串行流和并行流。list.stream()串行流只有一个线程,一个线程执行所有操作list.parallelStream()并行流,多线程,分工list.stream().map():这里的map表示映射Stream也是一个接口,里面的方法大部分都是Higher-orderfunctionStream流,伴随着Lambda表达式,我们可以通过流方法更好的操作集合流的三部分:(SQL语句和流非常非常相似)1.Source2.零个或几个中间操作(操作为source,操作值为过滤、排序、映射、分区等,这些操作本身有点像SQL语句)3.终止操作流操作分类:1.Lazyevaluation2.Earlyevaluation流的所有中间操作方法都是惰性的(或延迟,或惰性求值),除非遇到终止操作或早期求值操作,否则不会执行中间操作是的,只有当终止操作遇到,这些中间操作会一起执行stream().xxx().zzz().count();filter()是用来判断里面的条件是真还是假?如果为假,则从流中过滤掉;如果为真,则继续放入流中进行后续操作。本质是函数式的,对流的操作会产生一个结果,但不会修改底层数据源,集合可以作为它是流的底层数据源;延迟查找,很多流操作(过滤,映射,排序,分区等)关心底层是怎么实现的,流程其实是一样的,只需要知道做什么,不需要了解底层是如何工作的内迭代和外迭代本质分析:(操作流程就像英文的完形填空,直接操作集合就是完成一个完整的命题组合)内迭代使用流程,也就是并行化,下面的代码可能认为有多个循环,但是流的底层实际上只用了一个循环。你可以这样想,流其实就是一个容器,里面有一个集合。这个集合存放了对流的各种操作,流会尽可能的优化;下面的代码并没有按顺序一一执行。它是由集合框架自己决定的用于外部迭代的集合。它是连载的。下图是我的代码,可以帮助大家理解集合关注数据和数据存储本身;streams关注的是数据的计算;流和迭代器之间的相似之处在于流不能被重用或消费。如何判断是中间操作还是终止操作?中间操作会返回一个Stream对象,比如Stream,Stream,Stream终止操作不会返回Steam类型,可能不返回值,也可能返回其他类型的单个值。Stream流中的方法:intsum=Stream.iterate(1,item->item+2)。limit(6).filter(item->item>2).mapToInt(item->item*2).skip(2).limit(2).sum();skip():忽略前几个元素limit():获取前几个元素sum():sum(map映射没有sum方法)Stream分组与分区(partition):grouping:groupbypartition:partitionby(Booleanvalue)partition是分组中流的一种特性特殊情况:stream一旦被操作或使用,就不能被复用,或者stream一旦被关闭,就不能被复用。5.Optional中文意思:OptionalOptional其实其他语言的使用其他语言很早就开始使用了(比如Swift、Groovy、Scala),Java是最晚被使用的。它的出现主要是解决问题:NPE(NullPointerException)if(null!=person){addressaddress=person.getName();if(null!=address){}}6.高阶函数Higher-orderfunctions:如果一个函数接受一个一个函数作为参数,或者一个函数作为返回值返回,那么这个函数就称为高阶函数。方法的实现可以声明在默认方法接口中,但是这个方法的实现必须携带default关键字。从java8开始,为什么?要添加默认方法吗?收集器collector(很重要)Rcollect(Collectorcollector);collect:collectorCollector作为collect方法的参数Collector是一个接口,它是一个可变的集合操作,它将输入的元素累加到一个可变的结果容器中(ArrayList是一个可变的容器),并将累加的结果转化为最终的representation在所有元素都处理完之后(这是一个可选的操作),它支持串行(一个线程执行)和并行(多个线程执行)执行。Collectors本身提供了Collectors的通用聚合实现。Collectors本身其实就是一个工厂(Collectors提供了很多变量聚合操作的实现)publicinterfaceCollector{Suppliersupplier();BiConsumeraccumulator();//转化为累加器//将两个结果容器合并为一个(用于线程并发)BinaryOperatorcombiner();//combinerFunctionfinisher();//finisher}收集器恒等性与结合性解析组合器函数:Iterator迭代器总结以上是我对jdk1.8新特性的总结,欢迎大家交流。最后,最近有很多朋友找我要一份Linux学习路线图,所以我结合自己的经验,利用业余时间熬夜一个月,整理了一本电子书。无论你是面试还是自我提升,相信都会对你有所帮助!免费送给大家,只求大家给我点个赞!电子书|LinuxDevelopmentLearningRoadmap也希望有小伙伴可以和我一起把这本电子书做得更完美!获得?希望老铁们来个三连击,让更多人看到这篇文章。推荐阅读:干货|程序员和高级架构师免费发送工件的必备资源|支持搜索的资源网站