学习记录之JDK5\JDK8基础(总结)
1泛型应用
1.1:概述
泛型是JDK1.5以后推出的编译时类型,运行时无效, 我么通常可以将其理解为一种参数化类型,可以应用在接口、类、方法、属性上,用于约束类中属性,方法参数以及返回值类型,基于这种方式可以对变成过程中的接口、类、方法进行通用化设计,同时在编译阶段解决在运行时执行一些校验操作,提高其运行效率。
1.2:泛型类
单泛型参数,关键代码如下:
多泛型参数,关键代码如下:
泛型抽象类,代码如下
泛型具体类代码如下:
泛型方法
泛型方法主要应用与类中的静态方法,但不限于静态方法:
1.3:泛型通配符
无界通配符:
泛型中的无界通配符使用“?”表示,泛指不确定性类型,例如,可以代表一种任意参数类型(实参类型),通常会应用于变量的定义
有界通配符
泛型在应用时有时需要指定对象的上限和下限,我们称这种形式叫限定通配符,语法如下:
指定泛型下限:<? super T> 表示参数是类型T或者T的父类,不影响往里存,但是往外取之恶能放在Object对象里。
指定泛型上限:<? extends T>表示参数类型是T或T的子类,只能往外取,不能往里放,因为编译简单不能确定你要放入的类型
有界通配符应用案例,构建一打印工具类,关键代码如下:
说明,如果要从集合中读取类型T的数据, 并且不能写入,可以使用 上界通配符(<? extends>)—Producer Extends。如果要从集合中写入类型T 的数据, 并且不需要读取,可以使用下界通配符(<? super>)—Consumer Super。如果既要存又要取, 那么就要使用非限定通配符。
1.4:泛型类型擦除
泛型是在编译阶段由编译器执行类型检查和类型推断,然后生成普通的非泛型的字节码。这种实现技术称之为擦除(erasure),所以说泛型是编译时的一种类型,在运行时无效,运行时候都会编程Object类型。假如,尝试基于反射技术向List<String> list = new ArrayList<String>()对象中添加整数100。关键代码如下:
2注解应用
2.1:注解是JDK5推出的一种新的应用类型(特殊的class),是一种元数据(Meta Data):用于描述接口、类、属性、方法、参数等,可以理解为一个生活中标签
2.2:注解的定义
我们先来看一下,Java中自带的Overide注解,代码如下:
在实际项目注解可能由第三方定义,也可能会由我们自己定义,例如
其中:@Target 用于描述定义的注解能够修饰的对象,@Retention 用于描述定义的
注解何时有效。
创建注解时,指定注解属性,例如:
2.3:注解与反射
使用注解描述类及成员,代码如下:
通过反射技术判定注解的存在,例如:
通过反射获取注解属性值,例如:
2.4:
Java中的注解如何定义?(@interface)
Java中注解的作用是什么?(描述类,属性,方法,参数用于赋予这些成员特点的含义)
Java中注解与Annotation接口是什么关系(is a 的关系,继承关系)
如何获取类及成员上状态为运行时有效的注解对象?(反射技术)
3拆箱封箱
3.1:概述
Java中不仅提供了8种基本数据类型,还为了更好的体现面向对象特性,专门提供了 8种基本数据类型对应的对象封装类型。Jdk1.5之后为了更好的实现基本数据类型和8种封装类型之间的转换,提供自动封箱和拆箱机制。
3.2:自动封箱
自动拆箱就是将基本类型自动转换为对象数据类型,例如:
其中,对于等号右边的值会自动调用Integer.valueOf方法转换为对象类型。
3.3:自动拆箱
自动拆箱就是对象类型自动转换为基本数据类型,例如:
说明,无论是自动拆箱,还是自动封装,为Java编程过程中的类型转换提供了很好便利。
4枚举类型
4.1:概述
枚举是JDK1.5以后推出的一种新的类型(特殊的类),主要用于更加严格的约束变量类型,例如现有一个产品对象,此对象有一个性别属性,请问此属性的类型如何定义呢,还有现在有一订单,我们如何定义订单状态呢?在实际项目中这些类型的定义我们通常会借助枚举进行实现。
4.2:枚举的定义
Java中的枚举借助enum关键字进行定义,例如JDK中的枚举:
枚举类型,通常应用于值固定的实例类型,例如一周七天,代码如下:
其中:Week中MONDAY, TUESDAY等都属于枚举的实例,这些实例都是在类加载时创建,
可通过枚举类名直接访问,例如Week.MONDAY。
4.3:枚举类型应用
定义一性别枚举类,例如:
在定义产品类型时,应用性别枚举类,例如:
定义含有带参构建函数的枚举类型,例如:
定义会员类,应用Sex枚举类型,例如:
枚举应用测试,案例如下:
5可变参数
5.1:概述
JDK5中允许方法最后一个参数使用带三个点的类型进行定义,在进行方法调用时可传入任意个数的实际参数,其目标主要是用于简化方法名相同,参数类型也相同,但参数个数不同的一系列方法的定义。
5.2:应用实践
可变参数应用案例,关键代码如下:
说明,可变参数只能应用在方法参数列表中的最后一个参数的定义上。
JDK8简介
1.1:概述:
Java 8由Oracle从2014年3月18日发布,此版本是自Java 5(发布于2004年)之后的一个重量级版本,也是java发展史上的一个里程碑式的版本。这个版本在JVM、编译器、库、Java语法特性等方面都做了很大改进,同时在语言的表达力、简洁性等个方面也有了很大的提高。目前几乎所有的JAVA 框架也都已升级到支持JDK8。
1.2:新特性介绍
Java 8 这个版本提供了很多实用的新特性,针对接口推出了接口默认方法,接口静态方法以及函数式接口,同时为了简化代码编写,推出了 lambda 表达式,为了增强对数据的操作,还定义了 Stream 操作等。这个版本目前是市场上一个应用最广泛,也是最重要的一个版本。
2JDK8接口新特性
2.1:概述
JDK8 中对接口规范进行了新的定义,允许在接口中定义默认方法(使用 default 关键字修饰),静态方法,同时还推出了函数式接口(使用@FunctionInterface注解描述)设计。
2.2:应用场景
基于 JDK8 中接口新规范的定义,不仅可以扩展接口新功能(新的标准),还能保持接口向前兼容的特性。例如Java API中的集合操作规范。
2.3:快速入门分析
default 方法设计及实现
JDK8中为了对接口业务进行扩展,但又不影响实现类,提供了默认方法。此类型的方法实用default关键字修饰,可以有方法体的实现。例如list接口中的sort方法:
说明:一个接口中可以有多个默认方法,在实现类中可以有选择的对方法进行重写;
static 方法设计及实现,JDK8的接口中还支持静态方法;
函数式接口设计及实现:Java8引入了一个是函数式接口(Functional Interfaces),此接口使用 @FunctionalInterface修饰,并且此接口内部只能包含一个抽象方法。说明函数式接口推出的目的主要是为了配合后续Lambda表达式的应用。
2.4:应用案例增强分析及实现
在JDK中的java.util.function包中定义了大量函数式接口。常用的有如下几个:
消费型接口:
函数式接口。
判定式接口
供给式接口
JDK8接口的新特性
1.接口中所有的方法都是抽象的吗?要看版本
2.什么是接口默认方法?(接口中由default关键字描述的方法,可以有方法体)
3.JDK为什么要推出默认方法?扩展接口功能,而且还不需要修改实现
4.一个接口中可以有多个默认方法吗?(可以)
5.接口默认方法,在接口的实现类中必须要重写吗? 不是
6.接口中可以有静态方法吗?可以,JDK8或更新版本
7.什么是函数式接口?@functionInterface注解描述的接口
8.函数式接口最大的特点是什么?接口中最多有一个抽象方法
9.函数式接口中可以有默认方法或镜头方法吗?可以
JDK8中Lambda表达式应用
3.1:概述
Java中的Lambda表达式是JDK8中的一种新特性,它允许我们将一段代码(这段代码可以理解为一个接口的实现)当成参数传递给某个方法,然后进行业务处理,这种方式更像是一种函数式编程风格,可以让代码结构更简洁,开发效率更高。
3.2:应用场景
Java中的Lambda为JAVA编程注入了函数式编程思想,在迭代操作,映射操作,聚合操作等很多方面的实现上做出了很大努力。并从语法角度简化了程序员对特定代码的编写,通过底层的类型推断,方法引用等特性让代码表现的更加优雅。现在已成为我们编程过程中常用的一种编程方式。
3.3:快速入门分析
最简单的Lambda表达式可由逗号分隔的参数列表、->符号和语句块组成,例如:
在上面这个代码中的参数 e 的类型是由编译器推理得出的,你也可以显式指定该参数的类型,例如:
如果 Lambda 表达式需要更复杂的语句块,则可以使用花括号将该语句块括起来,类似于Java中的函数体,例如:
lambda 表达式可以让代码编写更加简洁。我们先来思考下普通的函数或方法具备的几个元素:
▪ 访问修饰符。
▪ 返回值类型。
▪ 方法名。
▪ 参数列表。
▪ 代码块。
在lambda 表达式应用过程中,你应该也注意到了,一般只有两个元素:
其中“->” 将参数列表与函数主体分离,旨在对给定参数进行处理。函数的主体可能
是一条或多条语句。例如其常见结构如下:
Lambda表达式有返回值,返回值的类型也由编译器推理得出。如果Lambda表达式中的语句块只有一行,则可以不用使用return语句,下列两个代码片段效果相同:
3.4:应用案例增强实现
结合对lambda表达式基本语法的认识,通过如下案例在对lambda表达式继续强化分
析和实现。案例1:构建一个线程对象,执行Runnable类型的任务。传统方式的实现,其关键代码如下:
基于JDK8中的Lambda表达式实现方式,对传统方式线程对象的创建进行简化,其关
键代码如下:
案例2:通过lambda表达式演示排序过程中代码的简化。
定义一字符串数组,然后对字符串数组中的内容,按字符串元素的长度对其进行排序。代码如下:
基于JDK8中的Lambda表达式,对排序如上排序方案的代码实现过程进行简化,关键代码如下:
JDK8中方法引用
4.1:概述
方法引用是用来直接引用类方法、实例方法或者构造方法的一种新的方式。这里要特别强调一点的是“方法引用”提供的是一种对方法的引用而不是执行方法的方式,简单点理解的话就是可以将方法作为参数进行传递,我们还可以将方法引用理解为 lambda 的一种深层表达。
4.2:应用场景
方法引用是一种更简洁易懂的 Lambda 表达式,操作符是双冒号"::",也可以将方法引用看成是一个更加紧凑,易读的Lambda表达式。
4.3:快速入门分析
方法引用是一种更简洁易懂的 Lambda 表达式,操作符是双冒号"::",也可以将方法引用定义一个list集合,然后基于Lambda表达式迭代集合中的内容进行输出,关键代码如下:
基于方法引用的方式,输出list集合中的具体内容的,然后与传统Lambda表达式方式,进行对比分析,关键代码如下:
说明:当你要访问的接口方法与执行的方法引用参数相同,返回值也相同即可直接使用方法引用。
4.4:应用案例增强分析
JDK8方法的引用可分为如下几类:
▪ 构造方法引用。
格式:ClassName::new,应用默认构造函数
▪ 类静态方法引用。
格式:ClassName::static_method。
▪ 类实例方法引用。
格式:ClassName::method,方法不能带参数。
▪ 对象实例方法引用。格式:对象实例::method,方法不能带参数。
5 JDK8中Stream API应用
5.1:概述
Stream 作为 Java 8 的一大亮点,它与 java.io 包里的 InputStream 和 OutputStream 是完全不同的概念。Java 8 中的 Stream 是对集合(Collection)对象功能的增强,它专注于对集合对象进行各种非常便利、高效的聚合操作,或者大批量数据操作。
5.2:应用场景
在当今这个数据爆炸的时代,数据来源多样化、数据海量化,很多时候不得不脱离 RDBMS,以底层返回的数据为基础进行更上层的数据统计。而原有 Java 的集合 API 中,仅仅有极少量的辅助型方法,更多的时候是程序员需要用 Iterator 来遍历集合,然后完成相关的聚合应用逻辑。这是一种远不够高效而且相对比较笨拙的方法。在 JDK8 中使用Stream 对象,不仅丰富了在业务层面对数据处理的方式,还可以让代码更加简洁、易读和高效。
5.3:快速入门分析
我们在使用Stream对象时,一般会按照如下为三个步骤进行操作:
▪ 第一步:创建Stream流对象。
▪ 第二步:Stream流中间操作。
▪ 第三步:Stream流终止操作。
Stream对象的操作过程,可通过图-1进行进一步分析。

