前言

​ 最近公司里比较新的项目里面,看到了很多关于java8新特性的用法,由于之前自己对java8的新特性不是很了解也没有去做深入研究,所以最近就系统的去学习了一下,然后总结了一篇文章第一时间和大家分享一下。

​ 在了解一项新技术之前,我们需要了解我们为什么要去学习它以及它的优点,以下是我总结的:

Java8(又称jdk1.8)是java语言开发的一个主要版本,Java8是oracal公司于2014年3月发布,可以看成是自java5以来最具有革命性的版本。

新特性的优点:速度更快、代码更少、便于并行、最大化减少空指针异常

函数式编程提供了一种更高层次的抽象化

排序:

List<RoleEntity> rolesListSort = rolesList.stream().sorted(Comparator.comparing(RoleEntity::getCreateDate)).collect(Collectors.toList());

Consumer是一个函数式接口

参数是Consumer类型的,Consumer里面的泛型表示泛型的类型要么是Integer,要么是Integer的父类,super表示它及它上面的,也就是父类。

下面这段代码是在Iterable接口里面的默认方法,jdk8之后的新方法,默认方法(默认方法的引入很大程度上是为了保证向后兼容)

default void forEach(Consumer<? super T> action) {
Objects.requireNonNull(action);
for (T t : this) {
action.accept(t);
}
}

关于Java8的新特性,我总结了以下6个方面,我们可以从以下6个方面进行学习了解:

一、Lambda表达式

​ 我的理解lambbda表达式其实是新的一套语法规则,主要是语法上面的要求。

那我们为啥需要Lambda表达式?

在java中,我们无法将函数作为参数传递给一个方法,也无法声明返回一个函数的方法;在JavaScript中,函数参数是一个函数,返回值是另一个函数的情况是非常常见的;JavaScript是一门非常典型的函数式语言。

addUser(e -> Sysout.out.println("hello"))e表示参数,->箭头符号,表示分隔符,他的作用是分割左边和右边的。Sysout.out.println("hello")是执行体,也就是代码块(如果执行体里面不止一行代码,那就可以加上花括号括起来)所以Lambda表达式分为三部分

Lambda表达式的基本结构:

  • 一个Lambda表达式可以有0个或多个参数,参数的类型可以明确声明,也可以通过上下文来推断。例如(int a)和(a)效果一样;
  • 所有参数都必须包含在圆括号内,参数之间用逗号相隔;
  • 空圆括号代表参数集为空。例如:()-> 42
  • 当只有一个参数,且其类型可以推导出时,圆括号()可以省略。例如:a -> return a*a
  • Lambda表达式的主体也就是body可以包含0条或多条语句。
  • 如果表达式的主体只有一条语句,花括号{}可以省略,匿名函数的返回类型与该主体表达式一致
  • 如果表达式的主体包含一条语句以上,则必须包含在花括号{}里面形成代码块。匿名函数的返回类型与该主体表达式一致,若没有返回则为空。
  • statement和expression的区别,expression只有一句,不需要花括号包裹,不需要return;statement需要花括号包裹,且如果有返回值,必须return

(argument)-> {body}

也可以:

(arg1, arg2)-> {body}

(type arg1, type arg2)-> {body}(这个是最完整的语法)

(param1,param2,param3)-> {} 左边圆括号里面表示方法的参数 ,右边花括号里面代表方法的具体实现

()-> {} 类型是通过上下文来推断的

实际就是去目标函数式接口里面去找那个特定的唯一的抽象方法,去看抽象方法里面的-参数和返回类型,而抽象方法的名字对于Lambda表达式来说是毫无意义的

Lambda表达式的作用:

  • Lambda表达式为Java添加了缺失的函数式编程特性,使我们能将函数当作一等公民看待
  • 在将函数作为一等公民的语言中,Lambda表达式的类型是函数。但在Java中,Lambda表达式是对象,他们必须依附于一类特别的对象类型——函数式接口(functional interface)
  • 传递的是行为,而不仅仅是值(在以前的方式中,是先定义好了行为(行为已经存在),然后再调用这个行为进行使用,而现在是相反,行为是提前并不存在,是通过方法的传递来进行告知的)

//内部迭代    integerList.forEach(new Consumer<Integer>() {
//匿名内部类 @Override public void accept(Integer integer) { System.out.println(integer);
}
});

