JDK8新特性之Stream API(三)

JDK8新特性之Stream API(三)

JAVA8中有两个最为重要的改变第一个是Lambda表达式,这个在JDK8新特性之Lambda表达式(一)JDK8新特性之方法引用(二)这两篇文章中介绍过,另一个则是Stream API。

Stream API 介绍

Stream API(java.util.stream)是Java8中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找、过滤和映射等操作,简而言之,Stream API提供了一种高效且易于使用的处理数据的方式。

Stream和Collection集合的区别

Collection是一种静态的内存数据结构简单说就是一个存放数据的容器,而Stream是有关计算的。前者主要是面向内存的,存储在内存中,后者主要是面向CPU,通过CPU实现计算。

Stream操作的三个步骤

  1. 创建Stream:一个数据源(如集合、数组),获取一个流
  2. 中间操作:一个中间操作链,对数据源的数据进行处理
  3. 终止操作:一旦执行终止操作,就执行中间操作链,并产生结果,之后不再被使用(意思就是之前所创建的stream已经无法使用了,要想继续操作必须重新执行三个步骤)

Stream 特点

  1. Stream自己不会存储元素(它不是一个容器)
  2. 不会改变源对象,相反他们会返回一个持有结果的新Stream(比如上图的操作1完成后会返回一个持有结果的新的stream)
  3. Stream操作是延迟执行的,这意味着它们会等到需要结果的时候才执行(只有在做终止操作的时候才会执行中间操作,换句话说如果不执行终止操作的话中间一些列操作是不会执行的)

Stream Api的使用

上面说了这么多都是概念类的东西,下面我们通过代码来解释上面的概念

步骤一:创建Stream

首先根据上面所说的Stream操作的三个步骤,第一个步骤自然就是创建Stream了,创建Stream主要有三种方式,通过集合创建、通过数组创建、通过Stream的of创建,由于这里仅仅只是创建Stream所以大家看不到效果,具体效果要等到终止操作那个步骤才能体现。
通过集合创建Stream

 @Test
    public void test01(){
        List<String> list = new ArrayList<>();
        list.add("张三");
        list.add("李四");
        list.add("王五");
        // 返回一个顺序流,顺序流就会安装集合的顺序进行操作
        Stream<String> stream = list.stream();
        // 返回一个并行流,并行流的意思就好比是多个同时操作,并不会保证顺序
        Stream<String> stringStream = list.parallelStream();
    }

通过数组创建Stream

 @Test
    public void test02(){
        Integer[] arr = new Integer[]{3,5,23,55,12};
        Stream<Integer> stream = Arrays.stream(arr);
    }

通过Stream的of

 @Test
    public void test03(){
        Stream<Integer> stream = Stream.of(34,12,23,2,4);
    }
Stream的中间操作

下面介绍stream中间操作的一些方法,首先我们创建一些数据集来方便我们完成中间操作,先创建一个User类,并重写toString、hashCode、equals方法

public class User {
    private String username;
    private int age;

    public User(String username, int age) {
        this.username = username;
        this.age = age;
    }

    public String getUsername() {
        return username;
    }

    public int getAge() {
        return age;
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        User user = (User) o;
        return age == user.age &&
                Objects.equals(username, user.username);
    }

    @Override
    public int hashCode() {
        return Objects.hash(username, age);
    }

    @Override
    public String toString() {
        return "User{" +
                "username='" + username + '\'' +
                ", age=" + age +
                '}';
    }
}

存放user数据的list

public class UserInfos {
    public static List<User> userList = new ArrayList<>();

    static {
        userList.add(new User("詹姆斯",36));
        userList.add(new User("韦德",28));
        userList.add(new User("科比",24));
        userList.add(new User("安东尼",34));
        userList.add(new User("保罗",31));
    }

}
筛选与切片

filter(Predicate p):接收Lambda,从流中排除某些元素
distinct():筛选,通过流所生成元素的hashCode()和equals()去除重复元素
limit(long maxSize):截断流,使其元素不超过给定数量
skip(long n):跳过元素,返回一个扔掉了前n个元素的流。若流中元素不足n个,则返回一个空流。与limit(n)互补

