java8实战读书笔记:数值流、Stream创建与Optional类的使用

简介: java8实战读书笔记:数值流、Stream创建与Optional类的使用

本节将详细介绍Java8中的数值流、流的创建与Optional类的使用。

image.png

不知大家还记不得,在介绍函数式编程接口中为了避免基础数据类型的装箱/拆箱带来的性能损耗,特意为函数式接口引入了基础数据类型的函数式编程接口,例如IntPredicate、LongPredicate、DoublePredicate。同样,流API也考虑到基本数据类型的装箱/拆箱会带来性能损耗,引入了数值流,例如IntStream、LongStream、DoubleStream。


1.1 原始数据特化流


java8中提供了3个原始数据特化流,分别为IntStream、LongStream、DoubleStream。本文将以IntStream进行讲解,其他流类似,只是数据类型分别代表Long或Double。


1.1.1 映射到数据流


首先我们还是从一个示例开始本节的学习:计算菜单中所有菜品的卡路里之和。

1public static void test_reduce_caluli(List<Dish> menu) {
2    int calories = menu.stream()
3                    .map(Dish::getCalories)
4                    .reduce(0, Integer::sum);
5    System.out.println("菜品中的总卡路里:" + calories);
6}

上面包含了一个基本数据类型的装箱/拆箱动作,java8的流API提供了mapToInt方法,直接返回int类型的流

我们先稍微看一下mapToInt的方法声明:

1IntStream mapToInt(ToIntFunction<? super T> mapper)
2

接受一个T->int的函数式编程接口,直接返回IntStream流对象,而且IntStream本身提供了一些常用的聚合函数,例如sum。


使用IntStream来实现计算菜单中所有菜品的卡路里之和,其示例如下:

1public static void test_reduce_caluli_intStream(List<Dish> menu) {
2    int calories = menu.stream()
3            .mapToInt(Dish::getCalories)
4            .sum();
5    System.out.println("菜品中的总卡路里:" + calories);
6
7}


1.1.2 转换回对象流


使用了特化流例如IntStream后,将不能再自动转换为其对应的封装对象流Stream< T >了,我们可以随意从IntStream对象中对应的通用方法的函数声明,例如IntStream#map函数的声明如下:

1IntStream map(IntUnaryOperator mapper);

只能接受int -> int的函数式编程接口,如果想将IntStream转回到Stream< Integer >,该如何处理呢?


IntStream提供了boxed()方法来实现将基础数据类型转换回对应的包装类型的流。


1.1.3 常用函数


Stream中定义的方法,IntStream也可以使用,例如map、flatMap、distinict等,IntStream除这些之外,还提供了常用的聚合函数,例如sum、min、max、average(平均数)。

1OptionalDouble average();
2OptionalInt max();
3OptionalInt min();
4int sum();

有关Optional相关的类将在下文详细介绍。


另外除了上面提到的聚合函数,IntStream还提供了两个与数值范围的方法:

1public static IntStream range(int startInclusive, int endExclusive);
2public static IntStream rangeClosed(int startInclusive, int endExclusive);

rangeClosed与range的区别就是rangeClosed包含结束边界,举一个简单示例如下:

1public static void test_range() {
2    long count = IntStream.range(1,100)
3                        .filter( i -> i % 2 == 0 )
4                         .count();
5    System.out.println("count:" + count);
6
7}

计算【1,100)中包含的偶数个数,将输出49。如果将range(1,100)修改为rangeClosed(1,100),在输出的个数为50。

image.png


2.1 通过值构建流


java 8的Stream提供了两个重载的of函数来显示的构建流,其声明如下:

1public static<T> Stream<T> of(T t)
2public static<T> Stream<T> of(T... values)
3


2.2 通过数组构建流


通过Arrays.stream构建流,其声明如下:


Arrays#stream

1public static <T> Stream<T> stream(T[] array)
2


2.3 通过文件流


可以通过文件流创建流,在java.nio.file.Files类中定义了如下创建流的方法。

1public static Stream<Path> list(Path dir) throws IOException
2public static Stream<Path> walk(Path start, int maxDepth, FileVisitOption... options)
3public static Stream<Path> walk(Path start, FileVisitOption... options)
4public static Stream<Path> find(Path start, int maxDepth,BiPredicate<Path, BasicFileAttributes> matcher,   
5    FileVisitOption... options)
6public static Stream<String> lines(Path path, Charset cs)
7public static Stream<String> lines(Path path) throws IOException
8