二、函数式(Functional)接口

​ 函数式接口是可以通过三种方式实现的:Lambda表达式、方法引用、构造器引用

通过Lambda表达式、方法引用或者构造器引用的来创建一个函数式接口的实例

关于函数式接口:

  1. 如果一个接口只有一个抽象方法,那么该接口就是一个函数式接口
  2. 如果我们在某个接口上声明了@FunctionalInterface注解,那么编译器就会按照函数式接口的定义来要求该接口。
  3. 如果一个接口只有一个抽象方法,但是在该接口上并没有声明@FunctionalInterface注解,那么编译器依旧会把该接口看作一个函数式接口

Java8里面引入的很多函数式接口它们都位于java.util.function下面。

以下是一些常用的函数式接口:

位于java.util.function这个包下面

Consumer消费者 接受一个参数,不返回结果

public interface Consumer { void accept(T t); }

Function,接受一个参数,返回一个结果

public interface Function<T, R> { R apply(T t); }

BiFunction接收两个参数,返回一个结果(其中BI是bidirectional的缩写,意思是双向)

public interface BiFunction<T, U, R> { R apply(T t, U u); }

Supplier 提供者,供应者,不接收任何参数,返回一个结果

public interface Supplier { T get(); }

Predicate谓语,接收一个参数,返回一个布尔值(根据给定的参数,返回布尔)

public interface Predicate { boolean test(T t); }

三、方法引用

方法引用是Lambda表达式的一种特殊情况(或者说是Lambda表达式的一个语法糖),可以理解为方法引用和Lambda表达式这两种方式所实现的功能其实一样的,完全等价,但是方法引用的方式更简洁。

我们可以将方法引用看作是一个函数指针(Function pointer)

方法引用(method references):

List<Integer> integerList = Arrays.asList(1,2,3,4,5); //方法引用的方式 integerList.forEach(System.out::println);

方法引用有4种:

1、类名::静态方法名

以下这两种形式是完全不等价的

classname::staticmethod(表示的是指向,函数指针的概念)

classname.staticmethod(真正表示的是方法调用的概念)

2、引用名(对象名)::实例方法名

3、类名::实例方法名

4、构造方法引用(constructor references):类名::new

四、强大的Stream API

其实就是JDK8提供给我们新的API,经常和Lambda表达式和函数式接口一起使用

分为串行流和并行流

list.stream()串行流,只有一个线程,一个线程执行所有操作

list.parallelStream()并行流,多线程,分工合作

list.stream().map():map此处的意思是映射的意思

Stream也是一个接口,里面的绝大多数方法都是高阶函数

Stream流,他是与Lambda表达式相伴相生的,通过流的方式我们可以更好的操作集合

流的三部分构成:(SQL语句和流非常非常像)

1、源

2、零个或若干个中间操作(操作的是这个源,操作值的是过滤,排序,映射,分区等,这些操作本身有点像SQL语句)

3、终止操作

流操作分类:

1、惰性求值

2、及早求值

流的所有的中间操作方法都是lazy的(或者说是延迟的,或者说是惰性求值的),在没有遇到终止操作或者及早求值的操作的情况下,中间操作是不会被执行的,只有在遇到终止操作的时候,这若干个中间操作才会一并的执行

stream().xxx().zzz().count();

filter()用来判断里面的条件是真还是假?如果是假,就从流当中过滤掉;如果是真,就继续放到流当中,供后续操作使用

流:

  • Colletion提供了新的Stream()方法;
  • 流不存储值,通过管道的方式获取值;
  • 本质是函数式的,对流的操作会造成一个结果,不过并不会修改底层的数据源,集合可以作为流的底层数据源;
  • 延迟查找,很多流操作(过滤,映射,排序,分区等)都可以延迟实现;

SQL语句是一种描述性的语言,只需要发送指令告诉底层需要做什么,而不关心底层是怎么实现的,而流其实也是一样的,只需要知道做什么,而不需要知道具体底层是怎么做的。

内部迭代和外部迭代本质刨析:(操作流就像英语中的完形填空,直接操作集合就是完成一个完整的命题作文)

内部迭代