因为要执行终止操作才会执行中间操作链,所以下面例子加上了终止操作.foreach(),这个是遍历操作,关于终止操作后面会介绍
filter(Predicate p):接收Lambda,从流中过滤某些元素

    /**
     * 过滤掉年龄小于30岁的人
     */
    @Test
    public void test1() {
        List<User> userList = UserInfos.userList;
        System.out.println(Arrays.toString(userList.toArray()));
        System.out.println("=========================================");
        // 1.获取stream流
        Stream<User> stream = userList.stream();
        // 2.中间操作,每次中间操作都返回一个新的stream
        Stream<User> userStream = stream.filter(user -> user.getAge() > 30);
        // 3.终止操作
        userStream.forEach(user -> System.out.print(user+","));
    }

结果:

[User{username='詹姆斯', age=36}, User{username='韦德', age=28}, User{username='科比', age=24}, User{username='安东尼', age=34}, User{username='保罗', age=31}, User{username='韦德', age=28}]
=========================================
User{username='詹姆斯', age=36},User{username='安东尼', age=34},User{username='保罗', age=31},

limit(n):截断流,使其元素不超过给定数量

/**
     * 只获取前2个元素
     */
    @Test
    public void test2()
    {
        List<User> userList = UserInfos.userList;
        System.out.println(Arrays.toString(userList.toArray()));
        // 1.获取stream流
        Stream<User> stream = userList.stream();
        // 2.中间操作,每次中间操作都返回一个新的stream:limit,限制元素个数
        Stream<User> userStream = stream.limit(2);
        // 3.终止操作
        userStream.forEach(user -> System.out.print(user+","));
    }

结果:

[User{username='詹姆斯', age=36}, User{username='韦德', age=28}, User{username='科比', age=24}, User{username='安东尼', age=34}, User{username='保罗', age=31}, User{username='韦德', age=28}]
=========================================
User{username='詹姆斯', age=36},User{username='韦德', age=28},

skip(n):跳过元素,这个和limit(n)正好相反,返回一个扔掉了前n个元素的流,若流中元素不足n个,则返回一个空流

 /**
     * 去掉前4个元素
     */
    @Test
    public void test3()
    {
        List<User> userList = UserInfos.userList;
        System.out.println(Arrays.toString(userList.toArray()));
        System.out.println("=========================================");
        // 1.获取stream流
        Stream<User> stream = userList.stream();
        // 2.中间操作,每次中间操作都返回一个新的stream:skip,跳过指定元素个数
        Stream<User> userStream = stream.skip(4);
        // 3.终止操作
        userStream.forEach(user -> System.out.print(user+","));
    }

结果:

[User{username='詹姆斯', age=36}, User{username='韦德', age=28}, User{username='科比', age=24}, User{username='安东尼', age=34}, User{username='保罗', age=31}, User{username='韦德', age=28}]
=========================================
User{username='保罗', age=31},User{username='韦德', age=28},

distinct():筛选,通过流所生成元素的hashCode()和equals()去除重复元素

    /**
     * 去掉重复元素
     */
    @Test
    public void test4()
    {
        List<User> userList = UserInfos.userList;
        System.out.println(Arrays.toString(userList.toArray()));
        System.out.println("=========================================");
        // 1.获取stream流
        Stream<User> stream = userList.stream();
        // 2.中间操作,每次中间操作都返回一个新的stream:distinct(),去掉重复元素
        Stream<User> userStream = stream.distinct();
        // 3.终止操作
        userStream.forEach(user -> System.out.print(user+","));
    }

结果:

[User{username='詹姆斯', age=36}, User{username='韦德', age=28}, User{username='科比', age=24}, User{username='安东尼', age=34}, User{username='保罗', age=31}, User{username='韦德', age=28}]
=========================================
User{username='詹姆斯', age=36},User{username='韦德', age=28},User{username='科比', age=24},User{username='安东尼', age=34},User{username='保罗', age=31},
Stream中间操作——映射

关于映射操作这里只讲两个常用的方法,map(Function f)和flatMap(Function f),同样为了看到效果我们会在下面代码展示的时候加上终止操作.foreach()

map(Function f):接收一个函数作为参数,该函数会被应用到每个元素上,并将其映射成一个新的元素