下面我们举一个示例:找出一个文件中不同词的个数。

1public static void test_file_stram() {
 2    long uniqueWords = 0;
 3    try(Stream<String> lines = Files.lines(Paths.get("d:/tmp/words.txt"), Charset.defaultCharset())) {  
 4        uniqueWords = lines.flatMap(line -> Arrays.stream(line.split("" )))
 5                .distinct()
 6                .count();
 7
 8        System.out.println("不重复字符个数:" + uniqueWords);
 9    } catch (IOException e) {
10        e.printStackTrace();
11    }
12}


2.4 函数生成流:创建无限流


Stream API提供了两个静态方法从函数生成流:iterate、generate,我们先来看一下其函数声明:

1public static<T> Stream<T> iterate(final T seed, final UnaryOperator<T> f)
2public static<T> Stream<T> generate(Supplier<T> s)
3


2.4.1 iterate


iterate方法的第一个参数类型为T,表示其初始值,第二个参数如下:

b413f61a005fd59e3859651fe606e4a5.jpg

即其函数式声明为为T-T。其示例如下:

1public static void test_iterate() {
2    Stream.iterate(0, a -> a + 2)
3            .limit(10)
4            .forEach(System.out::println);
5}

注意:由于是无限流,故千万记得使用limit截断流,否则会无限循环下去。


2.4.2 generate


其参数为Supplier< T >,其定义如下:

96e3ce9f1fc85b5d211a1be184358416.png

即构造一个T类型的对象,举例如下:

1public static void test_iterate() {
2    Stream.iterate(0, a -> a + 2)
3            .limit(10)
4            .forEach(System.out::println);
5}


2.5 集合对象的stream


这个在前面的示例中用的最多,就不做过多介绍。

image.png

为了更优雅的处理null值,避免空指针错误,java8中引入Optional类。

b7da52362435700d25de7897fc32f918.jpg

接下来对这些方法一一做个介绍。


  • public static< T> Optional< T> empty()
    创建一个Optional对象,其内部持有的对象为null。
  • public static < T > Optional< T > of(T value)
    使用value的值,创建一个Optional对象。
  • public static < T > Optional< T > ofNullable(T value)
    使用v去创建一个Optional对象,如果value为null,则返回empty()。
  • public T get()
    从Optional对象获取内嵌的对象,如果为空,则抛出NoSuchElementException。
  • public boolean isPresent()
    判断Optional对象中包含的值是否存在。
  • public void ifPresent(Consumer consumer)
    如果Optional包裹的对象存在,则消费该对象。Consumer的函数式编程接口:T -> void。
  • public Optional< T > filter(Predicate predicate)
    如果Optional中包裹的对象为空,则返回自身,否则如果包裹的对象满足predicate表达式,则返回自身,否则返回empty()。
  • public< U > Optional< U > map(Function mapper)
    如果Optional对象中包裹的对象为空,则返回empty(),否则运用(T-U),包裹U,当然如果U为空,则返回empty()。
  • public< U > Optional< U > flatMap(Function> mapper)
    如果Option对象中包裹的对象为空,则返回empty(),否则使用对Optional中的包裹的对象value应用Function,最终返回Optional对象。
  • public T orElse(T other)
    返回Optional中包裹的对象,如果其值为空,则返回other。
  • public T orElseGet(Supplier other)
    返回Optional中包裹的对象,如果其值为空,则返回 Supplier函数式编辑接口中创建的值。
  • public < X extends Throwable> T orElseThrow(Supplier< ? extends X> exceptionSupplier) throws X
    返回Optional中包裹的对象,如果其值为空,则抛出自定义一次,由Supplier函数式编程接口返回。


其示例代码如下:


1public static void test_option() {
2Map<String, String> data = new HashMap<>();
3Optional<String> value = Optional.ofNullable(data.get("userName"));
4// 如果存在userName值,则输出
5value.ifPresent(System.out::println);  
6}


本文就介绍到这里了,本文详细介绍了java8中的数值流、Stream的创建以及java8中Optional类的使用。


