package stream;
import model.Student;
import org.junit.jupiter.api.Test;
import java.util.*;
import java.util.stream.Collectors;
/**
* 使用Stram流操作集合信息
* @auther minghao
* @date 2019.02.11
*/
public class studyStream {
/**
*
*/
@Test
public void test3() {
//1.创建实体类准备数据
List<Student> students = new ArrayList<Student>() {
{
add(new Student(20160001, "孔明", 20, 1, "土木工程", "武汉大学"));
add(new Student(20160002, "伯约", 21, 2, "信息安全", "武汉大学"));
add(new Student(20162001, "仲谋", 19, 3, "土木工程", "浙江大学"));
add(new Student(20162001, "仲谋", 25, 3, "土木工程", "浙江大学"));
add(new Student(20162001, "仲谋", 28, 3, "土木工程", "浙江大学"));
add(new Student(20160003, "玄德", 22, 3, "经济管理", "武汉大学"));
add(new Student(20160004, "云长", 21, 2, "信息安全", "武汉大学"));
add(new Student(20161001, "翼德", 21, 2, "机械与自动化", "华中科技大学"));
add(new Student(20161002, "元直", 23, 4, "土木工程", "华中科技大学"));
add(new Student(20161003, "奉孝", 23, 4, "计算机科学", "华中科技大学"));
add(new Student(20162001, "仲谋", 22, 3, "土木工程", "浙江大学"));
add(new Student(20162002, "鲁肃", 23, 4, "计算机科学", "浙江大学"));
add(new Student(20163001, "丁奉", 24, 5, "土木工程", "南京大学"));
}
};
/**
* 方式一:前面例子中使用的方式
* 获取计算机科学专业学生的年龄总和
*/
int totalAge = students.stream().filter(student -> "计算机科学".equals(student.getMajor())).mapToInt(Student::getAge).sum();
/**
* 方式二:归约操作 reduce
* 获取计算机科学专业学生的年龄总和
*/
int totalAge2 = students.stream().filter(student -> "计算机科学".equals(student.getMajor()))
.mapToInt(Student::getAge).reduce(0,(a,b) -> a+b);
/**
* 方式三:归约操作(进一步简化) reduce
* 获取计算机科学专业学生的年龄总和
*/
int reduceAges = students.stream().filter(student -> "计算机科学".equals(student.getMajor()))
.mapToInt(Student::getAge).reduce(0,Integer::sum);
/**
* 方式四:归约操作(进一步简化) reduce
* 采用无初始值的重载版本,需要注意返回Optional
* 获取计算机科学专业学生的年龄总和
*/
Optional optAges = students.stream().filter(student -> "计算机科学".equals(student.getMajor()))
.map(Student::getAge).reduce(Integer::sum);
/**
* 求学生的总数
*/
long count = students.stream().collect(Collectors.counting());
//进一步简化
long count1 = students.stream().count();
/**
* 求年龄的最大值和最小值
*/
//求最大年龄
Optional<Student> collect = students.stream().collect(Collectors.maxBy((s1, s2) -> s1.getAge() - s2.getAge()));
//进一步简化 求最大年龄
Optional<Student> collect1 = students.stream().collect(Collectors.maxBy(Comparator.comparing(Student::getAge)));
//求最小年龄
Optional<Student> collect2 = students.stream().collect(Collectors.minBy(Comparator.comparing(Student::getAge)));
/**
* 求年龄总和
*/
Integer sunAges = students.stream().collect(Collectors.summingInt(Student::getAge));
/**
* 求年龄的平均值
*/
Double avgAge = students.stream().collect(Collectors.averagingInt(Student::getAge));
/**
* 一次性得到元素(年龄)的个数、总和、均值、最大值、最小值
*/
IntSummaryStatistics collect3 = students.stream().collect(Collectors.summarizingInt(Student::getAge));
/**
* 字符串拼接,拼接所有对象的名字
*/
String join = students.stream().map(Student::getName).collect(Collectors.joining());
String join2 = students.stream().map(Student::getName).collect(Collectors.joining(","));
/**
* 按照学校进行分组(一级分组)
*/
Map<String, List<Student>> groupbySchool = students.stream().collect(Collectors.groupingBy(Student::getSchool));
/**
* 按照学校进行分组,再按照专业分组(多级分组)
*/
Map<String,Map<String,List<Student>>> groupbySchoolAndMajor = students.stream().collect(
Collectors.groupingBy(Student::getSchool, //一级分组
Collectors.groupingBy(Student::getMajor) //二级分组(二级套在一级里面)
));
/**
* 传递一个Collector.counting,用以统计每个组的个数:
* 统计根据学校分组,每组的数量
*/
Map<String, Long> counting = students.stream().collect(Collectors.groupingBy(Student::getSchool, Collectors.counting()));
/**
* 使用分区Collectors.partitioningBy()
* 将学生分为武大学生和非武大学生
*/
Map<Boolean, List<Student>> isWhuSchool = students.stream().collect(Collectors.partitioningBy(student -> "武汉大学".equals(student.getSchool())));
System.out.println();
}
/**
*
*/
@Test
public void test2(){
//1.创建实体类准备数据
List<Student> students = new ArrayList<Student>() {
{
add(new Student(20160001, "孔明", 20, 1, "土木工程", "武汉大学"));
add(new Student(20160002, "伯约", 21, 2, "信息安全", "武汉大学"));
add(new Student(20162001, "仲谋", 19, 3, "土木工程", "浙江大学"));
add(new Student(20162001, "仲谋", 25, 3, "土木工程", "浙江大学"));
add(new Student(20162001, "仲谋", 28, 3, "土木工程", "浙江大学"));
add(new Student(20160003, "玄德", 22, 3, "经济管理", "武汉大学"));
add(new Student(20160004, "云长", 21, 2, "信息安全", "武汉大学"));
add(new Student(20161001, "翼德", 21, 2, "机械与自动化", "华中科技大学"));
add(new Student(20161002, "元直", 23, 4, "土木工程", "华中科技大学"));
add(new Student(20161003, "奉孝", 23, 4, "计算机科学", "华中科技大学"));
add(new Student(20162001, "仲谋", 22, 3, "土木工程", "浙江大学"));
add(new Student(20162002, "鲁肃", 23, 4, "计算机科学", "浙江大学"));
add(new Student(20163001, "丁奉", 24, 5, "土木工程", "南京大学"));
}
};
/**
* 2.使用stream流的filter过滤,通过lambda表达式从集合students中筛选出所有武汉大学的学生
*
* stream():将students集合转为stream流
* filter(): 中间操作 使用lambda表达式进行判断,返回boolean类型
* collect(): 终端操作
* Conllect.toList(): 将结果封装成为一个list集合
*/
List<Student> whustudents = students.stream().filter(student -> "武汉大学".equals(student.getSchool())).collect(Collectors.toList());
/**
* 3.使用stream流的filter + limit,通过lambda表达式从集合students中获取2个专业为土木工程专业的学生
*
* stream():将students集合转为stream流
* filter(): 中间操作 使用lambda表达式进行判断,返回boolean类型
* limit():限制返回的数量
* collect(): 终端操作
* Conllect.toList(): 将结果封装成为一个list集合
*
*/
List<Student> limitStudents = students.stream().filter(a -> "土木工程".equals(a.getMajor())).limit(2).collect(Collectors.toList());
/**
* 4.使用stream流的filter + limit + sorted,
* 通过lambda表达式从集合students中获取2个专业为土木工程专业的学生,
* 并按年龄从小到大排序,筛选出年龄最小的两个学生
*
* stream():将students集合转为stream流
* filter(): 中间操作 使用lambda表达式进行判断,返回boolean类型
* limit():限制返回的数量
* sorted():对两个值进行排序
* collect(): 终端操作
* Conllect.toList(): 将结果封装成为一个list集合
*
*/
List<Student> sortedStudents = students.stream().filter(student -> "土木工程".equals(student.getMajor()))
.sorted((s1,s2) -> s1.getAge() - s2.getAge()).limit(2).collect(Collectors.toList());
/**
* 5.使用stream流 filter + skip 找出排序在2之后的土木工程专业的学生
*
* stream():将students集合转为stream流
* filter(): 中间操作 使用lambda表达式进行判断,返回boolean类型
* skip():对筛选出来的结果,排除前面N个数量
* collect(): 终端操作
* Conllect.toList(): 将结果封装成为一个list集合
*/
List<Student> skipStudents = students.stream().filter(student -> "土木工程".equals(student.getMajor()))
.skip(2).collect(Collectors.toList());
/**
* 1.使用stream流 filter + map 通过map将学生实体映射成为学生姓名字符串
* 获取专业为计算机科学的所有学生姓名
*
* stream():将students集合转为stream流
* filter(): 中间操作 使用lambda表达式进行判断,返回boolean类型
* .map(Student :: getName):获取当前实体类的姓名,返回给list集合
* collect(): 终端操作
* Conllect.toList(): 将结果封装成为一个list集合
*/
List<String> names = students.stream().filter(student -> "计算机科学".equals(student.getMajor()))
.map(Student :: getName).collect(Collectors.toList());
/**
* 2.使用stream流 filter + map 通过map将学生实体映射成为学生年龄
* 获取专业为计算机科学的所有学生年龄
*
* stream():将students集合转为stream流
* filter(): 中间操作 使用lambda表达式进行判断,返回boolean类型
* .map(Student :: getAge):获取当前实体类的年龄,返回给list集合
* collect(): 终端操作
* Conllect.toList(): 将结果封装成为一个list集合
*/
List<Integer> ages = students.stream().filter(student -> "计算机科学".equals(student.getMajor()))
.map(Student :: getAge).collect(Collectors.toList());
/**
* 3.使用stream流 filter + maptoInt 通过map将学生实体映射成为学生年龄 的总和
* 获取专业为计算机科学的所有学生年龄 的总和(年龄相加)
*
* stream():将students集合转为stream流
* filter(): 中间操作 使用lambda表达式进行判断,返回boolean类型
* .map(Student :: getAge):获取当前实体类的年龄,返回给list集合
* sum(): 返回年龄的总和(年龄相加)
*/
int agesSum = students.stream().filter(student -> "计算机科学".equals(student.getMajor()))
.mapToInt(Student :: getAge).sum();
String[] strs = {"java8", "is", "easy", "to", "use"};
/**
* 将数组strs 中的每个元素使用split 拆分成为数组
* 返回list<String[]>的集合数组
* 注:distinct只有对于一个包含多个字符的流进行操作才能达到我们的目的
* 因此,下面两行代码效果一致
*/
List<String[]> distinctStrs = Arrays.stream(strs).map(str -> str.split("")).distinct().collect(Collectors.toList());
List<String[]> Strs = Arrays.stream(strs).map(str -> str.split("")).collect(Collectors.toList());
/**
* 将数组strs 中的每个元素使用split 拆分成为数组
* flatMap():将数组的多个元素转换成为一个元素进行输出
*
* flatMap与map的区别在于 flatMap是将一个流中的每个值都转成一个个流,然后再将这些流扁平化成为一个流 。
*/
List<String> flatMapStrs = Arrays.stream(strs).map(str -> str.split("")) //得到List<String[]>
.flatMap(Arrays::stream).collect(Collectors.toList()); //将List<String[]> 转化为List<String>
String a = flatMapStrs.toString();
/**
* allMatch():判断当前集合中所有人的年龄是否大于等于18岁
* true 全部大于/等于
* false 相反
*/
boolean isAdult = students.stream().allMatch(student -> student.getAge() >= 18);
/**
* anyMatch():判断当前集合中是否存在个别人的年龄是否大于等于18岁
* true 一个或多个存在
* false 一个也不存在
*/
boolean hasWhu = students.stream().anyMatch(student -> student.getAge() >= 18);
/**
* noneMatch():判断当前集合是否不存在filter指定的元素
* true 不存在
* false 存在
*/
boolean noneCs = students.stream().noneMatch(student -> "计算机科学".equals(student.getMajor()));
/**
* findFirst():返回返回满足条件的第一个元素
*/
Optional<Student> optstu = students.stream().filter(student -> "土木工程".equals(student.getMajor())).findFirst();
/**
* findAny():返回返回满足条件的任意一个元素
*/
Optional<Student> anyoptstu = students.stream().filter(student -> "土木工程".equals(student.getMajor())).findAny();
System.out.println();
}
/**
* 初步使用stream流完集合遍历
*/
@Test
public void test(){
//1.准备数据
List<Integer> nums = new ArrayList<>();
for (int i = 1; i <= 20;i++){
nums.add(i);
}
for (int i = 10; i <= 20;i++){
nums.add(i);
}
//2.使用foreach筛选偶数
List<Integer> evens = new ArrayList<>();
for (Integer num: nums) {
if (num % 2 == 0)
evens.add(num);
}
/**
* 3.1:使用stream流完成筛选偶数 返回list集合 filter() +collect()
*
* 一个流的处理分为三部分:转换成流 ——> 中间操作(自定义) ——> 终端操作
* 例如:
* stream():将集合转换成流
* filter():指定我们自定义的筛选处理/自定义其他处理
* collect():对结果进行封装处理
* Collectors.toList():指定其筛选流结果,封装成一个List集合进行返回
*/
List<Integer> StreamEvens = nums.stream().filter(num -> num % 2 == 0).collect(Collectors.toList());
System.out.println();
/**
* 3.2:使用stream流完成筛选偶数,并去掉重复 返回list集合 filter() + distinct() + collect()
*
* stream():将集合转换成流
* filter():指定我们自定义的筛选处理/自定义其他处理
* collect():对结果进行封装处理
* distinct():去掉重复
* Collectors.toList():指定其筛选流结果,封装成一个List集合进行返回
*/
List<Integer> distinctEvens = nums.stream().filter(num -> num % 2 == 0).distinct().collect(Collectors.toList());
/**
* 使用stream流+foreach遍历输出 集合中的所有数据
*/
nums.stream().forEach(num -> System.out.println(num));
System.out.println();
}
}