/**
     * 将集合中的字符转成大写字符
     */
    @Test
    public void test5(){
        String[] arr = {"wade","james","kobe","paul","anthony"};
        System.out.println(Arrays.toString(arr));
        List<String> strList = Arrays.asList(arr);
        //1.获取stream
        Stream<String> stream = strList.stream();
        //2.中间操作,map映射:将元素转成大写
        Stream<String> stringStream = stream.map(str -> str.toUpperCase());
        // 终止操作
        stringStream.forEach(str -> System.out.print(str+" "));
    }

结果:

[wade, james, kobe, paul, anthony]
WADE JAMES KOBE PAUL ANTHONY 

在上面的基础上我们再多加一个过滤的中间操作

 /**
     * 将集合中的字符转成大写字符
     */
    @Test
    public void test6(){
        String[] arr = {"wade","james","kobe","paul","anthony"};
        System.out.println(Arrays.toString(arr));
        List<String> strList = Arrays.asList(arr);
        //1.获取stream
        Stream<String> stream = strList.stream();
        //2.中间操作,map映射:将元素转成大写
        Stream<String> stringStream = stream.map(str -> str.toUpperCase());
        // 中间操作,filter过滤,去掉名字是paul的元素
        Stream<String> newStream = stringStream.filter(str -> !str.equalsIgnoreCase("paul"));
        // 终止操作
        newStream.forEach(str -> System.out.print(str+" "));
    }

简化版本

/**
     * 将集合中的字符转成大写字符
     */
    @Test
    public void test7(){
        String[] arr = {"wade","james","kobe","paul","anthony"};
        System.out.println(Arrays.toString(arr));
        List<String> strList = Arrays.asList(arr);
       strList.stream()
               .map(str -> str.toUpperCase())
               .filter(str -> !str.equalsIgnoreCase("paul"))
               .forEach(str -> System.out.print(str+" "));
    }

结果:

[wade, james, kobe, paul, anthony]
WADE JAMES KOBE ANTHONY 

flatMap(Function f):接收一个函数作为参数,将流中的每个值都换成另一个流,然后把所有流连接成一个流
关于flatMap其实还是挺不好理解的,这里其实讲的也不好,大家就看一下大概的区别吧,具体区别是什么可以去搜一搜,因为我也不是特别懂

/**
     * 将字符串list中的所有字符串转成一个个字符,比如:
     * ["hello","world"] -> ['h','e','l','l','o','w','o','r','l','d']
     */
    @Test
    public void test8() {
        // 创建一个字符串数组
        String[] strArr = new String[]{"curry", "durant", "allen", "bosh"};
        List<String> strList = new ArrayList<>(Arrays.asList(strArr));
        System.out.println("=================使用中间操作:map()完成=============");
        // 1. 获取stream
        Stream<String> stream1 = strList.stream();
        // 2. 中间操作map
        Stream<Stream<Character>> streamStream = stream1.map(str -> toCharStream(str));
        // 3. 由于上一步操作完返回的是Stream<Stream<Character>>并不是Stream<Character>,因此我们遍历的时候需要两次foreach,嵌套
        streamStream.forEach(characterStream -> {
            characterStream.forEach(character -> System.out.println(character));
        });
        System.out.println("=================使用中间操作:flatMap()完成=============");
        // 1.获取stream
        Stream<String> stream2 = strList.stream();
        // 2. 中间操作flatMap,注意这里和上面map操作时返回的stream是不一样的,这也是flatMap的作用
        Stream<Character> characterStream = stream2.flatMap(str -> toCharStream(str));
        // 返回的是 Stream<Character>可以直接进行遍历
        characterStream.forEach(character -> System.out.println(character));
    }

    /**
     * 将字符串中的多个字符构成的集合转成对应的stream实例
     *
     * @param str
     * @return
     */
    public Stream<Character> toCharStream(String str) {
        List<Character> characterList = new ArrayList<>();
        char[] chars = str.toCharArray();
        for (char c : chars) {
            characterList.add(c);
        }

        return characterList.stream();
    }

结果:

=================使用中间操作:map()完成=============
c
u
r
r
y
d
u
r
a
n
t
a
l
l
e
n
b
o
s
h
=================使用中间操作:flatMap()完成=============
c
u
r
r
y
d
u
r
a
n
t
a
l
l
e
n
b
o
s
h