用流,是并行化,以下代码可能你觉得有多个循环,但是流的底层实际上只用了一个循环,可以这样想,流实际上是一个容器,里面有一个集合,这个集合存放的是对流的各种操作,流会尽最大可能去优化;以下代码也不是按照顺序一个一个执行的,是由集合框架自己决定的

外部迭代

用集合,是串行化,下图是我的代码,可以帮助大家理解

集合关注的是数据与数据存储本身;

流关注的是对数据的计算;

流与迭代器类似的一点是:流是无法重复使用或消费的

如何判断是中间操作还是终止操作呢

中间操作都会返回一个Stream对象,比如Stream,Stream,Stream

终止操作则不会返回Steam类型,可能不返回值,也可能返回其他类型的单个值

Stream流里面的方法:

int sum = Stream.iterate(1, item -> item + 2).limit(6).filter(item -> item > 2)            .mapToInt(item -> item * 2) .skip(2).limit(2).sum();

skip():忽略掉前几个元素

limit():获取前几个元素

sum():求和(map映射是没有求和方法的)

Stream分组与分区(partition ):

分组:group by

分区:partition by (布尔值)

分区是分组的一种特殊情况

流的特性:

流一旦被操作或使用了,就不能再去重复的使用这个流,或者说流一旦被关闭了,也是不能再去重复使用了

五、Optional类

中文意思:可选

Optional类的使用其实在其他语言里很早就使用了(比如Swift、Groovy、Scala),Java是最晚使用的,

它的出现主要解决的问题:NPE(NullPointerException)

if (null != person){ Address address = person.getName(); if (null != address){ } }

六、高阶函数

高阶函数:如果一个函数接受一个函数作为参数,或者返回一个函数作为一个返回值,那么该函数就叫做高阶函数。

默认方法

接口当中可以声明方法的实现了,但是这个方法的实现必须要带上default关键字

从java8开始,为啥要增加默认方法?

Collector收集器(很重要)

<R, A> R collect(Collector<? super T, A, R> collector);

  • collect:收集器
  • Collector作为collect方法的参数
  • Collector是一个接口,它是一个可变的汇聚操作,将输入元素累积到一个可变的结果容器中(ArrayList就是一个可变的容器),它会在所有元素处理完毕之后,将累积的结果转换成一个最终的表示(这是一个可选操作),它支持串行(一个线程执行)和并行(多个线程执行)两种方式执行。
  • Collectors本身提供了关于Collector的常见汇聚实现,Collectors本身实际是一个工厂(Collectors提供了很多可变汇聚操作的实现)

public interface Collector<T, A, R>{ Suppliersupplier(); BiConsumer<A, T> accumulator();//翻译成累加器 //将两个结果容器合并成一个(用于线程并发) BinaryOperatorcombiner();//结合器 Function<A, R> finisher();//完成器 }

Collector同一性和结合性分析

combiner函数:

Iterator迭代器

总结

​ 以上是我关于jdk1.8新特性的一些总结,欢迎大家相互交流。


公众号:良许Linux

有收获?希望老铁们来个三连击,给更多的人看到这篇文章

关于Java8的精心总结的更多相关文章

  1. java8 新特性精心整理

    前言 越来越多的项目已经使用 Java 8 了,毫无疑问,Java 8 是Java自Java 5(发布于2004年)之后的最重要的版本.这个版本包含语言.编译器.库.工具和 JVM 等方面的十多个新特 ...

  2. java8 新特性精心整理(全)

    前言 越来越多的项目已经使用 Java 8 了,毫无疑问,Java 8 是Java自Java 5(发布于2004年)之后的最重要的版本.这个版本包含语言.编译器.库.工具和 JVM 等方面的十多个新特 ...

  3. Java8实战分享

    虽然很多人已经使用了JDK8,看到不少代码,貌似大家对于Java语言or SDK的使用看起来还是停留在7甚至6. Java8在流式 or 链式处理,并发 or 并行方面增强了很多,函数式的风格使代码可 ...

  4. java8中lambda表达式的应用,以及一些泛型相关

    语法部分就不写了,我们直接抛出一个实际问题,看看java8的这些新特性究竟能给我们带来哪些便利 顺带用到一些泛型编程,一切都是为了简化代码 场景: 一个数据类,用于记录职工信息 public clas ...

  5. Android Studio2.1.2 Java8环境下引用Java Library编译出错

