欢迎您访问365答案网,请分享给你的朋友!
生活常识 学习资料

JAVA8新特性

时间:2023-06-11

司内小分享, boss让我整理一下JAVA8新特性, 噢噢噢噢!!!还有这好事儿, 正好趁着这个机会好好学习下、基于我不写笔记学不下去的习惯(分享也要有文档), 就出现了本篇内容、主要讲使用条件和方式, 原理请自行Google

上篇: Lambda表达式, Collector, Optional
下篇: Stream流, LocalDateTime

说明: 测试中生成box测试数据均为一下方法生成

public static List generateBoxes() { List boxes = new ArrayList<>(); boxes.add(new Box().setName("黄色箱子").setColor("yellow").setSize(5)); boxes.add(new Box().setName("绿色箱子").setColor("green").setSize(8)); boxes.add(new Box().setName("红色箱子").setColor("red").setSize(2)); boxes.add(new Box().setName("黑色箱子").setColor("black").setSize(6)); boxes.add(new Box().setName("紫色箱子").setColor("purple").setSize(10)); boxes.add(new Box().setName("蓝色箱子").setColor("blue").setSize(1)); return boxes;}

一、Lambda表达式 1.1 描述

函数式编程: 允许把函数作为基本运算单元, 可以作为方法的参数,又称闭包.
方法引用: 是指如果某个方法签名和接口恰好一致(只看参数类型和返回类型,不看方法名称,也不看类的继承关系),就可以直接传入方法引用。

1.2 要求 Lambda
函数式接口: 有且只有一个必须实现方法的接口(由于JAVA8中接口可以有了默认实现, 所以定义了一个 @FunctionalInterface 注解作为函数式接口的标记)
如:Comparator、Runnable以及四大函数式接口Consumer(返回void)、Function(返回映射泛型)、Predicate(返回boolean)、Supplier(返回定义的泛型)
Lambda表达式可以直接访问外部域变量, 但只能引用标记了 final 的外层局部变 量,不能lambda 内部修改定义在域外的局部变量.方法引用
如果某个方法签名和接口恰好一致,就可以直接传入方法引用四大函数式接口
简单的讲就是四类函数容器(容器的概念和思想很重要啊) 1.3 格式

Lambda表达式: “->”
方法引用: “::”

1.4 特征 Lambda:
可选类型声明: 不需要声明参数类型,编译器可以统一识别参数值。
可选的参数圆括号: 一个参数无需定义圆括号,但无参数或者多个参数需要定义圆括 号。
可选的大括号:如果主体包含了一个语句,就不需要使用大括号。
可选的返回关键字:如果主体只有一个表达式返回值则编译器会自动返回值,大括号需 要指定表达式返回了一个数值。方法引用
1)在使用lambda表达式时, 如果匿名函数的返回值类型和参数类型都已经确定, 则可以将任何符合要求的方法以方法引用的方式传入
2)第一不关心方法名; 第二不关心类的继承关系; 第三不关心方法实现 1.5 案例

以数组、集合排序为例

Lambda