Stream中间操作——排序

关于排序这里也只有两个方法分别是:
sorted():产生一个新流,按自然方式排序
sorted(comparator com) :产生一个新流,按照比较器方式排序
同样为了看到效果我们会在下面代码展示的时候加上终止操作.foreach()
sorted():产生一个新流,按自然方式排序

 @Test
    public void test9()
    {
        List<Integer> list = new ArrayList<>(Arrays.asList(3,12,7,9,22,6,1));
        // 简写方式
        list.stream().sorted().forEach(num -> System.out.print(num+" "));
    }

结果:

1 3 6 7 9 12 22 

sorted(comparator com) :产生一个新流,按照比较器方式排序

 @Test
    public void test10() {
        // User和UserInfos在最开始的时候新建的数据
        List<User> userList = UserInfos.userList;
        System.out.println(Arrays.toString(userList.toArray()));
        System.out.println("=========================================");
        // 通过比较器的方式根据年龄给user排序
        userList.stream()
                .sorted((user1, user2) -> {
                    if (user1.getAge() > user2.getAge()) {
                        return 1;
                    } else if (user1.getAge() < user2.getAge()) {
                        return -1;
                    } else {
                        return 0;
                    }
                }).forEach(num -> System.out.print(num + " "));
    }

结果:

1 3 6 7 9 12 22 
Stream的终止操作

关于终止操作,只有执行了终止操作才会进行进行一系列的中间操作也就是说其实中间操作是延迟的,其次就是在终止操作完成后stream流将不能再次使用。其实在上面的stream中间操作时为了看到效果我们都用了终止操作,只不过我们用的都是forEach(),下面将介绍Stream的一些终止操作。

Stream的终止操作——匹配与查找

allMatch(Predicate p):检查是否匹配所有元素
anyMatch(Predicate p):检查是否匹配至少一个元素
noneMatch(Predicate p):检查是否没有匹配所有元素
findFirst():返回第一个元素
findAny():返回当前元素的任意元素
count():返回流中元素的总个数
max(Comparator c):返回流中最大值
min(Comparator c):返回流中最小值
forEach(consumer c):内部迭代

前面用了很多次的forEach,我们先来看看它的用法,以及说明一下在终止操作完成后stream流将不能再次使用是啥意思

/**
     * forEach(consumer c):内部迭代
     */
    @Test
    public void test11() {
        List<String> strList = new ArrayList<>(Arrays.asList("curry", "durant", "allen", "bosh"));
        System.out.println("遍历集合所有元素:");
        Stream<String> stream = strList.stream();
        stream.forEach(str -> System.out.println(str));
        // 由于forEach是终止操作,这时候stream已经不能再次使用了,下面这个代码执行是会报错:java.lang.IllegalStateException: stream has already been operated upon or closed
//        stream.forEach(str -> System.out.println(str));
    }

结果:

遍历集合所有元素:
curry
durant
allen
bosh
    /**
     * allMatch(Predicate p):检查是否匹配所有元素
     * anyMatch(Predicate p):检查是否匹配至少一个元素
     * noneMatch(Predicate p):检查是否没有匹配所有元素
     * findFirst():返回第一个元素
     * findAny():返回当前元素的任意元素
     * count():返回流中元素的总个数
     * max(Comparator c):返回流中最大值
     * min(Comparator c):返回流中最小值
     * forEach(consumer c):内部迭代
     */
    @Test
    public void test11() {
        List<String> strList = new ArrayList<>(Arrays.asList("curry", "durant", "allen", "bosh"));
        boolean flag1 = strList.stream().allMatch(str -> str.contains("a"));
        System.out.println("所有元素是否都含有字母a:" + flag1);

        boolean flag2 = strList.stream().anyMatch(str -> str.contains("a"));
        System.out.println("是否至少一个元素含有字母a:" + flag2);

        boolean flag3 = strList.stream().noneMatch(str -> str.contains("hh"));
        System.out.println("是否所有元素都不包含hh的元素:" + flag3);

        // 这里Optional不做介绍,大家可以自行百度
        Optional<String> first = strList.stream().findFirst();
        String element = first.get();
        System.out.println("集合中的第一个元素是:" + element);

        // 这里用parallelStream能体现效果
        Optional<String> any = strList.parallelStream().findAny();
        String el = any.get();
        System.out.println("集合中的任意一个元素是:" + el);

        long count = strList.stream().count();
        System.out.println("集合中的总个数是:" + count);


        List<Integer> intList = new ArrayList<>(Arrays.asList(3, 12, 7, 9, 22, 6, 1));
        // 1.匿名函数方式
        Optional<Integer> max = intList.stream().max(new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return Integer.compare(o1, o2);
            }
        });
        // 2.lambda表达式方式