    转载请注明出处:http://www.cnblogs.com/LT5505/p/5685242.html 问题:在Android Studio2.1.2+Java8的环境下,引用Java Librar ...

  6. Java笔记——Java8特性之Lambda、方法引用和Streams

    Java8已经推出了好一段时间了,而掌握Java8的新特性也是必要的,如果要进行Spring开发,那么可以发现Spring的官网已经全部使用Java8来编写示例代码了,所以,不学就看不懂. 这里涉及三 ...

  7. 关于Java8函数式编程你需要了解的几点

    函数式编程与面向对象的设计方法在思路和手段上都各有千秋,在这里,我将简要介绍一下函数式编程与面向对象相比的一些特点和差异. 函数作为一等公民 在理解函数作为一等公民这句话时,让我们先来看一下一种非常常 ...

  8. 精心挑选10款优秀的 jQuery 图片左右滚动插件

    在现代的网页设计中,图片和内容滑块是一种极为常见和重要的元素.你可以从头开始编写自己的滑动效果,但是这将浪费很多时间,因为网络上已经有众多的优秀的 jQuery 滑块插件.当然,如果要从大量的 jQu ...

  9. Java8并发教程:Threads和Executors

    来之:ImportNew 欢迎阅读我的Java8并发教程的第一部分.这份指南将会以简单易懂的代码示例来教给你如何在Java8中进行并发编程.这是一系列教程中的第一部分.在接下来的15分钟,你将会学会如 ...

随机推荐

  1. CentOS快速安装Nginx的方法,nginx如何启动重启停止

    1.防止 make: command not found,提前安装一些插件,取决于当前环境是否已安装,如果已经安装就不需要执行此命令 yum -y install gcc automake autoc ...

  2. WireShark——IP协议包分析(Ping分析IP协议包)

    互联网协议 IP 是 Internet Protocol 的缩写,中文缩写为“网协”.IP 协议是位于 OSI 模型中第三层的协议,其主要目的就是使得网络间能够互联通信.前面介绍了 ARP 协议, 该 ...

  3. Flutter学习笔记(38)--自定义控件之组合控件

    如需转载,请注明出处:Flutter学习笔记(38)--自定义控件之组合控件 在开始之前想先写点其他的,emm...就是今天在学习到自定义控件的时候,由于自定义控件这块一直是我的短板,无论是Andro ...

  4. Linux中bash的一些命令

    Linux——bash的简单使用 bash及其特性: 1.bash实质上是一个可执行的程序,一个用户的工作环境. 2.每一个shell下可以再打开一个shell,新打开的shell称为子shell,每 ...

  5. Java Service Wrapper 浅谈

    在实际开发过程中很多模块需要独立运行,他们并不会以web形式发布,传统的做法是将其压缩为jar包独立运行,这种形式简单易行也比较利于维护,但是 一旦服务器重启或出现异常时,程序往往无法自行修复或重启. ...

  6. android studio 中jni底层日志的打印

    1 添加ndk对log支持若需要添加ndk对log的支持,只需要通过以下2步即可实现. 1.1 修改Android.mk如生成的库文件是“.so文件”,则在Android.mk中添加如下内容:LOCA ...

  7. 入门大数据---Hbase 过滤器详解

    一.HBase过滤器简介 Hbase 提供了种类丰富的过滤器(filter)来提高数据处理的效率,用户可以通过内置或自定义的过滤器来对数据进行过滤,所有的过滤器都在服务端生效,即谓词下推(predic ...

  8. 入门大数据---Spark累加器与广播变量

    一.简介 在 Spark 中,提供了两种类型的共享变量:累加器 (accumulator) 与广播变量 (broadcast variable): 累加器:用来对信息进行聚合,主要用于累计计数等场景: ...

  9. TCP协议粘包问题详解

    TCP协议粘包问题详解 前言 在本章节中,我们将探讨TCP协议基于流式传输的最大一个问题,即粘包问题.本章主要介绍TCP粘包的原理与其三种解决粘包的方案.并且还会介绍为什么UDP协议不会产生粘包. 基 ...

  10. Asp.net Core AOP实现(采用Autofac)

    引用正确的库来实现AOP 新的.NET Core是基于.NET Standard的..所以我们在引用库的时候特别要注意相关的兼容问题. 在传统的ASP.NET中,使用过Autofac来进行AOP操作的 ...