Java8新特性——Stream API的创建方式及中间操作

简介: Java8新特性——Stream API的创建方式及中间操作

文章目录:


1.什么是StreamAPI

2.Stream API操作的三个步骤

2.1 创建Stream

2.2 中间操作

2.2.1 中间操作之筛选与切片

2.2.2 中间操作之映射

2.2.3 中间操作之排序

1.什么是StreamAPI?


Java8中有两大最为重要的改变。第一个是 Lambda 表达式;另外一个则是 m Stream API (java.util.stream.*)

Stream Java8 中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找、过滤和映射数据等操作。

使用Stream API 对集合数据进行操作,就类似于使用 SQL 执行的数据库查询。也可以使用 Stream API 来并行执行操作。简而言之,Stream API 提供了一种高效且易于使用的处理数据的方式。


(Stream) 到底是什么

是数据渠道,用于操作数据源(集合、数组等)所生成的元素序列。集合讲的是数据流讲的是计算

注意

1.    Stream 自己不会存储元素。

2.    Stream 不会改变源对象。相反,他们会返回一个持有结果的新Stream

3.    Stream 操作是延迟执行的。这意味着他们会等到需要结果的时候才执行。


2.Stream API操作的三个步骤


1.    创建Stream:一个数据源(如:集合、数组),获取一个流。

2.    中间操作:一个中间操作链,对数据源的数据进行处理。

3.    终止操作( ( 终端操作) ):一个终止操作,执行中间操作链,并产生结果。


2.1 创建Stream

Java8 中, Collection 接口被扩展,提供了两个获取流的方法

·       default Stream<E> stream() : 返回一个顺序流

·       default Stream<E> parallelStream() : 返回一个并行流


Java8中的 Arrays 的静态方法 stream() 可以获取数组流:

·       static <T> Stream<T> stream(T[] array): 返回一个流重载形式能够处理对应基本类型的数组

·       public static IntStream stream(int[] array)

·       public static LongStream stream(long[] array)

·       public static DoubleStream stream(double[] array)


可以使用静态方法 Stream.of(),通过显示值创建一个流。它可以接收任意数量的参数。

·       public static<T> Stream<T> of(T... values) : 返回一个流


可以使用静态方法 Stream.iterate() Stream.generate(),创建无限流。

·       迭代   public static<T> Stream<T> iterate(final T seed, final UnaryOperator<T> f)

·       生成   public static<T> Stream<T> generate(Supplier<T> s)

    //创建Stream
    @Test
    public void test1() {
        //1.Collection 提供了两个方法  stream() 与 parallelStream()
        List<String> list = new ArrayList<>();
        Stream<String> stream1 = list.stream();
        //2. 通过 Arrays 中的 stream() 获取一个数组流
        String[] strings = new String[10];
        Stream<String> stream2 = Arrays.stream(strings);
        //3. 通过 Stream 类中静态方法 of()
        Stream<String> stream3 = Stream.of("abc","666","???");
        //4. 创建无限流
        //迭代
        Stream<Integer> stream4 = Stream.iterate(0, (x) -> x + 2);
        stream4.limit(5).forEach(System.out::println);
        //生成
        Stream.generate(() -> Math.random())
                .limit(5)
                .forEach(System.out::println);
    }


2.2 中间操作


多个中间操作可以连接起来形成一个线,除非流水线上触发终止操作,否则间操作会执行理!

而在终止操作时一次性全部处理   常用的中间操作大体上可以分为三类:筛选与切片;映射;排序


2.2.1 中间操作之筛选与切片


下面通过一些代码案例来应用一下。  首先,我们准备一个Employee类,以及一个List集合。

package com.szh.java8;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
/**
 *
 */
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Employee {
    private Integer id;
    private String name;
    private Integer age;
    private Double salary;
    public Employee(Integer id) {
        this.id = id;
    }
    public Employee(Integer id,String name) {
        this.id = id;
        this.name = name;
    }
}
List<Employee> employees = Arrays.asList(
        new Employee(1001,"张三",26,6666.66),
        new Employee(1002,"李四",50,1111.11),
        new Employee(1003,"王五",18,9999.99),
        new Employee(1004,"赵六",35,8888.88),
        new Employee(1005,"田七",44,3333.33),
        new Employee(1005,"田七",44,3333.33),
        new Employee(1005,"田七",44,3333.33)
);