//        intList.stream().max((num1,num2)->{
//            return Integer.compare(num1,num2);
//        });
        System.out.println("集合中的最大值是:" + max.get());

        // 1.匿名函数方式
        Optional<Integer> min = intList.stream().min(new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return Integer.compare(o1, o2);
            }
        });
        // 2.lambda表达式方式
//        intList.stream().max((num1,num2)->{
//            return Integer.compare(num1,num2);
//        });
        System.out.println("集合中的最小值是:" + min.get());
    }

结果:

所有元素是否都含有字母a:false
是否至少一个元素含有字母a:true
是否所有元素都不包含hh的元素:true
集合中的第一个元素是:curry
集合中的任意一个元素是:allen
集合中的总个数是:4
集合中的最大值是:22
集合中的最小值是:1
Stream的终止操作——规约

Optional<T> reduce(BinaryOperator<T> accumulator):可以将流中元素反复结合起来,得到一个值。返回Optional<T>

@Test
    public void test12(){
        List<Integer> list = Arrays.asList(1,2,3,4,5);
        /*
            给定初始值和集合中的元素进行求和:
            reduce操作其实内部进行的是:10+num1,(10+num1)+num2,(10+num1+num2)+num3....
         */
        Integer result = list.stream().reduce(10, (num1, num2) -> {
            return Integer.sum(num1, num2);
        });
        System.out.println(result);
    }

结果:25

T reduce(T identity, BinaryOperator<T> accumulator):可以将流中元素反复结合起来,得到一个值。返回T

@Test
    public void test13(){
        List<Integer> list = Arrays.asList(1,2,3,4,5);
        /*
            给定初始值和集合中的元素进行求和,和上一个例子不同的是它不需要传递一个初始值,而且返回的是个Optional
         */
        Optional<Integer> result = list.stream().reduce((num1, num2) -> {
            return Integer.sum(num1, num2);
        });
        System.out.println(result.get());
    

结果:15

Stream的终止操作——收集

collect(Collector c):将流转换成为其他形式。接收一个Collector接口的实现,用于给Stream中元素做汇总方法。
Collector接口中方法的实现决定了如何对流执行收集的操作(如收集到List、Set、Map),另外Collectors实用类提供了很多静态方法,可以方便地创建常见收集器时期,我们下面以常用的Collectors.toList和Collectors.toSet举例,其他的可以自行看API

 @Test
    public void test14() {
        List<String> list = Arrays.asList("a", "ac", "bc", "de", "ea", "a");
        System.out.println("原集合:" + list);
        //1.获取一个只包含字母a的字符串的List集合
        List<String> aLIst = list.stream().filter(str -> str.contains("a")).collect(Collectors.toList());
        System.out.println("只包含字母a字符串的新List集合:" + aLIst);

        //2.获取一个值包含字母a的字符串的Set集合
        Set<String> aSet = list.stream().filter(str -> str.contains("a")).collect(Collectors.toSet());
        System.out.println("只包含字母a字符串的新Set集合:" + aSet);


    }

结果:

原集合:[a, ac, bc, de, ea, a]
只包含字母a字符串的新List集合:[a, ac, ea, a]
只包含字母a字符串的新Set集合:[a, ac, ea]

总结

在使用Stream API的时候要注意先要了解Lambda表达式,如果对Lambda表达式还看的不是太懂了话可能对于Stream API的使用有一定困难,Stream API只是一个API,刚开始看的时候可能很不习惯甚至别扭,只要多练、多用、多看,看得多了练得多了用起来自然也就熟练了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容