java8新特性——StreamAPI

news2025/1/22 12:28:57

1.集合处理数据的弊端

当我们在需要对集合中的元素进行操作的时候,除了必需的添加,删除,获取外,最典型的操作就是集合遍历。

package com.wxj.streamapi;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

public class StreamTest01 {
    public static void main(String[] args) {
        //定义一个集合
        List<String> list = Arrays.asList("张三","周星驰","张三丰","星");
        List<String> list1 = new ArrayList<>();
        //1.获取所有 姓张的信息
        for (String s:list) {
            if(s.startsWith("张"))
            {
                list1.add(s);
            }
        }
        //2.获取名称长度为3的用户
        List<String> list2 = new ArrayList<>();
        for (String s : list1) {
            if(s.length() == 3){
                list2.add(s);
            }
        }
        //3.输出所有的用户信息
        for (String s : list2) {
            System.out.println(s);
        }
    }
}

上面代码针对于我们不同的需求,总是一次次的循环循环循环,这时我们希望有更加高效的处理方式,这时我们可以通过JDK8中提供的Stream API来解决这个问题了。

Stream更加优雅的解决方案:

package com.wxj.streamapi;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

public class StreamTest02 {
    public static void main(String[] args) {
        //定义一个集合
        List<String> list = Arrays.asList("张三","周星驰","张三丰","星");
        List<String> list1 = new ArrayList<>();
        //1.获取所有 姓张的信息
        //2.获取名称长度为3的用户
        //3.输出所有的用户信息
        list.stream()
                .filter(s -> s.startsWith("张"))
                .filter(s -> s.length() == 3)
                .forEach(s -> System.out.println(s));
    }
}	

上面的StreamAPI代码的含义:获取流,过滤张,过滤长度,逐一打印。代码相比上面的更简洁。

2.Stream流式思想概述

Stream API能让我们快速完成许多复杂的操作,如筛选,切片,映射,查找,去除重复,统计,匹配和归约。

3.Stream流的获取方式.

3.1根据Collection获取

首先,java.util.Collection 接口中加入了default方法stream,也就是说Collection接口下的所有的实现都可以通过stream方法来获取Stream流。

package com.wxj.streamapi;

import java.util.*;

public class StreamTest03 {
    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.stream();
        Set<String> set = new HashSet<>();
        set.stream();
        Vector vector = new Vector();
        vector.stream();
    }
}

但是Map接口没有实现Collection接口,那这时怎么办呢?这时我们可以根据Map获取对应的key value集合。.

3.2通过Stream的of方法

在实际开发中我们不可避免的会操作到数组中的数据,由于数组对象不可能添加默认方法,所以Stream接口中提供了静态方法of。

上面中的基本数据类型,int类型,把它当成了一个整体,输出的是地址,而Integer是一个包装类,将数组换成了集合去处理。

4.Stream常用方法

Stream注意事项(重要)

1.Stream只能操作一次

2.Stream方法返回的是新的流

3.Stream不调用终结方法,中间操作不会执行

4.match也是一个终结方法

4.1 forEach方法

forEach用来遍历流中的数据

该方法接受一个Consumer接口,会将每一个流元素交给函数处理

4.2 count方法

Stream流中的count方法用来统计流中元素个数的

该方法返回一个long值,代表元素的个数。无参。

4.3 filter方法

filter方法的作用是用来过滤数据的。返回复合条件的数据

可以通过filter方法将一个流转换成另一个子集流

该接口接受一个Predicate函数式接口参数作为筛选条件

输出的内容是包含a的

4.4 limit方法

limit方法可以对流进行截取处理,只取前n个数据

参数是一个long类型的数值,如果集合当前长度大于参数就进行截取,否则不操作,参数为负数报错。

输出

4.5 skip方法

如果希望跳过前面几个元素,可以使用skip方法获取一个截取以后的新流:

操作:

输出:

4.6 map方法

如果我们需要将流中的元素映射到另一个流中,可以使用map方法

该接口需要一个Function函数式接口参数,可以将当前流中的T类型数据转换成另一种R类型的数据

map方法通过方法引用将字符串转换成整型,Integer::parseInt

4.7 sorted方法

如果需要将数据排序,可以使用sorted方法:

在使用的时候可以根据自然排序,也可以通过比较强来指定对应的排序规则

4.8 distinct方法

如果要去掉重复数据,可以使用distinct方法

操作:

Stream流中的distinct方法对于基本数据类型是可以直接去重的,但是对于自定义类型,我们是需要hashCode和equals方法来移除重复元素。

4.9 match方法

如果需要判断数据是否匹配指定的条件,可以使用match相关的方法

使用:

注意match是一个终结方法

4.10 find方法

如果我们需要找到某些数据,可以使用find来实现

使用:

4.11 max和min方法

如果我们想要获取最大值和最小值,那么可以使用max和min

使用:

4.12 reduce方法

如果需要将所有数据归纳得到一个数据,可以使用reduce方法。

使用:

4.13 map和reduce的组合

在实际开发中我们经常会将map和reduce一块来使用

输出结果:

4.14 mapToInt方法

如果需要将Stream中的Integer类型转换成int类型,可以使用mapToInt方法

使用:

4.15 concat方法

如果有两个流希望合并成为一个流,那么可以使用Stream接口的静态方法concat

使用:

4.16 综合案例

定义两个集合,然后再集合中存储多个用户名称。然后完成如下的操作:

1.第一个队伍只保留姓名长度为3的成员

2.第一个队伍筛选之后只要前3个人

3.第二个队伍只要姓张的成员

4.第二个队伍筛选之后不要前两个人

5.将两个队伍合并成一个队伍

6.根据姓名创建Person对象

7.打印整个队伍的Person信息

实现:

package com.wxj.streamapi;
import com.wxj.jdk.lambda.domain.Person;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Stream;

public class StreamDemo {
    public static void main(String[] args) {
        List<String> list1 = Arrays.asList("炎柱","音柱","霞柱","恋柱","水柱","风柱","蛇柱","岩柱","虫柱","继国缘一");
        List<String> list2 = Arrays.asList("上弦一","上弦二","上弦三","上弦四","上弦五","上弦六","黑死牟","童磨","半天狗");
        //1.第一个队伍只保留姓名长度为3的成员
        //2.第一个队伍筛选之后只要前3个人
        Stream<String> stringStream1 = list1.stream().filter(s -> s.length() == 3).limit(3);
        //3.第二个队伍只要姓张的成员
        //4.第二个队伍筛选之后不要前两个人
        Stream<String> stringStream2 = list2.stream().filter(s -> s.startsWith("上")).skip(2);
        //5.将两个队伍合并成一个队伍
        //6.根据姓名创建Person对象
        //7.打印整个队伍的Person信息
        Stream.concat(stringStream1,stringStream2).map(n -> new Person(n)).forEach(System.out::println);
    }
}

输出结果:

5.Stream结果收集

5.1结果收集到集合中

实现:

输出:

5.2结果收集到数组中

Stream中提供了toArray方法来将结果放到一个数组中,返回值类型是Object[],如果我们要指定返回的类型,那么可以使用另一个重载的toArray(IntFunction)方法

5.3对流中的数据做聚合计算

当我们使用Stream流处理数据后,可以像数据库的聚合函数一样对某个字段进行操作,比如获得最大值,最小值,求和,平均值,统计数量

5.4对流中的数据做分组操作

当我们使用Stream流处理数据后,可以根据某个属性将数据分组

输出结果:

多级分组:

先根据name分组然后再根据成年未成年分组,

5.5对流中的数据做分区操作

Collectors,partitoningBy会根据值是否为true,把集合中的数据分割成两个列表,一个true列表,一个false列表。

输出结果:

5.5对流中的数据做拼接操作

Collectors,joining会根据指定的连接符,将所有的元素连接成一个字符串

6.并行的Stream流

6.1串行的Stream流

我们前面使用的Stream流都是串行,也就是一个线程上面执行。

输出:

6.2并行流

parallelStream其实就是一个并行执行的流,它通过默认的FprkJoinPool,可以提高多线程的效率。

6.2.1获取并行流

我们可以通过两种方式来获取并行流

1.通过List接口中的parallelStream方法来获取

2.通过已有的串行流转换为并行流(parallel方法)

实现:

6.2.2并行流操作

结果:

6.3并行流和串行流对比