private static void lambda() { System.out.println("源数据boxes_1:"); List boxes = GenerateUtil.generateBoxes(); System.out.println(boxes); System.out.println("java7排序:"); // 本质是创建匿名内部类 // 编译时会生成Xxx$n.class boxes.sort(new Comparator() { @Override public int compare(Box o1, Box o2) { return o1.getSize() - o2.getSize(); } }); System.out.println(boxes); System.out.println("======================================="); System.out.println("源数据boxes_1"); List boxes2 = GenerateUtil.generateBoxes(); System.out.println(boxes2); System.out.println("java8排序:"); // 本质是匿名函数 boxes2.sort((box1, box2) -> box1.getSize() - box2.getSize()); System.out.println(boxes2);// 特别的, 可以用来创建函数式接口的实例 Comparator comparator = (o1, o2) -> o1.getSize() - o2.getSize();}

方法引用

public static int ohMyGod(Integer num1, Integer num2) {// 为了与原数字排序方式混淆, 此处使用倒序排序 return num2 - num1;}private static void methodReference() { Integer[] numbers = {5, 8, 2, 3, 4, 1, 9, 0}; Arrays.sort(numbers, LambdaDemo::ohMyGod);// Arrays.sort(numbers, Math::max); System.out.println(Arrays.toString(numbers));}

如上: ohMyGod的方法返回值和参数与compare方法一致, 此处就可以使用方法引用, Math的max方法也符合要求, 也可以使用、这两个方法名字, 继承和实现关系, 方法实现都不相同, 但不影响使用.

二、Collector和Collectors 2.1 描述

Collector: 可变的聚合操作, 定义一个可变容器, 将新输入的元素添加到可变容器中(可以认为是可变容器的操作容器)
Collectors: 收集器工具类, 定义部分常用收集器

2.2 原理

有三个泛型:
T: 元素类型
A: 可变容器类型
R: 结果容器类型
有五个方法:
Supplier: 创建新的结果容器
Accumulator: 合并元素到结果容器中
Combiner: 合并两个结果容器
Finisher: 将结果容器转为最终表示
Characteristics: 给collector设置特征值

工作原理即使用Collector中定义的方法对内容进行可达到目的性的操作

2.3 特征

有三个特征值:
CONCURRENT: 表示结果容器只有一个
UNORDERED: 表示流中的元素无序
IDENTITY_FINISH: 表示中间结果容器类型与最终结果类型一致

2.4 案例

Collector

自定义一个实现

public class MyCollector implements Collector, List> {// 创建一个新容器 @Override public Supplier> supplier() { System.out.println("supplier execute"); return ArrayList::new; }// 向容器中添加元素 @Override public BiConsumer, T> accumulator() { System.out.println("accumulator execute"); return List::add; }// 容器合并 @Override public BinaryOperator> combiner() { System.out.println("combiner execute"); return (container1, container2) -> { container1.addAll(container2); return container1; }; }// 结果容器 @Override public Function, List> finisher() { System.out.println("finisher execute"); return (result) -> result; } @Override public Set characteristics() { return Collections.unmodifiableSet(EnumSet .of(Collector.Characteristics.IDENTITY_FINISH)); }}

自定义一个操作

public static R operate(T[] ts1, A ts2, Collector collector) { // 获取一个新容器 A a = collector.supplier().get(); BiConsumer accumulator = collector.accumulator(); // 向新容器中添加元素 for (T t : ts1) { // 遇到c时便不再继续添加 if ("c".equals(t)) { break; } accumulator.accept(a, t); } BinaryOperator combiner = collector.combiner(); // 合并容器操作 A apply = combiner.apply(a, ts2); Function finisher = collector.finisher(); // 容器转为结果操作 return finisher.apply(apply);}

执行

private static void myCollector() {// 创建一个数组和一个集合, 进行操作 String[] strings = {"a", "b", "c", "d"}; List strs = new ArrayList<>(Arrays.asList("e", "f", "g")); List operate = Operations.operate(strings, strs, new MyCollector<>()); System.out.println(operate);}

结果

Collectors
工具集不做过多介绍
常用的方法: toXxx, groupBy, joining, reducing等
大部分的容器操作都可以通过下篇的Stream流实现 三、Optional 3.1 描述

容器对象, 容器中的值可以为null
定义对象检测方法和对象处理方法, 用于进行对象检测(空值检测), 空值问题的解决方 案

3.2 案例 创建
Optional.of: 不允许为null 值的创建方式, 会报空指针异常
Optional.ofNullable: 允许为null 值的创建方式
Optional.empty: 值为null 的创建方式(Optional内部维护的空值对象)

Optional box = Optional.of(new Box());Optional box1 = Optional.ofNullable(new Box());Optional box2 = Optional.empty();

null值调用
orElse: 当前处理容器中的Optional值为null时, 则将orElse的参数作为返回值
orElseGet: 当前处理容器中的Optional值为 null时, 则将orElseGet中定义的方法返回值作为方法返回值
orElseThrow: 当前处理容器中的Optional值为null时, 则将orElseThrow中定义的方法返回的异常抛出

private static void orElse() { Box b = null; Box box1 = Optional.ofNullable(b).orElse(new Box().setName("1")); Box box2 = Optional.ofNullable(b).orElseGet(() -> new Box().setName("2")); Box box = Optional.ofNullable(b).orElseThrow(() -> new RuntimeException("box 不存在!"));}

映射值
map: 根据参数定义的函数, 获取值对象的相应映射值, 创建映射值的Optional容器并返回
flagMap: 参数定义的函数的返回值必须是Optional容器, 方法的返回值结果都已映射值对象的Optional容器

Optional box = Optional.of(new Box().setName("1"));Optional s1 = box.map((b) -> b.getName());Optional s2 = box.flatMap((b) -> Optional.ofNullable(b.getName()));

判空
isPresent: 判断空值
ifPresent: 判断空值, 在判断不为null时进行处理操作

Optional box = Optional.empty();Random random = new Random();int i = random.nextInt();if (i % 2 == 0) { box = Optional.of(new Box().setName("1"));}System.out.println(box.isPresent() ? "不为 null" : "为 null");box.ifPresent(System.out::println);

筛选
filter: 对容器中的值进行过滤; 如果存在值符合条件, 则返回这个Optional容器, 如果不存在, 返回Optional的empty对象

double random = Math.random() * 10;Optional box = Optional.of(new Box().setName("1").setSize((int) random));Optional box1 = box.filter(b -> b.getSize() > 4);System.out.println(box1);

空指针解决方案演示:

User user = new User();String s = Optional.ofNullable(user) .map(User::getAddress)// .orElse(new Address()) .map(Address::getProvince)// .orElseGet(Province::new) .flatMap((p -> { City city = new City(); p.setCity(city); return Optional.of(city); })) .map(City::getStreet) .orElseThrow(() -> new RuntimeException("容器中值为 null"));

方式一: 添加默认值的方式
方式二: 抛异常的方式(map方法在某次返回empry之后, 后续的所有操作都是对 empry的操作, 除非使用flatMap进行手动创建新的数据容器)

日 期 : 2022 − 02 − 12 color{#00FF00}{日期:2022-02-12} 日期:2022−02−12

Copyright © 2016-2020 www.365daan.com All Rights Reserved. 365答案网 版权所有 备案号:

部分内容来自互联网,版权归原作者所有,如有冒犯请联系我们,我们将在三个工作时内妥善处理。