筛选年龄小于35岁的员工信息。  这里每次都是先 println 打印结果,然后进行比较,满足则输出。

    //内部迭代,在 Stream API 内部完成
    @Test
    public void test2() {
        //所有的中间操作不会做任何的处理
        Stream<Employee> stream = employees.stream()
                                    .filter((e) -> {
                                        System.out.println("测试中间操作");
                                        return e.getAge() <= 35;
                                    });
        //只有当做终止操作时,所有的中间操作会一次性的全部执行,称为“惰性求值”
        stream.forEach(System.out::println);
    }


筛选薪资大于5000的员工信息,但是只保留前两条。

    @Test
    public void test3() {
        employees.stream()
                .filter((e) -> {
                    System.out.println("短路....");
                    return e.getSalary() > 5000;
                })
                .limit(2)
                .forEach(System.out::println);
    }


筛选薪资大于5000的员工信息,但是扔掉前两条。

    @Test
    public void test4() {
        employees.stream()
                .filter((e) -> e.getSalary() > 5000)
                .skip(2)
                .forEach(System.out::println);
    }


对集合中的所有员工信息做去重处理。

    @Test
    public void test5() {
        employees.stream()
                .distinct()
                .forEach(System.out::println);
    }


2.2.2 中间操作之映射


这里仍然是借助上面案例中的Employee类、List集合完成。

首先对自定义的List集合中的所有字符串进行大写转换,同时对员工集合进行 name 字段的映射处理。

    @Test
    public void test1() {
        List<String> list = Arrays.asList("aaa","bbb","ccc","ddd","eee");
        list.stream()
            .map((str) -> str.toUpperCase())
            .forEach(System.out::println);
        System.out.println("--------------------------------------");
        employees.stream()
                .map(Employee::getName)
                .forEach(System.out::println);
    }