相关文章
|
22天前
|
负载均衡 算法 关系型数据库
大数据大厂之MySQL数据库课程设计:揭秘MySQL集群架构负载均衡核心算法:从理论到Java代码实战,让你的数据库性能飙升!
本文聚焦 MySQL 集群架构中的负载均衡算法,阐述其重要性。详细介绍轮询、加权轮询、最少连接、加权最少连接、随机、源地址哈希等常用算法,分析各自优缺点及适用场景。并提供 Java 语言代码实现示例,助力直观理解。文章结构清晰,语言通俗易懂,对理解和应用负载均衡算法具有实用价值和参考价值。
大数据大厂之MySQL数据库课程设计:揭秘MySQL集群架构负载均衡核心算法:从理论到Java代码实战,让你的数据库性能飙升!
|
4天前
|
数据采集 自然语言处理 Java
Playwright 多语言一体化——Python/Java/.NET 全栈采集实战
本文以反面教材形式,剖析了在使用 Playwright 爬取懂车帝车友圈问答数据时常见的配置错误(如未设置代理、Cookie 和 User-Agent),并提供了 Python、Java 和 .NET 三种语言的修复代码示例。通过错误示例 → 问题剖析 → 修复过程 → 总结教训的完整流程,帮助读者掌握如何正确配置爬虫代理及其它必要参数,避免 IP 封禁和反爬检测,实现高效数据采集与分析。
Playwright 多语言一体化——Python/Java/.NET 全栈采集实战
|
3天前
|
存储 安全 Java
【高薪程序员必看】万字长文拆解Java并发编程!(4-1):悲观锁底层原理与性能优化实战
目录4. JVM字节码文件4.1. 字节码文件-组成4.1.1. 组成-基础信息4.1.1.1. 基础信息-魔数4.1.1.2. 基础信息-主副版本号4.1.2. 组成-常量池4.1.3. 组成-方法4.1.3.1. 方法-工作流程4.1.4. 组成-字段4.1.5. 组成-属性4.2. 字节码文件-查看工具4.2.1. javap4.2.2. jclasslib4.2.3. 阿里Arthas
10 0
|
3天前
|
安全 Java 程序员
【高薪程序员必看】万字长文拆解Java并发编程!(6-2):从CAS无锁机制到Atomic原子类实战指南
🌟 ​🌟今天给大家带来的是 ​💻⚡在这篇文章中,我们将一起探索:🔹 ​的底层原理,它是如何通过 ​实现无锁并发的?🔹 ​的终极对决,为什么高并发场景下CAS性能更优?🔹 ​的陷阱与解决方案——和实战演示!🔹 ​​(LongAdder等)的使用场景与性能对比🔹 危险的 ​黑魔法:为什么阿里禁止使用却又是并发库的基石?无论你是:✅ ​​(BATJ高频考点)✅ ​​(如何设计百万级计数器)✅ ​​(从Java代码到CPU指令的全链路分析)这篇文章都会让你收获满满!✨。
15 0
|
3天前
|
安全 Java 程序员
【高薪程序员必看】万字长文拆解Java并发编程!(6-1):从CAS无锁机制到Atomic原子类实战指南
🌟 ​🌟今天给大家带来的是 ​💻⚡在这篇文章中,我们将一起探索:🔹 ​的底层原理,它是如何通过 ​实现无锁并发的?🔹 ​的终极对决,为什么高并发场景下CAS性能更优?🔹 ​的陷阱与解决方案——和实战演示!🔹 ​​(LongAdder等)的使用场景与性能对比🔹 危险的 ​黑魔法:为什么阿里禁止使用却又是并发库的基石?无论你是:✅ ​​(BATJ高频考点)✅ ​​(如何设计百万级计数器)✅ ​​(从Java代码到CPU指令的全链路分析)这篇文章都会让你收获满满!✨。
12 0
|
3月前
|
存储 监控 Java
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
246 60
【Java并发】【线程池】带你从0-1入门线程池
|
1月前
|
Java 中间件 调度
【源码】【Java并发】从InheritableThreadLocal和TTL源码的角度来看父子线程传递
本文涉及InheritableThreadLocal和TTL,从源码的角度,分别分析它们是怎么实现父子线程传递的。建议先了解ThreadLocal。
77 4
【源码】【Java并发】从InheritableThreadLocal和TTL源码的角度来看父子线程传递
|
17天前
|
Java
java 多线程异常处理
本文介绍了Java中ThreadGroup的异常处理机制,重点讲解UncaughtExceptionHandler的使用。通过示例代码展示了当线程的run()方法抛出未捕获异常时,JVM如何依次查找并调用线程的异常处理器、线程组的uncaughtException方法或默认异常处理器。文章还提供了具体代码和输出结果,帮助理解不同处理器的优先级与执行逻辑。
|
3天前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
21 0
|
2月前
|
存储 网络协议 安全
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
120 23

热门文章

最新文章