首先来认识一下stream
在java8中stream提供了很多Api,通过这些Api以及Lamda表达式可以进行许多高效并且遍历的操作
创建流有四种方式,具体使用如下:
@Test
/**
* 通过Collection集合创建流
*/
public void create1(){
Stream<Person> stream = personList.stream();//创建顺序流
Stream<Person> personStream = personList.parallelStream();//创建并行流
}
@Test
/**
* 通过数组创建流
*/
public void create2(){
Stream<Person> stream = Arrays.stream(personArr);
}
@Test
/**
* 通过stream中的of()方法创建流
*/
public void create3(){
Stream<Person> stream = Stream.of(new Person("emp1", "1"), new Person("emp2", "2"));
}
@Test
/**
* 无限流创建,流中的元素无限多
* 1.iterate():该方法需要传入一个初始值
* 2.generate():该方法不需要初始值
*/
public void create4(){
Stream.iterate(0,n -> n+1);//生成初始值为0,后续元素依次加一的流
Stream.generate(Math::random);//生成一个随机元素的流
}
@Test
/**
*筛选操作 filter(Predicate p)
* 需要传入一个接口,接口的返回值为boolean类型
*/
public void operate1(){
//通过filter()方法筛选列表中性别为“1”的数据
Stream<Person> stream = personList.stream().filter(e -> e.getSex().equals("1"));
}
@Test
/**
* 截断操作 limit(long maxSize)
* 需要传入一个值,截断流,使其元素不会超过给定的数量
*/
public void operate2(){
//筛选第一个元素
Stream<Person> stream = personList.stream().limit(1);
}
@Test
/**
* 跳过元素 skip(long n)
* 跳过元素,返回一个扔掉了前 n 个元素的流
* 当列表中的元素不足 n 个的时候,则返回一个空流
*/
public void operate3(){
Stream<Person> stream = personList.stream().skip(1);
}
@Test
/**
* 去重筛选 distinct()
* 通过流中元素的hashcode()和equals()方法进行去重
*/
public void operate4(){
Stream<Person> stream = personList.stream().distinct();
}
@Test
/**
* 映射操作 map(function f)
* 接收一个函数作为参数,将元素转换为其他形式或是提取信息
* 该函数会应用到每一个元素上,并将其映射成一个新的元素
*/
public void operate5(){
//筛选列表中姓名包含“2”的元素并得到他们的姓名
//先筛选,筛选后提取姓名得到新的String类型的流
Stream<String> stream = personList.stream()
.filter(e -> e.getName().contains("2"))
.map(Person::getName);
}
@Test
/**
* 排序操作 sorted() sorted(Comparator com)
* 第一种是自然排序方式
* 第二种是需要制定排序的方式
*/
public void operate6(){
//首先演示一个普通的整形列表的自然排序
List<Integer> intList = Arrays.asList(1, 2, 5, 4, 3);
intList.stream().sorted().forEach(System.out::println);
//输出结果为由小到大排序 1,2,3,4,5
//接下来演示一个针对自定义类列表的指定排序
List<Employee> employees = Arrays.asList(new Employee("员工一", 35, 5000.31),
new Employee("员工二", 30, 8000.31),
new Employee("员工三", 30, 5000.00));
//对员工列表进行排序,排序规则:根据年龄由小到大排序,再根据工资由小到大排序
employees.stream().sorted((e1,e2) ->{
int ageCompare = Integer.compare(e1.getAge(),e2.getAge());
if(ageCompare != 0){
return ageCompare;
}else{
return Double.compare(e1.getSalary(),e2.getSalary());
}
}).forEach(System.out::println);
//查询结果:
//Employee{name='员工三', age=30, salary=5000.0}
//Employee{name='员工二', age=30, salary=8000.31}
//Employee{name='员工一', age=35, salary=5000.31}
}
为什么需要终止操作呢?
因为stream是惰性的,我们前面所用到的中间操作实际上是没有进行的,只有我们进行了终止操作也就是意味着我们需要最终结果的时候,才会去执行相关的中间操作,因此,我们通过stream流操作数据时,终止操作时必不可少的。
@Test
/**
* allMatch(Predicate p)
* 检查是否匹配所有元素
*/
public void end1(){
//如果所有员工的年龄都大于30则返回true,否则返回false
boolean allMatch = employees.stream().allMatch(e -> e.getAge() > 30);
}
@Test
/**
* anyMatch(Predicate p)
* 检查是否至少匹配一个元素
*/
public void end2(){
//如果有一个员工的年龄为30岁则返回true,否则返回false
boolean anyMatch = employees.stream().anyMatch(e -> e.getAge() == 30);
}
@Test
/**
* noneMatch(Predicate p)
* 检查是否没有匹配的元素
*/
public void end3(){
//如果没有员工的年龄小于18岁则返回true,否则返回false
boolean noneMatch = employees.stream().noneMatch(e -> e.getAge() < 18);
}
@Test
/**
* findFirst --- 返回第一个元素
* findAny --- 返回当前流中的任意一个元素
*/
public void end4(){
Optional<Employee> first = employees.stream().findFirst();
Optional<Employee> any = employees.stream().findAny();
}
@Test
/**
* count() --- 返回流中元素的数量
* max(Comparator c) --- 返回流中的最大值
* min(Comparator c) --- 返回流中的最小值
*/
public void end5(){
//返回流中元素的数量
long count = employees.stream().count();
//返回年龄最大的员工
Optional<Employee> max = employees.stream()
.max((e1, e2) -> Integer.compare(e1.getAge(), e2.getAge()));
//返回年龄最小的员工
Optional<Employee> min = employees.stream()
.min((e1,e2) -> Integer.compare(e1.getAge(),e2.getAge()));
}
@Test
/**
* foreach(Consumer c)内部迭代操作
*/
public void end6(){
employees.stream().forEach(System.out :: println);
}
@Test
/**
* 规约:可以将流中元素反复结合,得到一个值
* reduce(T identity,BinaryOperator)
* 需要传入一个初始值和计算方式
* reduce(BinaryOperator)
* 需要传入计算方式
*/
public void reduce1(){
//计算一个整形列表的元素相加之和
List<Integer> list = Arrays.asList(1, 2, 3, 4, 5);
Integer reduce = list.stream().reduce(5, Integer::sum);
//计算员工列表中所有员工工资累加之和
Optional<Double> optional = employees.stream().map(e -> e.getSalary()).reduce(Double::sum);
}
@Test
/**
* 收集:将流转换为其他形式,接收一个Collector接口的实现
* Collect(Collector c)
* Collector接口中方法的实现决定了如何对流执行收集操作(例如收集到list,map等)
*/
public void collect1(){
//收集为list
List<Employee> list = employees.stream().collect(Collectors.toList());
//收集为set
Set<Employee> set = employees.stream().collect(Collectors.toSet());
}