我们通过for循环,串行Stream流,并行Stream流来对5亿个数字求和。来消耗时间。

通过案例我们可以看到parallelStream的效率是最高的。

Stream并行处理的过程会分而治之,也就是将一个大的任务切分成了多个小任务,这表示每个任务都是一个线程操作。

6.线程安全问题

在多线程的处理下,肯定会出现数据安全问题。

运行结果:

直接抛异常,或者执行结果不对。

针对这个问题,我们的解决方案有哪些呢?

1.加同步锁

2.使用线程安全的容器

3.将线程不安全的容器包装为线程安全的容器

4.通过Stream中的toArray方法或者Collect方法来操作

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/500779.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

E. Vasya and Good Sequences(异或)

Problem - E - Codeforces Vasya有一个由n个整数组成的序列a。 Vasya可以执行以下操作&#xff1a;从序列中选择一些数字&#xff0c;并交换其二进制表示中的任意一对位。例如&#xff0c;Vasya可以将数字6&#xff08;… 000000001102&#xff09;转换为3&#xff08;… 00000…

[数据库系统] 一、外键约束 (educoder)

1.任务&#xff1a;给表添加外码。 2.相关知识 需要掌握&#xff1a; (1)什么是外码&#xff1b; (2)怎么给表添加外码。 (1)什么是外码 外部关键字 外码是另一张表中的主码。**外码的主要作用是保持数据的一致性&#xff0c;完整性。 如图有两张表&#xff0c;classId 是T…

蓝晓转02上市价格预测

蓝晓转02 基本信息 转债名称&#xff1a;蓝晓转02&#xff0c;评级&#xff1a;A&#xff0c;发行规模&#xff1a;5.460645亿元。 正股名称&#xff1a;蓝晓科技&#xff0c;今日收盘价&#xff1a;88.16元&#xff0c;转股价格&#xff1a;92.73元。 当前转股价值 转债面值 …

10.集合

1.泛型 1.1泛型概述 泛型的介绍 ​ 泛型是JDK5中引入的特性&#xff0c;它提供了编译时类型安全检测机制 泛型的好处 把运行时期的问题提前到了编译期间避免了强制类型转换 泛型的定义格式 <类型>: 指定一种类型的格式.尖括号里面可以任意书写,一般只写一个字母.例如:…

深度学习模型部署的步骤和相关注意事项

文章目录 深度学习模型部署的步骤和相关注意事项什么是模型部署&#xff1f;步骤1&#xff1a;选择合适的部署环境步骤2&#xff1a;选择合适的部署框架步骤3&#xff1a;将模型转换为部署格式步骤4&#xff1a;创建API接口步骤5&#xff1a;部署模型总结 深度学习模型部署的步…

JWT漏洞基础

JWT漏洞基础 什么是JWTJWT漏洞介绍工具使用 身份认证(Authentication)又称鉴权&#xff0c;是指通过一定的手段&#xff0c;完成对用户身份的确认。认证的方式&#xff1a;sessioncookie、JWT、Token session认证的局限性 session认证机制需要配合cookie才能实现。由于cookie默…

极客时间- 数据结构与算法之美 - 王争 前 Google 工程师【学习笔记】

本文是 极客时间- 数据结构与算法之美 - 王争 前 Google 工程师。专栏学习笔记整理&#xff0c;课程链接&#xff1a;https://time.geekbang.org/column/intro/100017301?tabcatalog 01 | 为什么要学习数据结构和算法&#xff1f;面试业务开发工程师写出达到开源水平的框架才是…

大数据应用案例:如何在金融行业中利用数据挖掘实现风险控制和预测

引言 随着互联网和移动技术的发展&#xff0c;金融行业已经进入了数字化时代。大数据技术在金融行业的应用越来越广泛&#xff0c;其中最重要的应用就是风险控制和预测。本文将介绍如何利用数据挖掘技术在金融行业中实现风险控制和预测&#xff0c;以及一些成功的案例。 章节…

Android 系统的分区和文件系统(4)- Android 伪文件系统

声明 Android系统中有很多分区&#xff0c;每个分区内的文件系统一般都不同的&#xff0c;使用ADB进入系统/目录下可发现挂载这很多的目录&#xff0c;不同的目录中可来自不同的分区及文件系统&#xff1b;此篇参考了一些书籍及论文&#xff0c;仅供学习使用。只介绍大概理论&…

