流式编程是1.8中的新特性,基于常用的四种函数式接口以及Lambda表达式对集合类数据进行类似流水线一般的操作

流式编程分为大概三个步骤:获取流 → 操作流 → 返回操作结果

流的获取方式

这里先了解获取流的常用的两种方式,后面在进行流的操作

集合中获取流

众所周知Java中所有的集合都是Collection下的实现类,在Collection接口中就提供了获取流的方法:

public class ApplicationMain {
public static void main(String[] args) {
ArrayList<Integer> list = new ArrayList<>();
// 获取流
Stream<Integer> stream1 = list.stream();
// 获取流(多线程,大数据量下效率较高)
Stream<Integer> stream2 = list.parallelStream();
}
}

数组中获取流

针对数组Java中提供了一个Arrays工具类,我们可以将数组转换为集合在获取流

public class ApplicationMain {
public static void main(String[] args) {
Integer[] arr = {1, 2, 3, 4, 5};
List<Integer> list = Arrays.asList(arr);
Stream<Integer> stream1 = list.stream();
Stream<Integer> stream2 = list.parallelStream();
}
}

或者直接通过Arrays类获取到流

public class ApplicationMain {
public static void main(String[] args) {
Integer[] arr = {1, 2, 3, 4, 5};
Stream<Integer> stream = Arrays.stream(arr);
}
}

流的获取方式 总结与补充

方法名 说明
Collection.stream() 从集合中获取流
Collection.parallelStream() 从集合中获取流 ( 多线程 )
Arrays.stream(T[]) 从数组中获取流
Stream.of(T... values) 直接传入多个元素返回一个流
Stream.generate(Supplier s) Lambda返回的每个实例都是流中的一个元素
Stream iterate(final T seed, final UnaryOperator f) Lambda接收参数一个参数,返回一个结果作为元素,每次返回的结果都将作为下一个Lambda的参数 ( 迭代 )

操作流中的数据

假数据模拟

知道了如何获取到流之后,就要开始学习操作流了,在练习之间先写一个假的接口来模拟数据:

// 接口模拟数据
public class UserService {
public List<UserEntity> selectList() {
ArrayList<UserEntity> list = new ArrayList<>();
list.add(new UserEntity("老八", 32, '男', 8000));
list.add(new UserEntity("郭老师", 36, '女', 7000));
list.add(new UserEntity("卢本伟", 32, '男', 18000));
list.add(new UserEntity("张春德", 22, '男', 2800));
list.add(new UserEntity("大司马", 34, '男', 12000));
list.add(new UserEntity("老八", 32, '男', 8000));
list.add(new UserEntity("贾玲", 22, '女', 21000));
list.add(new UserEntity("周淑怡", 26, '女', 14800));
list.add(new UserEntity("PDD", 37, '男', 26300));
return list;
}
}
// UserEntity实体类
public class UserEntity { private String name;
private int age;
private char gender;
private int salary; public UserEntity(){}
public UserEntity(String name, int age, char gender, int salary) {
this.name = name;
this.age = age;
this.gender = gender;
this.salary = salary;
} public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public char getGender() {
return gender;
}
public void setGender(char gender) {
this.gender = gender;
}
public int getSalary() {
return salary;
}
public void setSalary(int salary) {
this.salary = salary;
} @Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
UserEntity that = (UserEntity) o;
return getAge() == that.getAge() &&
getGender() == that.getGender() &&
getSalary() == that.getSalary() &&
getName().equals(that.getName());
}
@Override
public int hashCode() {
return Objects.hash(getName(), getAge(), getGender(), getSalary());
} @Override
public String toString() {
return "{" +
"姓名='" + name + '\'' +
", 年龄=" + age +
", 性别=" + gender +
", 薪水=" + salary +
'}';
}
}

函数式接口复习

在操作流式编程之前先来复习一下函数式接口,这里以实现集合的过滤器为例:

public class ApplicationMain {

    public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
// 获取所有女性员工
ArrayList<UserEntity> result = filter(list, item -> item.getGender() == '女');
result.forEach(System.out::println);
} // 自定义集合过滤器
public static ArrayList<UserEntity> filter(List<UserEntity> list, Predicate<UserEntity> predicate) {
// 创建一个结果集
ArrayList<UserEntity> result = new ArrayList<>();
// 空值校验
if (list == null || list.size()==0)
return null;
// 遍历传入的集合,根据调用者制定的过滤规则进行判断,符合条件就添加到结果集中
for (UserEntity entity : list) {
if (predicate.test(entity)) {
result.add(entity);
}
}
return result;
} }

这样我们就获取到了一个 ArrayList 集合的过滤器

操作流的方法

流式编程操作流非常类似上面的代码,常用函数如下所示:

方法名 说明
filter() 循环集合中每个元素进行判断,返回false的元素会被过滤掉
limit() 截取方法,传入 int 类型的 n,从第一个元素开始只获取 n 个
skip() 跳过方法,传入 long 类型的 n,流将从第 n+1 个元素开始操作
distinct() 去重方法,去掉集合中重复的元素,只保留第一个
sorted() 排序方法,通过判断返回的 boolean 值作为参考进行排序
map() 这个方法比较特殊,后面用到会详细说明

使用流式编程需要了解他的特点:

  1. 我们通过流式编程操作集合是不会影响集合本身
  2. 流式编程的代码都是延迟执行的,只有在获取结果的时候才会执行

filter 过滤方法

// 获取到薪水大于10000的所有用户
public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
list.stream()
.filter(item->item.getSalary()>10000)
.forEach(System.out::println);
}
}

limit 截取方法

// 只获取结果中的5条数据(从首个开始截取)
public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
list.stream()
.limit(5)
.forEach(System.out::println);
}
}

skip 跳过方法

// 这里就跳过了2个元素,从"卢本伟"开始操作
public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
list.stream()
.skip(2)
.forEach(System.out::println);
}
}

distinct:去重方法

这里需要提一嘴,使用distinct方法操作的实体类必须复写equalshashCode方法

// 去重操作,可以看到两个老八只剩下一个
public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
list.stream()
.distinct()
.forEach(System.out::println);
}
}

sorted :排序方法

// 通过薪水对集合进行排序
public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
list.stream()
// 排序需要传入一个Comparator比较器,这里通过Integer直接获取
// 通过调换item1和item2的顺序实现升序降序的调整
.sorted((item1, item2)->Integer.compare(item1.getSalary(), item2.getSalary()))
.forEach(System.out::println);
}
}

map:生成新的数据

map同之前那几个稍微有些区别,类似filter或者sorted方法都是对集合进行编辑,而map可以将集合改变为一组新的数据:

// 通过map返回每个用户的姓名,将用户集合改变成了用户姓名集合
public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
list.stream()
.map(UserEntity::getName)
.forEach(System.out::println);
}
}

返回操作结果

返回操作结果是流式编程最后一步,也是最关键的一步,之前说过流式编程操作集合不会影响集合本身,那么想要获取结果就需要进行这最后一步,需要注意的是流式编程只有在返回操作结果的时候才会执行操作代码

之前使用的forEach就属于返回结果的代码,如果只调用了filter方法而没有调用返回结果,那么filter方法是不会执行的,可以使用下面这段代码进行测试

// 代码正常执行,如果将forEach移除就会发现filter中的打印语句同样没有执行
public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
list.stream()
.filter(item->{
System.out.println("过滤代码执行了");
return item.getSalary() > 10000;
}).forEach(System.out::println);
}
}

常用获取结果的方法

方法名 作用
forEach() 对流中的数据进行遍历
min() 传入比较器获取最小值
max() 传入比较器获取最大值
count() 计算最终结果的数量
collect() 将操作结果转换为集合

forEach:循环遍历

// forEach是最简单的循环遍历,没什么好说的
public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
list.stream()
.filter(item->item.getSalary() > 10000)
.forEach(System.out::println);
}
}

min:取最小值

public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
UserEntity entity = list.stream()
.filter(item -> item.getSalary() > 10000)
// 获取最小值需要传入Comparator比较器,直接从Comparator中取出Int类型比较器
.mim(Comparator.comparingInt(UserEntity::getSalary))
// 这里并不会直接返回实体类,需要在get一下才能获取到
.get();
System.out.println(entity);
}
}

max:取最大值

public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
UserEntity entity = list.stream()
.filter(item -> item.getSalary() > 10000)
// max同min一致
.max(Comparator.comparingInt(UserEntity::getSalary))
.get();
System.out.println(entity);
}
}

count:对结果进行计数

// 获取月薪大于10000的人数
public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
long count = list.stream()
.filter(item -> item.getSalary() > 10000)
.count();
System.out.println(count);
}
}

collect:返回操作结果

public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
List<UserEntity> collect = list.stream()
.filter(item -> item.getSalary() > 10000)
// 直接调用collect方法,然后调用toList将结果转换为List集合
.collect(Collectors.toList());
System.out.println(collect);
}
}

流式编程综合练习

// 流式编程+链式编程
public class ApplicationMain {
public static void main(String[] args) {
UserService userService = new UserService();
List<UserEntity> list = userService.selectList();
List<String> result = list.stream()
// 找到所有男性员工
.filter(item->item.getGender()=='男')
// 去除重复数据
.distinct()
// 按照年龄进行排序
.sorted(Comparator.comparingInt(UserEntity::getAge))
// 获取他们所有人的名字
.map(UserEntity::getName)
// 最后转换为List集合
.collect(Collectors.toList());
// 打印查看效果
System.out.println(result);
}
}