List集合中的每个字符串中的每个字符进行遍历输出。(使用map使用flatMap

这两种方式主要区别就在获取到Stream流之后,流中的泛型不一样了。


我们使用map了话,它会将strList集合中元素以"aaa""bbb""ccc" 这样的形式返回,此时Stream流的泛型就仍然是一个Stream流(在这个Stream流中才是我们想要的数据),也就是我们第一次对stream流进行遍历之后,拿到的是一个又一个的Character字符数组,然后还需要再对这个字符数组进行二次遍历(也就是代码上半部分中那两个 forEach)。


如果我们使用flatMap了话,它会将strList集合中元素以 "a'"b""c""d" 这样的形式返回,所以此时Stream流的泛型就只是一个Character字符数组,那么一次遍历就可以取出数据了(也就是代码下半部分的那个forEach)。

    @Test
    public void test2() {
        List<String> strList = Arrays.asList("aaa","bbb","ccc","ddd","eee");
        Stream<Stream<Character>> stream1 = strList.stream()
                .map(MyTest6::filterCharacter);
        stream1.forEach((sm) -> {
            sm.forEach(System.out::println);
        });
        System.out.println("----------------------------------------");
        Stream<Character> stream2 = strList.stream()
                .flatMap(MyTest6::filterCharacter);
        stream2.forEach(System.out::println);
    }
    private static Stream<Character> filterCharacter(String str) {
        List<Character> list = new ArrayList<>();
        for (Character character : str.toCharArray()) {
            list.add(character);
        }
        return list.stream();
    }


2.2.3 中间操作之排序


List集合中的字符串进行自然排序。对员工集合中的员工信息进行定制化排序(先按照年龄排序,年龄一样再按照姓名排序)。

    @Test
    public void test3() {
        List<String> strList = Arrays.asList("ccc","aaa","bbb","eee","ddd");
        strList.stream()
                .sorted()
                .forEach(System.out::println);
        System.out.println("----------------------------------------");
        employees.stream()
                .sorted((e1,e2) -> {
                    if (e1.getAge().equals(e2.getAge())) {
                        return e1.getName().compareTo(e2.getName());
                    } else {
                        return e1.getAge().compareTo(e2.getAge());
                    }
                })
                .forEach(System.out::println);
    }

相关文章
|
2天前
|
数据采集 JSON Java
Java爬虫获取微店快递费用item_fee API接口数据实现
本文介绍如何使用Java开发爬虫程序,通过微店API接口获取商品快递费用(item_fee)数据。主要内容包括:微店API接口的使用方法、Java爬虫技术背景、需求分析和技术选型。具体实现步骤为:发送HTTP请求获取数据、解析JSON格式的响应并提取快递费用信息,最后将结果存储到本地文件中。文中还提供了完整的代码示例,并提醒开发者注意授权令牌、接口频率限制及数据合法性等问题。
|
2天前
|
数据采集 存储 Java
Java爬虫获取微店店铺所有商品API接口设计与实现
本文介绍如何使用Java设计并实现一个爬虫程序,以获取微店店铺的所有商品信息。通过HttpClient发送HTTP请求,Jsoup解析HTML页面,提取商品名称、价格、图片链接等数据,并将其存储到本地文件或数据库中。文中详细描述了爬虫的设计思路、代码实现及注意事项,包括反爬虫机制、数据合法性和性能优化。此方法可帮助商家了解竞争对手,为消费者提供更全面的商品比较。
|
24天前
|
算法 Java 程序员
菜鸟之路Day06一一Java常用API
《菜鸟之路Day06——Java常用API》由blue编写,发布于2025年1月24日。本文详细介绍了Java中常用的API,包括JDK7的时间类(Date、SimpleDateFormat、Calendar)和JDK8新增的时间API(ZoneId、Instant、DateTimeFormatter等),以及包装类的使用。通过多个实例练习,如时间计算、字符串转整数、十进制转二进制等,帮助读者巩固所学内容,提升编程技能。文章强调了理论与实践结合的重要性,鼓励读者多做练习以提高学习效率。
76 28
|
6天前
|
缓存 Java 应用服务中间件
java语言后台管理若依框架-登录提示404-接口异常-系统接口404异常如何处理-登录验证码不显示prod-api/captchaImage 404 (Not Found) 如何处理-解决方案优雅草卓伊凡
java语言后台管理若依框架-登录提示404-接口异常-系统接口404异常如何处理-登录验证码不显示prod-api/captchaImage 404 (Not Found) 如何处理-解决方案优雅草卓伊凡
31 5
|
1月前
|
JSON Java 数据挖掘
利用 Java 代码获取淘宝关键字 API 接口
在数字化商业时代,精准把握市场动态与消费者需求是企业成功的关键。淘宝作为中国最大的电商平台之一,其海量数据中蕴含丰富的商业洞察。本文介绍如何通过Java代码高效、合规地获取淘宝关键字API接口数据,帮助商家优化产品布局、制定营销策略。主要内容包括: 1. **淘宝关键字API的价值**:洞察用户需求、优化产品标题与详情、制定营销策略。 2. **获取API接口的步骤**:注册账号、申请权限、搭建Java开发环境、编写调用代码、解析响应数据。 3. **注意事项**:遵守法律法规与平台规则,处理API调用限制。 通过这些步骤,商家可以在激烈的市场竞争中脱颖而出。
|
2月前
|
JSON Java Apache
Java基础-常用API-Object类
继承是面向对象编程的重要特性,允许从已有类派生新类。Java采用单继承机制,默认所有类继承自Object类。Object类提供了多个常用方法,如`clone()`用于复制对象,`equals()`判断对象是否相等,`hashCode()`计算哈希码,`toString()`返回对象的字符串表示,`wait()`、`notify()`和`notifyAll()`用于线程同步,`finalize()`在对象被垃圾回收时调用。掌握这些方法有助于更好地理解和使用Java中的对象行为。
|
2月前
|
Rust 安全 Java
Java Stream 使用指南
本文介绍了Java中Stream流的使用方法,包括如何创建Stream流、中间操作(如map、filter、sorted等)和终结操作(如collect、forEach等)。此外,还讲解了并行流的概念及其可能带来的线程安全问题,并给出了示例代码。
131 0
|
9月前
|
Java API 数据处理
Java 8新特性之Stream API详解
【2月更文挑战第22天】本文将深入探讨Java 8中引入的Stream API,这是一种基于函数式编程的新特性,用于处理集合数据。我们将详细介绍Stream的基本概念、操作方法以及在实际开发中的应用,帮助读者更好地理解和使用这一强大的工具。
|
9月前
|
Java API
Java 8新特性之Stream API详解
【5月更文挑战第30天】本文将详细介绍Java 8中的一个重要新特性——Stream API。Stream API是Java 8中引入的一种新的数据处理方式,它允许我们以声明式的方式处理数据,使得代码更加简洁、易读。文章将从Stream的基本概念、创建方式、常用操作以及实战案例等方面进行详细讲解,帮助读者深入理解并掌握Stream API的使用。
100 2
|
9月前
|
Java API
Java 8新特性之Stream API详解
【5月更文挑战第28天】本文将详细解析Java 8中的一个重要新特性——Stream API。Stream API是Java 8引入的一种新的数据处理方式,它允许我们以声明性和函数式的方式处理数据。我们将详细介绍Stream API的基本概念,使用方法,以及如何使用它来简化我们的代码。

热门文章

最新文章