jdk8中的StreamAPI

1.实体类

package com.zy.model;

import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;

import java.io.Serializable;

@Data
@Builder
@AllArgsConstructor
public class Employee implements Serializable {

    private Integer id;
    private String name;
    private Integer age;
    private double salary;

}

备注,该实体类采用了JetBrains的插件,IDEA编译器需要先导入插件(链接:http://plugins.jetbrains.com/plugin/6317-lombok-plugin)

并在pom.xml中引入Jar包

    <dependency>
      <groupId>org.projectlombok</groupId>
      <artifactId>lombok</artifactId>
      <version>1.16.20</version>
      <scope>provided</scope>
    </dependency>

2.Stream 的操作步骤

/*

* 1. 创建 Stream
*
* 2. 中间操作
*
* 3. 终止操作
*/

2.1创建Stream

package com.zy.stream;

import com.zy.model.Employee;
import org.junit.Test;

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

public class TestStreamAPI01 {

    // 1.创建stream流的方式
    @Test
    public void test1(){
        // 1.1第一种方式:通过Collection系列集合提供的Stream或paralleStream
        List<Object> list = new ArrayList<>();
        // 串行流,按顺序
        Stream<Object> streamList1 = list.stream();
        // 并行流,不按顺序
        Stream<Object> parallelStream = list.parallelStream();

        // 1.2第二种方式:通过Arrays中的静态方法获取数组流
        Employee[] employee = new Employee[10];
        Stream<Employee> streamArray = Arrays.stream(employee);

        // 1.3第三种方式:通过Stream类中的of()方法创建流
        Stream<String> stringStream = Stream.of("a", "b", "c");

        // 1.4创建无限流
        // 1.4.1第一种方式:迭代
        Stream.iterate(0, (x) -> (x + 1))
                .limit(20)
                .forEach(System.out::println);
        System.out.println("=================");
        // 1.4.2第二种方式:生成
        Stream.generate(()->Math.random())
                .limit(10)
                .forEach(System.out::println);
    }

}

2.2Stream的中间操作

package com.zy.stream;

import com.zy.model.Employee;
import org.junit.Test;

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

public class TestStreamAPI02 {

    // 2.中间操作
    /**
     *
     * 筛选与切片
     * filter--接收Lambda,从流中排除某些元素
     * limit(n)--截断流,使其元素不超过给定数量n
     * skip(n)--跳过元素,返回一个扔掉了前n个元素的流,若元素不足n个,则返回null,与limit(n)互补
     * distinct--筛选,通过流所生成的hashCode(),和equals()去除重复元素
     *
     * map--接收Lambda表达式,将元素转换成其他形式或提取信息,接收一个函数作为参数,该函数会应用到每个元素上,并将其映射成一个新的元素.
     * flatMap--接收一个函数作为参数,将流中的每个值都换成另一个流,然后把所有流凑成一个流
     * map与flatMap的关系类似于List中的add()与addALl()
     *
     * sorted--自然排序
     *
     */

    List<Employee> emps = Arrays.asList(
            new Employee(102, "李四", 59, 6666.66),
            new Employee(101, "张三", 18, 9999.99),
            new Employee(103, "王五", 28, 3333.33),
            new Employee(104, "赵六", 8, 7777.77),
            new Employee(104, "赵六", 38, 7777.77),
            new Employee(104, "赵六", 38, 7777.77),
            new Employee(105, "田七", 38, 5555.55)
    );

    // 筛选元素
    @Test
    public void fn1(){
        // 内部迭代,由StreamAPI完成
        Stream<Employee> employeeStream = emps.stream()
                .filter((e) -> e.getAge() > 30);
        // 一次性执行完所有内容,即"惰性求值"
        employeeStream.forEach(System.out::println);
    }

    // 只选前几个元素
    @Test
    public void fn2(){
        emps.stream()
                .filter((e)->e.getSalary()>6000)
                .limit(3)
                .forEach(System.out::println);
    }

    // 跳过前几个元素
    @Test
    public void fn3(){
        emps.stream()
                .filter((e)->e.getAge()>20)
                .skip(2)
                .forEach(System.out::println);

    }

    // 去重
    @Test
    public void fn4(){
        emps.stream()
                .distinct()
                .forEach(System.out::println);
    }

    // map
    @Test
    public void fn5(){
        emps.stream()
                .map(Employee::getName)
                .forEach(System.out::println);
    }

    @Test
    public void fn6(){
        List<String> list = Arrays.asList("aaa", "bbb", "ccc", "ddd");
        Stream<Character> characterStream = list.stream()
                .flatMap(TestStreamAPI02::filterCharacter);
        characterStream.forEach(System.out::println);
    }

    public static Stream<Character> filterCharacter(String str){
        List<Character> list = new ArrayList<>();
        for(Character character : str.toCharArray()){
            list.add(character);
        }
        return list.stream();
    }

    // 排序
    @Test
    public void fn7(){
        List<String> list = Arrays.asList("b", "c", "a", "m", "d");
        list.stream()
                .sorted()
                .forEach(System.out::println);
    }

    @Test
    public void fn8(){
        emps.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.3Stream的终止操作

package com.zy.stream;

import com.zy.model.Employee;
import org.junit.Test;

import java.util.Arrays;
import java.util.List;
import java.util.Optional;

public class TestStreamAPI03 {

    //3. 终止操作
    /*
        allMatch——检查是否匹配所有元素
        anyMatch——检查是否至少匹配一个元素
        noneMatch——检查是否没有匹配的元素
        findFirst——返回第一个元素
        findAny——返回当前流中的任意元素
        count——返回流中元素的总个数
        max——返回流中最大值
        min——返回流中最小值
     */
    List<Employee> emps = Arrays.asList(
            new Employee(102, "李四", 59, 6666.66),
            new Employee(101, "张三", 18, 9999.99),
            new Employee(103, "王五", 28, 3333.33),
            new Employee(104, "赵六", 8, 7777.77),
            new Employee(104, "赵六", 38, 7777.77),
            new Employee(104, "赵六", 38, 7777.77),
            new Employee(105, "田七", 38, 5555.55)
    );

    // 终止操作集锦
    @Test
    public void fn1(){
        boolean b = emps.stream()
                .allMatch((e) -> e.getAge() > 7);
        System.out.println("allMatch()================="+b);

        boolean b1 = emps.stream()
                .anyMatch((e) -> e.getAge() > 70);
        System.out.println("anyMatch()================="+b1);

        boolean b2 = emps.stream()
                .noneMatch((e) -> e.getAge() == 9);
        System.out.println("noMatch()================"+b2);

        Optional<Employee> optional = emps.stream()
                .sorted((e1, e2) -> e1.getAge().compareTo(e2.getAge()))
                .findFirst();
        System.out.println("findFirst()================" + optional.get());

        Optional<Employee> any = emps.stream()
                .filter((e)->e.getAge()>30)
                .findAny();
        System.out.println("findAny()============="+any.get());

        long count = emps.stream()
                .count();
        System.out.println("count()========="+count);

        Optional<Employee> max = emps.stream()
                .max((e1, e2) -> Double.compare(e1.getSalary(), e2.getSalary()));
        System.out.println("max()==================="+max);

        Optional<Employee> min = emps.stream()
                .min((e1, e2) -> Double.compare(e1.getSalary(), e2.getSalary()));
        System.out.println("min()====================="+min);

    }
}

原文地址:https://www.cnblogs.com/psy-code/p/9461665.html

时间: 2024-10-09 12:25:10

jdk8中的StreamAPI的相关文章

JDK7中匿名内部类中使用局部变量要加final,JDK8中不需要,但jdk会默认加上final

今天看书的时候看到了局部内部类,书上说局部内部类可以访问局部变量,但是必须是final的.因为局部变量在方法调用之后就消失了,使用final声明的话该局部变量会存入堆中,和内部类有一样的声明周期.但是我写了一个局部内部类,竟然可以访问非final的局部变量,请问这是什么回事呢.ps:我的jdk是8 难道和这个有关系? public class jubuneibulei { public void p(int a, int b){ class te{ void print(){ System.ou

JDK7与JDK8中HashMap的实现

JDK7中的HashMap HashMap底层维护一个数组,数组中的每一项都是一个Entry transient Entry<K,V>[] table; 我们向 HashMap 中所放置的对象实际上是存储在该数组当中: 而Map中的key,value则以Entry的形式存放在数组中 static class Entry<K,V> implements Map.Entry<K,V> { final K key; V value; Entry<K,V> next

forEach与jdk8中的lambda, Stream

增强for循环 :forEach 反编译后可以看到实际使用的仍然是Iterator+while遍历的 forEach的优点是写法简单,缺点是不能使用xxx.remove(e)或者iter.remove(), 如果有删除的需要,还是老老实实用Iterator 对于数组的遍历,使用的是length遍历 参考: https://stackoverflow.com/questions/85190/how-does-the-java-for-each-loop-work https://docs.orac

JDK8中JVM堆内存划分

一:JVM中内存 JVM中内存通常划分为两个部分,分别为堆内存与栈内存,栈内存主要用运行线程方法 存放本地暂时变量与线程中方法运行时候须要的引用对象地址. JVM全部的对象信息都 存放在堆内存中.相比栈内存,堆内存能够所大的多,所以JVM一直通过对堆内存划分 不同的功能区块实现对堆内存中对象管理. 堆内存不够最常见的错误就是OOM(OutOfMemoryError) 栈内存溢出最常见的错误就是StackOverflowError.程序有递归调用时候最easy发生 二:堆内存划分 在JDK7以及其

Hashmap在JDK8中的提升

HashMap使用key的hashCode()和equals()方法来将值划分到不同的桶里.桶的数量通常要比map中的记录的数量要稍大,这样 每个桶包括的值会比较少(最好是一个).当通过key进行查找时,我们可以在常数时间内迅速定位到某个桶(使用hashCode()对桶的数量进行取模) 以及要找的对象. 这些东西你应该都已经知道了.你可能还知道哈希碰撞会对hashMap的性能带来灾难性的影响.如果多个hashCode()的值落到同一个桶内的 时候,这些值是存储到一个链表中的.最坏的情况下,所有的

JDK8中Java调用Javascript脚本引擎动态定义与执行代码

import java.lang.*; import java.util.Arrays; import java.util.List; import javax.script.Invocable; import javax.script.ScriptEngine; import javax.script.ScriptEngineManager; public class ScriptEngineTest { public static void main(String[] args) throw

通过字节码分析JDK8中Lambda表达式编译及执行机制

关于Lambda字节码相关的文章,很早之前就想写了,[蜂潮运动]APP 产品的后端技术,能快速迭代,除了得益于整体微服架构之外,语言层面上,也是通过Java8的lambda表达式的运用以及rxJava响应式编程框架,使代码更加简洁易维护,调用方式更加便捷.本文将介绍JVM中的方法调用相关的字节码指令,重点解析JDK7(JSR-292)之后新增的invokedynamic指令给lambda表达式的动态调用特性提供的实现机制,最后再探讨一下lambda性能方面的话题. 方法调用的字节码指令 在介绍i

JDK8中HashMap

引用别人的一句话: JDK1.6,JDK1.7中,HashMap采用位桶+链表实现,即使用链表处理冲突,同一hash值的链表都存储在一个链表里.但是当位于一个桶中的元素较多,即hash值相等的元素较多时,通过key值依次查找的效率较低.而JDK1.8中,HashMap采用位桶+链表+红黑树实现,当链表长度超过阈值(8)时,将链表转换为红黑树,这样大大减少了查找时间. 红黑树:根,叶子节点全是黑的,红色节点的两个子节点都是黑色.(从每个叶子到根的所有路径上不能有两个连续的红色节点): 红黑树和AV

2、JDK8中的HashMap实现原理及源码分析

本篇提纲.png 本篇所述源码基于JDK1.8.0_121 在写上一篇线性表的文章的时候,笔者看的是Android源码中support24中的Java代码,当时发现这个ArrayList和LinkedList的源码和Java官方的没有什么区别,然而在阅读HashMap源码的时候,却发现Android中的Java与官方版的出入略大,遂不得不转而用Eclipse导入jdk源码阅读,这里不得不吐槽一句,用惯了IDEA的快捷键,Eclispe还真是用不习惯~~好了,接下来我们言归正传: 一.什么是Has