JDK8新特性(二) 流式编程Stream的更多相关文章

  1. 深入理解java虚拟机---jdk8新特性(二)

    1.jdk8新特性 1.新特性 2.lambda函数表达式的作用 A: 替换内部类 B:对集合的操作并行化

  2. 010-jdk1.8版本新特性二-Optional类,Stream流

    1.5.Optional类 1.定义 Optional 类是一个可以为null的容器对象.如果值存在则isPresent()方法会返回true,调用get()方法会返回该对象. Optional 是个 ...

  3. Java8 新特性之流式数据处理

    一. 流式处理简介 在我接触到java8流式处理的时候,我的第一感觉是流式处理让集合操作变得简洁了许多,通常我们需要多行代码才能完成的操作,借助于流式处理可以在一行中实现.比如我们希望对一个包含整数的 ...

  4. Java8 新特性之流式数据处理(转)

    转自:https://www.cnblogs.com/shenlanzhizun/p/6027042.html 一. 流式处理简介 在我接触到java8流式处理的时候,我的第一感觉是流式处理让集合操作 ...

  5. 类的加载、时机、反射、模板设计、jdk7/jdk8新特性(二十六)

    1.类的加载概述和加载时机 * A:类的加载概述 * 当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三步来实现对这个类进行初始化. * 加载 * 就是指将class文 ...

  6. Stream流式编程

    Stream流式编程   Stream流 说到Stream便容易想到I/O Stream,而实际上,谁规定“流”就一定是“IO流”呢?在Java 8中,得益于Lambda所带来的函数式编程,引入了一个 ...

  7. JDK8新特性关于Stream流

    在Java1.8之前还没有stream流式算法的时候,我们要是在一个放有多个User对象的list集合中,将每个User对象的主键ID取出,组合成一个新的集合,首先想到的肯定是遍历,如下: 1 2 3 ...

  8. JDK8新特性:使用stream、Comparator和Method Reference实现集合的优雅排序

    大家对java接口Comparator和Comparable都不陌生,JDK8里面Comparable还和以前一样,没有什么改动:但是Comparator在之前基础上增加了很多static和defau ...

  9. JDK8新特性一览

    转载自:http://blog.csdn.net/qiubabin/article/details/70256683 官方新特性说明地址 Jdk8新特性.png 下面对几个常用的特性做下重点说明. 一 ...

随机推荐

  1. WPF -- 一种圆形识别方案

    本文介绍一种圆形的识别方案. 识别流程 判断是否为封闭图形: 根据圆的方程,取输入点集中的1/6.3/6.5/6处的三个点,求得圆的方程,获取圆心及半径: 取点集中的部分点,计算点到圆心的距离与半径的 ...

  2. Linux流量查看工具

    目录 监控总体带宽使用 nload.bmon.slurm.bwm-ng.cbm.speedometer和netload 监控总体带宽使用(批量式输出) vnstat.ifstat.dstat和coll ...

  3. 剑指 Offer 35. 复杂链表的复制

    剑指 Offer 35. 复杂链表的复制 Offer_35 题目详情 方法一 可以使用一个HashMap来存储旧结点和新结点的映射. 这种方法需要遍历链表两遍,因为需要首先知道映射关系才能求出next ...

  4. pytorch(16)损失函数(二)

    5和6是在数据回归中用的较多的损失函数 5. nn.L1Loss 功能:计算inputs与target之差的绝对值 代码: nn.L1Loss(reduction='mean') 公式: \[l_n ...

  5. 记录自己第一次搭建本地fabric框架

    写在前,第一次搭建fabric框架,对于小白的我很是艰辛,参考了很多博主的博客才最终完成,在此记录一下搭建过程. 参考的网站 https://blog.csdn.net/smallone233/art ...

  6. CF995E Number Clicker (双向BFS)

    题目链接(洛谷) 题目大意 给定两个数 \(u\) , \(v\) .有三种操作: \(u=u+1(mod\) \(p)\) . \(u=u+p−1(mod\) \(p)\) . \(u=u^{p−2 ...

  7. Sentinel熔断降级

    sentinel流量控制 Sentinel流量控制&服务熔断降级介绍 流量控制介绍 在这里我用景区的例子解释一下 一个旅游景点日接待游客数量为8K,8K以后的游客就无法买票进去景区. 对应编程 ...

  8. [SPOJ2021] Moving Pebbles

    [SPOJ2021] Moving Pebbles 题目大意:给你\(N\)堆\(Stone\),两个人玩游戏. 每次任选一堆,首先拿掉至少一个石头,然后移动任意个石子到任意堆中. 谁不能移动了,谁就 ...

  9. 关于Python编写时候的一些数据格式调用问题

    utf-8 可变长度字符串,互联网通用,目的是减少内存占用Unicode 万国码, 对于英文多占用一个字节ASCII码 美国编码1个字节Gb2313 中国编码 编码 encode解码 decodepy ...

  10. 【java框架】SpringBoot(3) -- SpringBoot集成Swagger2

    1.SpringBoot web项目集成Swagger2 1.1.认识Swagger2 Swagger 是一个规范和完整的框架,用于生成.描述.调用和可视化 RESTful 风格的 Web 服务.总体 ...