Steam对象简易应用,代码如下:
5.4:应用案例增强分析
▪ Stream 对象创建。
Stream对象的创建,常见方式有如下几种:
1) 借助Collection接口中的stream()或parallelStream()方法。
2) 借助Arrays类中的stream(...)方法。
3) 借助Stream类中的of(...)方法。
4) 借助Stream类中的iterator和generator方法(无限操作)。
Stream对象创建,案例分析如下:
▪ Stream 中间操作。
Stream 对象创建以后可以基于业务执行一些中间操作,但这些操作的结果需要借助终止操作进行输出,案例分析如下:
初始条件:给定 list 集合作为 Stream 操作的对象,代码如下:
对数据进行过滤:
限定操作(limit):
跳过操作(skip):
去重操作(distinct):
排序操作(sorted):底层基于内部比较器 Comparable 或外部 Comparator 比较器进行比对。
映射操作(map):
▪ Stream 终止操作。
Stream 终止操作是 Stream 的结束操作,案例分析如下:
案例:初始条件定义,给定一个 list 集合:
match 操作:
find 操作:
count 操作:
求最大,最小值:
forEach 迭代操作:
Reduce(规约)操作:
Collector(收集)操作:
JDK8中新日期对象应用
6.1:概述
在Java8之前,日期和时间的管理一直是令Java开发者很痛苦的一个的问题。java.util.Date、java.util.Calendar,SimpleDateFormat 都一直没有很好解决这个问题。故此,Java8引入了一套全新的日期时间处理 API,新的 API基于 ISO标准日历系统,解决了以前日期和时间类的很多弊端问题。
6.2:应用场景
Java8 中的时间处理 API 定义在 java.time 包中,这些 API 具备不可变且线程安全特性,具备准确和灵活特性。所以现在基本可以使用这组 API 替换所有原有历史版本中时间 API 的应用。
7.3:快速入门分析
▪ Instant 时间戳对象应用,默认是 0 时区,比北京少 8 个时区。
获取时间间隔。
▪ LocalDate 日期对象,不包含具体时间。
▪ LocalTime 时间对象,不包含日期。
▪ LocalDateTime 包含了日期和时间对象,没有时区信息。
▪ ZoneDateTime 包含时区的完整日期时间对象,偏移量以 UTC 时间为基准。
6.4:应用案例增强分析
项目中我们经常会用到日期类型转换,在JDK8中的实现方式如下:
说明:jdk8之前的日期与字符串之间的转换通常会借助 SimpleDateFormat对象,但是此对象线程不安全,通常要借助 ThreadLocal对象,保证 SimpleDateFormat对象每个线程一份。
8 JDK8新特性总结
8.1 重难点分析
▪ 接口新特性?(默认方法,函数式接口,静态方法)
▪ Lambda 表达式?(函数式,简化语法,开发效率)
▪ 方法引用(MethodReference)
▪ Stream 应用实践分析?(数据操作更方便,创建,中间操作,终止操作)
▪ 常用新的日期对象应用?(灵活,安全,准确)
8.2 FAQ 分析
▪ 常用的函数式接口有哪些?(消费型,供给型,判定型,函数式)
▪ Lambda 表达式关键作用是什么?(简洁,开发效率)
▪ Stream 应用中有哪些常用操作?(过滤,统计,映射,规约,收集,...)
▪ JDK8 中常用的日期 API 有哪些?(Instant,LocalDateTime,DateTimeFormater)
▪ …...