Valarrays

C标准库提供了一个class valarray用以进行数值数组的运算。 它声明于头文件<valarray> namespace std{template<class T> class valarray; //numeric array of type Tclass slice;template<class T> class slice_array; //slice out of a valarrayclass gs…

Java经典笔试题—day03

Java经典笔试题—day03 &#x1f50e;选择题&#x1f50e;编程题&#x1f95d;字符串中找出连续最长的数字串&#x1f95d;数组中出现次数超过一半的数字 &#x1f50e;结尾 &#x1f50e;选择题 (1)以下代码运行输出的是 public class Person{private String name "Pe…

怎么洗稿容易过稿-在线洗稿软件

自媒体洗稿软件 即使您是一位优秀的自媒体写作人员&#xff0c;也难免遇到让人头疼的撰写问题&#xff0c;例如无法处理大量原始文本、需要手动删除冗余信息、缺少时间针对每篇文章进行深入修改等问题。但是&#xff0c;现在有了我们的一款自媒体洗稿软件&#xff0c;您再也不需…

Android System crash DeadSystemException(Service/Activity/终极解决方案)

DeadSystemException&#xff1a; The core Android system has died and is going through a runtime restart. All running apps will be promptly killed. Android 核心系统服务已经死亡&#xff0c;正在重启中。全部正在运行的app即将被kill杀死。 更多请阅读&#xff0c;D…

Lecture 11:How versatile are self-supervised models

目录 Story 1: Cross-lingual Story 2: Cross-discipline Story 3: Pre-training with artificial data &#xff08;story1和story2的内容在前面课程中有讲过&#xff0c;这里笔记部分不再详述&#xff09; Story 1: Cross-lingual 多语言BERT具有跨语言的能力&#xff0…

Linux-Day01

Linux-Day01 课程内容 Linux简介Linux安装Linux常用命令 1. 前言 1.1 什么是Linux Linux是一套免费使用和自由传播的操作系统。说到操作系统&#xff0c;大家比较熟知的应该就是Windows和MacOS操作系统&#xff0c;我们今天所学习的Linux也是一款操作系统。 1.2 为什么要学…

12.IO流

1.字符流 1.1为什么会出现字符流【理解】 字符流的介绍 由于字节流操作中文不是特别的方便&#xff0c;所以Java就提供字符流 字符流 字节流 编码表 中文的字节存储方式 用字节流复制文本文件时&#xff0c;文本文件也会有中文&#xff0c;但是没有问题&#xff0c;原因是最…

消息队列中的事务消息

大家好&#xff0c;我是易安&#xff01;今天我们谈一谈消息队列中的事务消息这个话题。 一说起事务&#xff0c;你可能自然会联想到数据库。我们日常使用事务的场景&#xff0c;绝大部分都是在操作数据库的时候。像MySQL、Oracle这些主流的关系型数据库&#xff0c;也都提供了…

Java --- springboot2之异常处理

仅供参考 目录 一、异常处理 二、异常处理自动配置原理 三、异常处理流程 四、定制错误处理逻辑 4.1、自定义错误页 4.2、ControllerAdviceExceptionHandler处理全局异常 4.3、ResponseStatus自定义异常 4.4、Spring底层的异常&#xff0c;如 参数类型转换异常 4.5、自定义…

7种常见网络并发模型介绍

概述 对于网络服务器后端开发&#xff0c;为满足不同并发场景的需要&#xff0c;一般来说&#xff0c;不外乎几种常见的并发模型&#xff0c;除了一些教学场景常用的单线程、多进程&#xff08;线程&#xff09;的服务器实现外&#xff0c;生产用的服务器&#xff0c;一般都会…

Linux多路IO复用:epoll

1. epoll epoll是为克服select、poll每次监听都需要在用户、内核空间反复拷贝&#xff0c;以及需要用户程序自己遍历发现有变化的文件描述符的缺点的多路IO复用技术。 epoll原理 创建内核空间的红黑树&#xff1b; 将需要监听的文件描述符上树&#xff1b; 内核监听红黑树上…